clavitor/clavitor.com/main.go

202 lines
5.4 KiB
Go

package main
import (
"database/sql"
"embed"
"fmt"
"html/template"
"io"
"io/fs"
"log"
"net"
"net/http"
"os"
"strings"
"time"
_ "github.com/mattn/go-sqlite3"
)
//go:embed templates/*.tmpl
var tmplFS embed.FS
//go:embed *.svg *.css
var static embed.FS
var templates *template.Template
var devMode bool
var db *sql.DB
type Pop struct {
PopID int
City string
Country string
Lat float64
Lon float64
RegionName string
IP string
DNS string
Status string
Provider string
}
type PageData struct {
Page string
Title string
Desc string
ActiveNav string
Pops []Pop
}
func loadTemplates() {
if devMode {
templates = template.Must(template.ParseGlob("templates/*.tmpl"))
} else {
sub, _ := fs.Sub(tmplFS, "templates")
templates = template.Must(template.ParseFS(sub, "*.tmpl"))
}
}
func loadPops() []Pop {
rows, err := db.Query("SELECT pop_id, city, country, lat, lon, region_name, ip, dns, status, provider FROM pops ORDER BY pop_id")
if err != nil {
log.Printf("pops query error: %v", err)
return nil
}
defer rows.Close()
var pops []Pop
for rows.Next() {
var p Pop
if err := rows.Scan(&p.PopID, &p.City, &p.Country, &p.Lat, &p.Lon, &p.RegionName, &p.IP, &p.DNS, &p.Status, &p.Provider); err != nil {
log.Printf("pops scan error: %v", err)
continue
}
pops = append(pops, p)
}
return pops
}
func render(w http.ResponseWriter, data PageData) {
if devMode {
loadTemplates()
}
w.Header().Set("Content-Type", "text/html; charset=utf-8")
if err := templates.ExecuteTemplate(w, "base.tmpl", data); err != nil {
log.Printf("template error: %v", err)
http.Error(w, "Internal error", 500)
}
}
func geoHandler(w http.ResponseWriter, r *http.Request) {
ip := r.Header.Get("X-Forwarded-For")
if ip == "" {
ip = r.RemoteAddr
}
if i := strings.LastIndex(ip, ":"); i >= 0 {
ip = ip[:i]
}
ip = strings.Trim(ip, "[]")
resp, err := http.Get("https://ipapi.co/" + ip + "/json/")
if err != nil {
http.Error(w, `{"error":"geo failed"}`, 502)
return
}
defer resp.Body.Close()
w.Header().Set("Content-Type", "application/json")
io.Copy(w, resp.Body)
}
func main() {
if _, err := os.Stat("templates"); err == nil {
devMode = true
log.Println("dev mode: templates loaded from disk")
}
loadTemplates()
var err error
db, err = sql.Open("sqlite3", "clavitor.db?mode=ro")
if err != nil {
log.Fatalf("failed to open clavitor.db: %v", err)
}
defer db.Close()
port := os.Getenv("PORT")
if port == "" {
port = "8099"
}
http.HandleFunc("/geo", geoHandler)
http.HandleFunc("/ping", func(w http.ResponseWriter, r *http.Request) {
host := r.URL.Query().Get("host")
if host == "" {
http.Error(w, `{"error":"missing host"}`, 400)
return
}
start := time.Now()
conn, err := net.DialTimeout("tcp", host+":1984", 5*time.Second)
if err != nil {
w.Header().Set("Content-Type", "application/json")
w.Write([]byte(`{"error":"unreachable"}`))
return
}
conn.Close()
ms := time.Since(start).Milliseconds()
w.Header().Set("Content-Type", "application/json")
fmt.Fprintf(w, `{"ms":%d}`, ms)
})
http.HandleFunc("/hosted", func(w http.ResponseWriter, r *http.Request) {
data := PageData{Page: "hosted", Title: "clavitor — Hosted", ActiveNav: "hosted"}
data.Pops = loadPops()
render(w, data)
})
http.HandleFunc("/install", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "install", Title: "Self-host — clavitor", Desc: "Self-host clavitor in 30 seconds. One binary, no dependencies.", ActiveNav: "install"})
})
http.HandleFunc("/pricing", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "pricing", Title: "Pricing — clavitor", Desc: "Free self-hosted or $12/year hosted (launch price). No tiers, no per-seat, no contact sales.", ActiveNav: "pricing"})
})
http.HandleFunc("/privacy", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "privacy", Title: "Privacy Policy — clavitor"})
})
http.HandleFunc("/terms", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "terms", Title: "Terms of Service — clavitor"})
})
http.HandleFunc("/sources", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "sources", Title: "Sources — clavitor"})
})
http.HandleFunc("/styleguide", func(w http.ResponseWriter, r *http.Request) {
render(w, PageData{Page: "styleguide", Title: "clavitor — Styleguide"})
})
http.HandleFunc("/glass", func(w http.ResponseWriter, r *http.Request) {
data := PageData{Page: "glass", Title: "Looking Glass — clavitor"}
data.Pops = loadPops()
render(w, data)
})
// Catch-all: index page at "/" or static files or .html redirects
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
if r.URL.Path == "/" {
render(w, PageData{Page: "index", Title: "clavitor — AI-native password manager", Desc: "Field-level encryption for password managers that live alongside AI assistants. Your AI gets what it needs. Your secrets stay yours."})
return
}
// Redirect old .html URLs to clean paths
if strings.HasSuffix(r.URL.Path, ".html") {
clean := strings.TrimSuffix(r.URL.Path, ".html")
if clean == "/index" {
clean = "/"
}
http.Redirect(w, r, clean, http.StatusMovedPermanently)
return
}
http.FileServer(http.FS(static)).ServeHTTP(w, r)
})
log.Printf("clavitor-web starting on :%s", port)
if err := http.ListenAndServe(":"+port, nil); err != nil {
log.Fatal(err)
}
}