clavitor/clavitor.com/account/main.go

203 lines
5.0 KiB
Go

package main
import (
"embed"
"html/template"
"io/fs"
"log"
"net/http"
"os"
"path/filepath"
"strings"
)
//go:embed templates/*.tmpl account.css favicon.svg
var embedded embed.FS
var devMode bool
var basePath string
type PageData struct {
Page string
Title string
Desc string
ActiveNav string
Base string
Data any
}
func main() {
port := os.Getenv("PORT")
if port == "" {
port = "8098"
}
devMode = os.Getenv("DEV") == "1"
basePath = strings.TrimRight(os.Getenv("BASE_PATH"), "/")
dbPath := os.Getenv("DB_PATH")
if dbPath == "" {
dbPath = "account.db"
}
initDB(dbPath)
mux := http.NewServeMux()
// Pages
mux.HandleFunc("/", handleIndex)
mux.HandleFunc("/login", handleLogin)
mux.HandleFunc("/verify", handleVerify)
mux.HandleFunc("/dashboard", handleDashboard)
mux.HandleFunc("/checkout", handleCheckout)
mux.HandleFunc("/regions", handleRegions)
mux.HandleFunc("/settings", handleSettings)
// API
mux.HandleFunc("/api/auth/email", apiAuthEmail)
mux.HandleFunc("/api/auth/verify", apiAuthVerify)
mux.HandleFunc("/api/auth/logout", apiAuthLogout)
mux.HandleFunc("/api/checkout", apiCheckout)
mux.HandleFunc("/api/vaults", apiVaults)
mux.HandleFunc("/api/vault/create", apiVaultCreate)
mux.HandleFunc("/api/vault/", apiVaultDelete) // /api/vault/{id}/delete
mux.HandleFunc("/api/account", apiAccount)
// Static assets
mux.HandleFunc("/account.css", serveStatic("account.css", "text/css"))
mux.HandleFunc("/favicon.svg", serveStatic("favicon.svg", "image/svg+xml"))
// Strip .html extensions
handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if strings.HasSuffix(r.URL.Path, ".html") {
http.Redirect(w, r, strings.TrimSuffix(r.URL.Path, ".html"), http.StatusMovedPermanently)
return
}
mux.ServeHTTP(w, r)
})
log.Printf("clavitor account · :%s", port)
if devMode {
log.Println(" dev mode: templates reload from disk")
}
log.Fatal(http.ListenAndServe(":"+port, handler))
}
func loadTemplates(page string) (*template.Template, error) {
if devMode {
base := filepath.Join("templates", "base.tmpl")
pg := filepath.Join("templates", page+".tmpl")
return template.ParseFiles(base, pg)
}
return template.ParseFS(embedded, "templates/base.tmpl", "templates/"+page+".tmpl")
}
func render(w http.ResponseWriter, page string, data PageData) {
data.Base = basePath
tmpl, err := loadTemplates(page)
if err != nil {
log.Printf("template error: %v", err)
http.Error(w, "internal error", 500)
return
}
w.Header().Set("Content-Type", "text/html; charset=utf-8")
if err := tmpl.Execute(w, data); err != nil {
log.Printf("render error: %v", err)
}
}
func serveStatic(name, contentType string) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
var data []byte
var err error
if devMode {
data, err = os.ReadFile(name)
} else {
data, err = fs.ReadFile(embedded, name)
}
if err != nil {
http.NotFound(w, r)
return
}
w.Header().Set("Content-Type", contentType)
w.Header().Set("Cache-Control", "public, max-age=3600")
w.Write(data)
}
}
func handleIndex(w http.ResponseWriter, r *http.Request) {
if r.URL.Path != "/" {
http.NotFound(w, r)
return
}
http.Redirect(w, r, basePath+"/login", http.StatusTemporaryRedirect)
}
func handleLogin(w http.ResponseWriter, r *http.Request) {
// Already logged in? Go to dashboard
if authEmail(r) != "" {
http.Redirect(w, r, basePath+"/dashboard", http.StatusTemporaryRedirect)
return
}
render(w, "login", PageData{
Page: "login",
Title: "Sign in — clavitor",
Desc: "Sign in to your clavitor account",
})
}
func handleVerify(w http.ResponseWriter, r *http.Request) {
render(w, "verify", PageData{
Page: "verify",
Title: "Enter code — clavitor",
Desc: "Verify your login code",
Data: r.URL.Query().Get("email"),
})
}
func handleDashboard(w http.ResponseWriter, r *http.Request) {
if authEmail(r) == "" {
http.Redirect(w, r, basePath+"/login", http.StatusTemporaryRedirect)
return
}
render(w, "dashboard", PageData{
Page: "dashboard",
Title: "Dashboard — clavitor",
Desc: "Manage your vaults",
ActiveNav: "dashboard",
})
}
func handleCheckout(w http.ResponseWriter, r *http.Request) {
render(w, "checkout", PageData{
Page: "checkout",
Title: "Get started — clavitor",
Desc: "Create your clavitor account",
})
}
func handleRegions(w http.ResponseWriter, r *http.Request) {
if authEmail(r) == "" {
http.Redirect(w, r, basePath+"/login", http.StatusTemporaryRedirect)
return
}
render(w, "regions", PageData{
Page: "regions",
Title: "Choose your region — clavitor",
Desc: "Pick where your vault lives",
ActiveNav: "dashboard",
})
}
func handleSettings(w http.ResponseWriter, r *http.Request) {
if authEmail(r) == "" {
http.Redirect(w, r, basePath+"/login", http.StatusTemporaryRedirect)
return
}
render(w, "settings", PageData{
Page: "settings",
Title: "Settings — clavitor",
Desc: "Account settings",
ActiveNav: "settings",
})
}