Phase 2: auth, session management, layout, PWA manifest

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Jānis Kacēns
2026-05-11 11:54:37 +03:00
parent 0bc9160d97
commit d9de37d3d8
17 changed files with 487 additions and 1 deletions
+49
View File
@@ -0,0 +1,49 @@
package handlers
import (
"html/template"
"log/slog"
"net/http"
"path/filepath"
"qbank/internal/auth"
)
// Renderer parses and executes HTML templates from a directory.
type Renderer struct {
dir string
}
func NewRenderer(dir string) *Renderer { return &Renderer{dir: dir} }
// Render executes layout.html + <name>.html, passing data to the "layout" template.
func (r *Renderer) Render(w http.ResponseWriter, status int, name string, data any) {
t, err := template.ParseFiles(
filepath.Join(r.dir, "layout.html"),
filepath.Join(r.dir, name+".html"),
)
if err != nil {
slog.Error("parse template", "name", name, "err", err)
http.Error(w, "internal error", http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "text/html; charset=utf-8")
w.WriteHeader(status)
if err := t.ExecuteTemplate(w, "layout", data); err != nil {
slog.Error("execute template", "name", name, "err", err)
}
}
// HTTPError writes a plain-text HTTP error.
func HTTPError(w http.ResponseWriter, status int) {
http.Error(w, http.StatusText(status), status)
}
// BaseData builds the common template map (User, CSRFToken, Flash).
func BaseData(a *auth.Manager, r *http.Request) map[string]any {
return map[string]any{
"User": auth.UserFromCtx(r.Context()),
"CSRFToken": a.CSRFToken(r),
"Flash": a.PopFlash(r),
}
}