Compare commits

..

No commits in common. "3a0ab175ed6a91607349e304d78af8853e448a3e" and "cb45f6aa53c51824f5e829ca034d70d5204c7b37" have entirely different histories.

14 changed files with 172 additions and 343 deletions

View file

@ -3,30 +3,31 @@ package handlers
import ( import (
"encoding/json" "encoding/json"
"errors" "errors"
"github.com/gofiber/fiber/v3" "github.com/gofiber/fiber/v2"
"lmika.dev/lmika/hugo-cms/models" "lmika.dev/lmika/hugo-cms/models"
"lmika.dev/lmika/hugo-cms/services/users" "lmika.dev/lmika/hugo-cms/services/users"
"net/http"
) )
type AuthHandler struct { type AuthHandler struct {
UserService *users.Service UserService *users.Service
} }
func (h *AuthHandler) ShowLogin(c fiber.Ctx) error { func (h *AuthHandler) ShowLogin(c *fiber.Ctx) error {
return c.Render("auth/login", fiber.Map{}, "layouts/login") return c.Render("auth/login", fiber.Map{}, "layouts/login")
} }
func (h *AuthHandler) Login(c fiber.Ctx) error { func (h *AuthHandler) Login(c *fiber.Ctx) error {
var req struct { var req struct {
Email string `form:"email"` Email string `form:"email"`
Password string `form:"password"` Password string `form:"password"`
} }
if err := c.Bind().Body(&req); err != nil { if err := c.BodyParser(&req); err != nil {
return errors.New("invalid email or password") return errors.New("invalid email or password")
} }
user, err := h.UserService.VerifyLogin(c.Context(), req.Email, req.Password) user, err := h.UserService.VerifyLogin(c.UserContext(), req.Email, req.Password)
if err != nil { if err != nil {
return errors.New("invalid email or password") return errors.New("invalid email or password")
} }
@ -40,20 +41,20 @@ func (h *AuthHandler) Login(c fiber.Ctx) error {
Name: models.AuthCookieName, Name: models.AuthCookieName,
Value: string(bts), Value: string(bts),
}) })
return c.Redirect().To("/") return c.Redirect("/", http.StatusFound)
} }
func (h *AuthHandler) RequireAuth(c fiber.Ctx) error { func (h *AuthHandler) RequireAuth(c *fiber.Ctx) error {
user, err := h.readAuthCookie(c) user, err := h.readAuthCookie(c)
if err != nil { if err != nil {
return c.Redirect().To("/auth/login") return c.Redirect("/auth/login", http.StatusFound)
} }
c.Locals("user", user) c.Locals("user", user)
return c.Next() return c.Next()
} }
func (h *AuthHandler) readAuthCookie(c fiber.Ctx) (user models.User, err error) { func (h *AuthHandler) readAuthCookie(c *fiber.Ctx) (user models.User, err error) {
authData := c.Cookies(models.AuthCookieName) authData := c.Cookies(models.AuthCookieName)
if authData == "" { if authData == "" {
return models.User{}, errors.New("no auth cookie") return models.User{}, errors.New("no auth cookie")
@ -64,5 +65,5 @@ func (h *AuthHandler) readAuthCookie(c fiber.Ctx) (user models.User, err error)
return models.User{}, err return models.User{}, err
} }
return h.UserService.GetUserByID(c.Context(), ac.UserID) return h.UserService.GetUserByID(c.UserContext(), ac.UserID)
} }

View file

@ -2,26 +2,35 @@ package handlers
import ( import (
"encoding/json" "encoding/json"
"github.com/gofiber/fiber/v3" "errors"
"github.com/gofiber/fiber/v2"
"lmika.dev/lmika/hugo-cms/models" "lmika.dev/lmika/hugo-cms/models"
"log" "log"
) )
func GetUser(c fiber.Ctx) models.User { func GetUser(c *fiber.Ctx) models.User {
return fiber.Locals[models.User](c, "user") u, ok := c.Locals("user").(models.User)
if !ok {
panic(errors.New("user not found in context"))
}
return u
} }
func GetSite(c fiber.Ctx) models.Site { func GetSite(c *fiber.Ctx) models.Site {
return fiber.Locals[models.Site](c, "site") s, ok := c.Locals("site").(models.Site)
if !ok {
panic(errors.New("no site in context"))
}
return s
} }
func UpdatePrefCookie(c fiber.Ctx, update func(prefs *models.PrefCookie)) { func UpdatePrefCookie(c *fiber.Ctx, update func(prefs *models.PrefCookie)) {
cookie := GetPrefCookie(c) cookie := GetPrefCookie(c)
update(&cookie) update(&cookie)
setPrefCookie(c, cookie) setPrefCookie(c, cookie)
} }
func GetPrefCookie(c fiber.Ctx) models.PrefCookie { func GetPrefCookie(c *fiber.Ctx) models.PrefCookie {
prefCookieValue := c.Cookies(models.PrefCookieName) prefCookieValue := c.Cookies(models.PrefCookieName)
if prefCookieValue == "" { if prefCookieValue == "" {
return models.PrefCookie{} return models.PrefCookie{}
@ -36,7 +45,7 @@ func GetPrefCookie(c fiber.Ctx) models.PrefCookie {
return prefCookie return prefCookie
} }
func setPrefCookie(c fiber.Ctx, prefCookie models.PrefCookie) { func setPrefCookie(c *fiber.Ctx, prefCookie models.PrefCookie) {
if prefJson, err := json.Marshal(prefCookie); err == nil { if prefJson, err := json.Marshal(prefCookie); err == nil {
c.Cookie(&fiber.Cookie{ c.Cookie(&fiber.Cookie{
Name: models.PrefCookieName, Name: models.PrefCookieName,

View file

@ -2,16 +2,17 @@ package handlers
import ( import (
"fmt" "fmt"
"github.com/gofiber/fiber/v3" "github.com/gofiber/fiber/v2"
"net/http"
) )
type IndexHandler struct { type IndexHandler struct {
} }
func (h IndexHandler) Index(c fiber.Ctx) error { func (h IndexHandler) Index(c *fiber.Ctx) error {
prefs := GetPrefCookie(c) prefs := GetPrefCookie(c)
if prefs.SiteID > 0 { if prefs.SiteID > 0 {
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", prefs.SiteID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", prefs.SiteID), http.StatusFound)
} }
return c.Render("index", fiber.Map{}, "layouts/main") return c.Render("index", fiber.Map{}, "layouts/main")

View file

@ -1,33 +1,33 @@
package handlers package handlers
import "github.com/gofiber/fiber/v3" import "github.com/gofiber/fiber/v2"
type mimeTypeHandler interface { type mimeTypeHandler interface {
CanHandle(c fiber.Ctx) bool CanHandle(c *fiber.Ctx) bool
Handle(c fiber.Ctx) error Handle(c *fiber.Ctx) error
} }
type HTMX func(c fiber.Ctx) error type HTMX func(c *fiber.Ctx) error
func (h HTMX) CanHandle(c fiber.Ctx) bool { func (h HTMX) CanHandle(c *fiber.Ctx) bool {
return c.Get("Hx-request") == "true" return c.Get("Hx-request") == "true"
} }
func (h HTMX) Handle(c fiber.Ctx) error { func (h HTMX) Handle(c *fiber.Ctx) error {
return h(c) return h(c)
} }
type Otherwise func(c fiber.Ctx) error type Otherwise func(c *fiber.Ctx) error
func (h Otherwise) CanHandle(c fiber.Ctx) bool { func (h Otherwise) CanHandle(c *fiber.Ctx) bool {
return true return true
} }
func (h Otherwise) Handle(c fiber.Ctx) error { func (h Otherwise) Handle(c *fiber.Ctx) error {
return h(c) return h(c)
} }
func Select(c fiber.Ctx, mimeTypes ...mimeTypeHandler) error { func Select(c *fiber.Ctx, mimeTypes ...mimeTypeHandler) error {
for _, mt := range mimeTypes { for _, mt := range mimeTypes {
if mt.CanHandle(c) { if mt.CanHandle(c) {
return mt.Handle(c) return mt.Handle(c)

View file

@ -1,9 +1,8 @@
package handlers package handlers
import ( import (
"errors"
"fmt" "fmt"
"github.com/gofiber/fiber/v3" "github.com/gofiber/fiber/v2"
"lmika.dev/lmika/hugo-cms/models" "lmika.dev/lmika/hugo-cms/models"
"lmika.dev/lmika/hugo-cms/services/posts" "lmika.dev/lmika/hugo-cms/services/posts"
"net/http" "net/http"
@ -13,10 +12,10 @@ type Post struct {
Post *posts.Service Post *posts.Service
} }
func (h *Post) Posts(c fiber.Ctx) error { func (h *Post) Posts(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
posts, err := h.Post.ListPostOfSite(c.Context(), site) posts, err := h.Post.ListPostOfSite(c.UserContext(), site)
if err != nil { if err != nil {
return err return err
} }
@ -26,24 +25,24 @@ func (h *Post) Posts(c fiber.Ctx) error {
}, "layouts/site") }, "layouts/site")
} }
func (h *Post) New(c fiber.Ctx) error { func (h *Post) New(c *fiber.Ctx) error {
return c.Render("posts/new", fiber.Map{ return c.Render("posts/new", fiber.Map{
"post": models.Post{}, "post": models.Post{},
}, "layouts/site") }, "layouts/site")
} }
func (h *Post) Create(c fiber.Ctx) error { func (h *Post) Create(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
var req struct { var req struct {
Title string `json:"title" form:"title"` Title string `json:"title" form:"title"`
Body string `json:"body" form:"body"` Body string `json:"body" form:"body"`
} }
if err := c.Bind().Body(&req); err != nil { if err := c.BodyParser(&req); err != nil {
return err return err
} }
_, err := h.Post.Create(c.Context(), site, posts.NewPost{ _, err := h.Post.Create(c.UserContext(), site, posts.NewPost{
Title: req.Title, Title: req.Title,
Body: req.Body, Body: req.Body,
}) })
@ -51,18 +50,18 @@ func (h *Post) Create(c fiber.Ctx) error {
return err return err
} }
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", site.ID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", site.ID))
} }
func (h *Post) Edit(c fiber.Ctx) error { func (h *Post) Edit(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
postID := fiber.Params[int](c, "postId") postID, err := c.ParamsInt("postId")
if postID == 0 { if err != nil {
return errors.New("postId is required") return err
} }
post, err := h.Post.GetPost(c.Context(), postID) post, err := h.Post.GetPost(c.UserContext(), postID)
if err != nil { if err != nil {
return err return err
} else if post.SiteID != site.ID { } else if post.SiteID != site.ID {
@ -74,23 +73,23 @@ func (h *Post) Edit(c fiber.Ctx) error {
}, "layouts/site") }, "layouts/site")
} }
func (h *Post) Update(c fiber.Ctx) error { func (h *Post) Update(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
postID := fiber.Params[int](c, "postId") postID, err := c.ParamsInt("postId")
if postID == 0 { if err != nil {
return errors.New("postId is required") return err
} }
var req struct { var req struct {
Title string `json:"title" form:"title"` Title string `json:"title" form:"title"`
Body string `json:"body" form:"body"` Body string `json:"body" form:"body"`
} }
if err := c.Bind().Body(&req); err != nil { if err := c.BodyParser(&req); err != nil {
return err return err
} }
post, err := h.Post.GetPost(c.Context(), postID) post, err := h.Post.GetPost(c.UserContext(), postID)
if err != nil { if err != nil {
return err return err
} else if post.SiteID != site.ID { } else if post.SiteID != site.ID {
@ -100,31 +99,31 @@ func (h *Post) Update(c fiber.Ctx) error {
post.Title = req.Title post.Title = req.Title
post.Body = req.Body post.Body = req.Body
if err := h.Post.Save(c.Context(), site, &post); err != nil { if err := h.Post.Save(c.UserContext(), site, &post); err != nil {
return err return err
} }
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", site.ID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", site.ID))
} }
func (h *Post) Delete(c fiber.Ctx) error { func (h *Post) Delete(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
postID := fiber.Params[int](c, "postId") postID, err := c.ParamsInt("postId")
if postID == 0 { if err != nil {
return errors.New("postId is required") return err
} }
if err := h.Post.DeletePost(c.Context(), site, postID); err != nil { if err := h.Post.DeletePost(c.UserContext(), site, postID); err != nil {
return err return err
} }
return Select(c, return Select(c,
HTMX(func(c fiber.Ctx) error { HTMX(func(c *fiber.Ctx) error {
return c.Status(http.StatusOK).SendString("") return c.Status(http.StatusOK).SendString("")
}), }),
Otherwise(func(c fiber.Ctx) error { Otherwise(func(c *fiber.Ctx) error {
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", site.ID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", site.ID), http.StatusSeeOther)
}), }),
) )
} }

View file

@ -1,13 +1,11 @@
package handlers package handlers
import ( import (
"bufio"
"errors" "errors"
"fmt" "fmt"
"github.com/gofiber/fiber/v3" "github.com/gofiber/fiber/v2"
"github.com/jackc/pgx/v5" "github.com/jackc/pgx/v5"
"lmika.dev/lmika/hugo-cms/models" "lmika.dev/lmika/hugo-cms/models"
"lmika.dev/lmika/hugo-cms/providers/bus"
"lmika.dev/lmika/hugo-cms/services/sites" "lmika.dev/lmika/hugo-cms/services/sites"
"net/http" "net/http"
"time" "time"
@ -15,13 +13,12 @@ import (
type Site struct { type Site struct {
Site *sites.Service Site *sites.Service
Bus *bus.Bus
} }
func (s *Site) Create(c fiber.Ctx) error { func (s *Site) Create(c *fiber.Ctx) error {
user := GetUser(c) user := GetUser(c)
site, err := s.Site.CreateSite(c.Context(), user, "New Site "+time.Now().Format("2006-01-02 15:04:05")) site, err := s.Site.CreateSite(c.UserContext(), user, "New Site "+time.Now().Format("2006-01-02 15:04:05"))
if err != nil { if err != nil {
return err return err
} }
@ -30,16 +27,16 @@ func (s *Site) Create(c fiber.Ctx) error {
prefs.SiteID = site.ID prefs.SiteID = site.ID
}) })
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", site.ID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", site.ID))
} }
func (s *Site) Show(c fiber.Ctx) error { func (s *Site) Show(c *fiber.Ctx) error {
id := fiber.Params[int](c, "siteId") id, err := c.ParamsInt("siteId")
if id == 0 { if err != nil {
return errors.New("siteId is required") return err
} }
site, err := s.Site.GetSite(c.Context(), id) site, err := s.Site.GetSite(c.UserContext(), id)
if err != nil { if err != nil {
return err return err
} }
@ -49,25 +46,17 @@ func (s *Site) Show(c fiber.Ctx) error {
}, "layouts/main") }, "layouts/main")
} }
func (s *Site) Settings(c fiber.Ctx) error { func (s *Site) Settings(c *fiber.Ctx) error {
site := GetUser(c)
prodTarget, err := s.Site.GetProdTargetOfSite(c.Context(), int(site.ID))
if err != nil && !errors.Is(err, pgx.ErrNoRows) {
return err
}
return c.Render("sites/settings", fiber.Map{ return c.Render("sites/settings", fiber.Map{
"themes": s.Site.Themes(), "themes": s.Site.Themes(),
"target": prodTarget,
}, "layouts/site") }, "layouts/site")
} }
func (s *Site) SaveSettings(c fiber.Ctx) error { func (s *Site) SaveSettings(c *fiber.Ctx) error {
site := GetSite(c) site := GetSite(c)
var req sites.NewSettings var req sites.NewSettings
if err := c.Bind().Body(&req); err != nil { if err := c.BodyParser(&req); err != nil {
return err return err
} }
@ -75,60 +64,25 @@ func (s *Site) SaveSettings(c fiber.Ctx) error {
return err return err
} }
return c.Redirect().To(fmt.Sprintf("/sites/%v/settings", site.ID)) return c.Redirect(fmt.Sprintf("/sites/%v/settings", site.ID))
} }
func (s *Site) Rebuild(c fiber.Ctx) error { func (s *Site) Rebuild(c *fiber.Ctx) error {
if err := s.Site.Rebuild(c.Context(), GetSite(c)); err != nil { if err := s.Site.Rebuild(c.UserContext(), GetSite(c)); err != nil {
return err return err
} }
return c.Redirect().To(fmt.Sprintf("/sites/%v/posts", GetSite(c).ID)) return c.Redirect(fmt.Sprintf("/sites/%v/posts", GetSite(c).ID))
}
func (s *Site) SSE(c fiber.Ctx) error {
siteOfInterest := GetSite(c)
c.Set("Content-Type", "text/event-stream")
c.Set("Cache-Control", "no-cache")
c.Set("Connection", "keep-alive")
c.Set("Transfer-Encoding", "chunked")
return c.SendStreamWriter(func(w *bufio.Writer) {
sub := s.Bus.Subscribe()
defer s.Bus.Unsubscribe(sub)
for e := range sub.C {
switch e.Type {
case models.EventSiteBuildingStart:
eventSite := e.Data.(models.Site)
if eventSite.ID == siteOfInterest.ID {
fmt.Fprintf(w, "event: site-build-status\n")
fmt.Fprintf(w, "data: Building\n")
}
case models.EventSiteBuildingDone:
eventSite := e.Data.(models.Site)
if eventSite.ID == siteOfInterest.ID {
fmt.Fprintf(w, "event: site-build-status\n")
fmt.Fprintf(w, "data: \n")
}
}
if err := w.Flush(); err != nil {
break
}
}
})
} }
func (s *Site) WithSite() fiber.Handler { func (s *Site) WithSite() fiber.Handler {
return func(c fiber.Ctx) (err error) { return func(c *fiber.Ctx) (err error) {
id := fiber.Params[int](c, "siteId") id, err := c.ParamsInt("siteId")
if id == 0 { if err != nil {
return errors.New("siteId is required") return err
} }
site, err := s.Site.GetSite(c.Context(), id) site, err := s.Site.GetSite(c.UserContext(), id)
if err != nil { if err != nil {
return err return err
} }
@ -140,7 +94,7 @@ func (s *Site) WithSite() fiber.Handler {
c.Locals("site", site) c.Locals("site", site)
if prodTarget, err := s.Site.GetProdTargetOfSite(c.Context(), int(site.ID)); err == nil { if prodTarget, err := s.Site.GetProdTargetOfSite(c.UserContext(), int(site.ID)); err == nil {
c.Locals("prodTarget", prodTarget) c.Locals("prodTarget", prodTarget)
} else if !errors.Is(err, pgx.ErrNoRows) { } else if !errors.Is(err, pgx.ErrNoRows) {
return err return err

47
main.go
View file

@ -5,16 +5,15 @@ import (
"context" "context"
"flag" "flag"
"fmt" "fmt"
"github.com/gofiber/fiber/v3" "github.com/gofiber/fiber/v2"
"github.com/gofiber/fiber/v3/middleware/encryptcookie" "github.com/gofiber/fiber/v2/middleware/encryptcookie"
"github.com/gofiber/fiber/v3/middleware/static" "github.com/gofiber/fiber/v2/middleware/filesystem"
"github.com/gofiber/template/html/v2" "github.com/gofiber/template/html/v2"
"github.com/yuin/goldmark" "github.com/yuin/goldmark"
"html/template" "html/template"
"lmika.dev/lmika/hugo-cms/assets" "lmika.dev/lmika/hugo-cms/assets"
"lmika.dev/lmika/hugo-cms/config" "lmika.dev/lmika/hugo-cms/config"
"lmika.dev/lmika/hugo-cms/handlers" "lmika.dev/lmika/hugo-cms/handlers"
"lmika.dev/lmika/hugo-cms/providers/bus"
"lmika.dev/lmika/hugo-cms/providers/db" "lmika.dev/lmika/hugo-cms/providers/db"
"lmika.dev/lmika/hugo-cms/providers/git" "lmika.dev/lmika/hugo-cms/providers/git"
"lmika.dev/lmika/hugo-cms/providers/hugo" "lmika.dev/lmika/hugo-cms/providers/hugo"
@ -37,7 +36,7 @@ func main() {
flag.Parse() flag.Parse()
if *flagGenKey { if *flagGenKey {
fmt.Println(encryptcookie.GenerateKey(32)) fmt.Println(encryptcookie.GenerateKey())
return return
} }
@ -78,16 +77,15 @@ func main() {
gitProvider := git.New() gitProvider := git.New()
themesProvider := themes.New() themesProvider := themes.New()
netlifyProvider := netlify.New(cfg.NetlifyAuthToken) netlifyProvider := netlify.New(cfg.NetlifyAuthToken)
bus := bus.New()
jobService := jobs.New() jobService := jobs.New()
siteBuilderService := sitebuilder.New(dbp, themesProvider, gitProvider, hugoProvider, netlifyProvider, bus) siteBuilderService := sitebuilder.New(dbp, themesProvider, gitProvider, hugoProvider, netlifyProvider)
siteService := sites.NewService(cfg, dbp, themesProvider, siteBuilderService, jobService) siteService := sites.NewService(cfg, dbp, themesProvider, siteBuilderService, jobService)
postService := posts.New(dbp, siteBuilderService, jobService) postService := posts.New(dbp, siteBuilderService, jobService)
indexHandlers := handlers.IndexHandler{} indexHandlers := handlers.IndexHandler{}
siteHandlers := handlers.Site{Site: siteService, Bus: bus} siteHandlers := handlers.Site{Site: siteService}
postHandlers := handlers.Post{Post: postService} postHandlers := handlers.Post{Post: postService}
authHandlers := handlers.AuthHandler{UserService: userService} authHandlers := handlers.AuthHandler{UserService: userService}
@ -102,21 +100,16 @@ func main() {
if cfg.EncryptedCookieKey == "" { if cfg.EncryptedCookieKey == "" {
log.Println("No encrypt cookie key defined. Generating random key") log.Println("No encrypt cookie key defined. Generating random key")
cfg.EncryptedCookieKey = encryptcookie.GenerateKey(32) cfg.EncryptedCookieKey = encryptcookie.GenerateKey()
} }
bus.Start()
defer bus.Stop()
app := fiber.New(fiber.Config{ app := fiber.New(fiber.Config{
Views: tmplEngine, Views: tmplEngine,
PassLocalsToViews: true, PassLocalsToViews: true,
}) })
app.Use(encryptcookie.New(encryptcookie.Config{Key: cfg.EncryptedCookieKey})) app.Use(encryptcookie.New(encryptcookie.Config{Key: cfg.EncryptedCookieKey}))
app.Use("/assets", static.New("", static.Config{ app.Use("/assets", filesystem.New(filesystem.Config{Root: http.FS(assets.FS)}))
FS: assets.FS,
}))
app.Get("/auth/login", authHandlers.ShowLogin) app.Get("/auth/login", authHandlers.ShowLogin)
app.Post("/auth/login", authHandlers.Login) app.Post("/auth/login", authHandlers.Login)
@ -126,20 +119,20 @@ func main() {
app.Post("/sites", siteHandlers.Create) app.Post("/sites", siteHandlers.Create)
app.Get("/sites/:siteId", siteHandlers.Show) app.Get("/sites/:siteId", siteHandlers.Show)
sr := app.Group("/sites/:siteId") app.Route("/sites/:siteId", func(r fiber.Router) {
sr.Use(siteHandlers.WithSite()) r.Use(siteHandlers.WithSite())
sr.Post("/rebuild", siteHandlers.Rebuild) r.Post("/rebuild", siteHandlers.Rebuild)
sr.Get("/posts", postHandlers.Posts) r.Get("/posts", postHandlers.Posts)
sr.Get("/posts/new", postHandlers.New) r.Get("/posts/new", postHandlers.New)
sr.Post("/posts", postHandlers.Create) r.Post("/posts", postHandlers.Create)
sr.Get("/posts/:postId", postHandlers.Edit) r.Get("/posts/:postId", postHandlers.Edit)
sr.Post("/posts/:postId", postHandlers.Update) r.Post("/posts/:postId", postHandlers.Update)
sr.Delete("/posts/:postId", postHandlers.Delete) r.Delete("/posts/:postId", postHandlers.Delete)
sr.Get("/settings", siteHandlers.Settings) r.Get("/settings", siteHandlers.Settings)
sr.Post("/settings", siteHandlers.SaveSettings) r.Post("/settings", siteHandlers.SaveSettings)
sr.Get("/sse", siteHandlers.SSE) })
jobService.Start() jobService.Start()
defer jobService.Stop() defer jobService.Stop()

View file

@ -1,31 +0,0 @@
package models
import "container/list"
type EventType int
const (
// EventTypeSubscribe event type for the bus indicating a new subscription.
// Data is a (chan Sub) to send the new subscription
EventTypeSubscribe EventType = iota
// EventTypeUnsubscribe event type for the bus indicating to remove a subscription.
// Data is the Sub type
EventTypeUnsubscribe
// EventSiteBuildingStart indicates that the site has started being built. Data = site
EventSiteBuildingStart = 2
// EventSiteBuildingDone indicates that the site has finish building. Data = site
EventSiteBuildingDone = 3
)
type Event struct {
Type EventType
Data any
}
type Sub struct {
C chan Event
Elem *list.Element
}

View file

@ -5,3 +5,14 @@ import "context"
type Job struct { type Job struct {
Do func(ctx context.Context) error Do func(ctx context.Context) error
} }
func Jobs(jobs ...Job) Job {
return Job{Do: func(ctx context.Context) error {
for _, job := range jobs {
if err := job.Do(ctx); err != nil {
return err
}
}
return nil
}}
}

View file

@ -1,64 +0,0 @@
package bus
import (
"container/list"
"lmika.dev/lmika/hugo-cms/models"
)
type Bus struct {
subs *list.List
eventQueue chan models.Event
}
func New() *Bus {
return &Bus{
subs: list.New(),
eventQueue: make(chan models.Event, 20),
}
}
func (b *Bus) Fire(event models.Event) {
b.eventQueue <- event
}
func (b *Bus) Start() {
go func() {
for e := range b.eventQueue {
switch e.Type {
case models.EventTypeSubscribe:
retChan := e.Data.(chan *models.Sub)
newSub := &models.Sub{C: make(chan models.Event, 1)}
newSub.Elem = b.subs.PushBack(newSub)
retChan <- newSub
case models.EventTypeUnsubscribe:
sub := e.Data.(*models.Sub)
close(sub.C)
b.subs.Remove(sub.Elem)
default:
for f := b.subs.Front(); f != nil; f = f.Next() {
sub := f.Value.(*models.Sub)
select {
case sub.C <- e:
default:
}
}
}
}
}()
}
func (b *Bus) Stop() {
close(b.eventQueue)
}
func (b *Bus) Subscribe() *models.Sub {
resChan := make(chan *models.Sub)
b.eventQueue <- models.Event{Type: models.EventTypeSubscribe, Data: resChan}
return <-resChan
}
func (b *Bus) Unsubscribe(sub *models.Sub) {
b.eventQueue <- models.Event{Type: models.EventTypeUnsubscribe, Data: sub}
}

View file

@ -15,14 +15,11 @@ import (
func (s *Service) WritePost(site models.Site, post models.Post) models.Job { func (s *Service) WritePost(site models.Site, post models.Post) models.Job {
return models.Job{ return models.Job{
Do: func(ctx context.Context) error { Do: func(ctx context.Context) error {
s.signalSiteBuildingStarted(ctx, site)
defer s.signalSiteBuildingFinished(ctx, site)
rbn, err := s.fullRebuildNecessary(ctx, site) rbn, err := s.fullRebuildNecessary(ctx, site)
if err != nil { if err != nil {
return err return err
} else if rbn { } else if rbn {
return s.rebuildSite(ctx, site, site) return s.RebuildSite(site, site).Do(ctx)
} }
if err := s.writePost(site, post); err != nil { if err := s.writePost(site, post); err != nil {
@ -36,64 +33,53 @@ func (s *Service) WritePost(site models.Site, post models.Post) models.Job {
func (s *Service) WriteAllPosts(site models.Site) models.Job { func (s *Service) WriteAllPosts(site models.Site) models.Job {
return models.Job{ return models.Job{
Do: func(ctx context.Context) error { Do: func(ctx context.Context) error {
s.signalSiteBuildingStarted(ctx, site) var startId int64
defer s.signalSiteBuildingFinished(ctx, site) now := time.Now()
for {
posts, err := s.db.ListPublishablePosts(ctx, int64(startId), site.ID, now)
if err != nil {
return err
} else if len(posts) == 0 {
return nil
}
if err := s.writeAllPosts(ctx, site); err != nil { for _, post := range posts {
return err if err := s.writePost(site, post); err != nil {
return err
}
}
startId = posts[len(posts)-1].ID
} }
return s.publish(ctx, site)
}, },
} }
} }
func (s *Service) DeletePost(site models.Site, post models.Post) models.Job { func (s *Service) DeletePost(site models.Site, post models.Post) models.Job {
return models.Job{ return models.Jobs(
Do: func(ctx context.Context) error { models.Job{
s.signalSiteBuildingStarted(ctx, site) Do: func(ctx context.Context) error {
defer s.signalSiteBuildingFinished(ctx, site) themeMeta, ok := s.themes.Lookup(site.Theme)
if !ok {
return errors.New("theme not found")
}
themeMeta, ok := s.themes.Lookup(site.Theme) postFilename := s.postFilename(site, themeMeta, post)
if !ok {
return errors.New("theme not found")
}
postFilename := s.postFilename(site, themeMeta, post) if _, err := os.Stat(postFilename); err != nil {
if errors.Is(err, os.ErrNotExist) {
return nil
}
return err
}
if _, err := os.Stat(postFilename); err != nil { if os.Remove(postFilename) != nil {
if errors.Is(err, os.ErrNotExist) {
return nil return nil
} }
return err
}
if os.Remove(postFilename) != nil {
return nil return nil
} },
return s.publish(ctx, site)
}, },
} s.Publish(site),
} )
func (s *Service) writeAllPosts(ctx context.Context, site models.Site) error {
var startId int64
now := time.Now()
for {
posts, err := s.db.ListPublishablePosts(ctx, int64(startId), site.ID, now)
if err != nil {
return err
} else if len(posts) == 0 {
return nil
}
for _, post := range posts {
if err := s.writePost(site, post); err != nil {
return err
}
}
startId = posts[len(posts)-1].ID
}
} }
func (s *Service) writePost(site models.Site, post models.Post) error { func (s *Service) writePost(site models.Site, post models.Post) error {

View file

@ -8,9 +8,6 @@ import (
func (s *Service) Publish(site models.Site) models.Job { func (s *Service) Publish(site models.Site) models.Job {
return models.Job{ return models.Job{
Do: func(ctx context.Context) error { Do: func(ctx context.Context) error {
s.signalSiteBuildingStarted(ctx, site)
defer s.signalSiteBuildingFinished(ctx, site)
return s.publish(ctx, site) return s.publish(ctx, site)
}, },
} }

View file

@ -4,7 +4,6 @@ import (
"context" "context"
"errors" "errors"
"lmika.dev/lmika/hugo-cms/models" "lmika.dev/lmika/hugo-cms/models"
"lmika.dev/lmika/hugo-cms/providers/bus"
"lmika.dev/lmika/hugo-cms/providers/db" "lmika.dev/lmika/hugo-cms/providers/db"
"lmika.dev/lmika/hugo-cms/providers/git" "lmika.dev/lmika/hugo-cms/providers/git"
"lmika.dev/lmika/hugo-cms/providers/hugo" "lmika.dev/lmika/hugo-cms/providers/hugo"
@ -21,7 +20,6 @@ type Service struct {
git *git.Provider git *git.Provider
hugo *hugo.Provider hugo *hugo.Provider
netlify *netlify.Provider netlify *netlify.Provider
bus *bus.Bus
} }
func New( func New(
@ -30,7 +28,6 @@ func New(
git *git.Provider, git *git.Provider,
hugo *hugo.Provider, hugo *hugo.Provider,
netlify *netlify.Provider, netlify *netlify.Provider,
bus *bus.Bus,
) *Service { ) *Service {
return &Service{ return &Service{
db: db, db: db,
@ -38,48 +35,33 @@ func New(
git: git, git: git,
hugo: hugo, hugo: hugo,
netlify: netlify, netlify: netlify,
bus: bus,
} }
} }
func (s *Service) CreateNewSite(site models.Site) models.Job { func (s *Service) CreateNewSite(site models.Site) models.Job {
return models.Job{ return models.Job{
Do: func(ctx context.Context) error { Do: func(ctx context.Context) error {
s.signalSiteBuildingStarted(ctx, site)
defer s.signalSiteBuildingFinished(ctx, site)
return s.createSite(ctx, site) return s.createSite(ctx, site)
}, },
} }
} }
func (s *Service) RebuildSite(oldSite, newSite models.Site) models.Job { func (s *Service) RebuildSite(oldSite, newSite models.Site) models.Job {
return models.Job{ return models.Jobs(
Do: func(ctx context.Context) error { models.Job{
s.signalSiteBuildingStarted(ctx, newSite) Do: func(ctx context.Context) error {
defer s.signalSiteBuildingFinished(ctx, newSite) // Teardown the existing site
siteDir := s.hugo.SiteStagingDir(oldSite, hugo.BaseSiteDir)
return s.rebuildSite(ctx, oldSite, newSite) if err := os.RemoveAll(siteDir); err != nil {
return err
}
return nil
},
}, },
} s.CreateNewSite(newSite),
} s.WriteAllPosts(newSite),
s.Publish(newSite),
func (s *Service) rebuildSite(ctx context.Context, oldSite, newSite models.Site) error { )
// Teardown the existing site
siteDir := s.hugo.SiteStagingDir(oldSite, hugo.BaseSiteDir)
if err := os.RemoveAll(siteDir); err != nil {
return err
}
if err := s.createSite(ctx, newSite); err != nil {
return err
}
if err := s.writeAllPosts(ctx, newSite); err != nil {
return err
}
return s.publish(ctx, newSite)
} }
func (s *Service) fullRebuildNecessary(ctx context.Context, site models.Site) (bool, error) { func (s *Service) fullRebuildNecessary(ctx context.Context, site models.Site) (bool, error) {
@ -140,11 +122,3 @@ func (s *Service) createSite(ctx context.Context, site models.Site) error {
} }
return nil return nil
} }
func (s *Service) signalSiteBuildingStarted(ctx context.Context, site models.Site) {
s.bus.Fire(models.Event{Type: models.EventSiteBuildingStart, Data: site})
}
func (s *Service) signalSiteBuildingFinished(ctx context.Context, site models.Site) {
s.bus.Fire(models.Event{Type: models.EventSiteBuildingDone, Data: site})
}

View file

@ -7,7 +7,6 @@
<link rel="stylesheet" href="/assets/css/main.css"> <link rel="stylesheet" href="/assets/css/main.css">
<title>Hugo CMS</title> <title>Hugo CMS</title>
<script src="https://unpkg.com/htmx.org@2.0.4"></script> <script src="https://unpkg.com/htmx.org@2.0.4"></script>
<script src="https://unpkg.com/htmx-ext-sse@2.2.2/sse.js"></script>
</head> </head>
<body class="role-site"> <body class="role-site">
<header> <header>