2024-02-14 17:16:41 +01:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/fs"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-02-19 22:03:06 +01:00
|
|
|
"sync/atomic"
|
2024-02-14 17:16:41 +01:00
|
|
|
|
2024-02-16 19:29:43 +01:00
|
|
|
"github.com/facundoolano/jorge/config"
|
|
|
|
"github.com/facundoolano/jorge/site"
|
2024-02-14 17:16:41 +01:00
|
|
|
"github.com/fsnotify/fsnotify"
|
|
|
|
)
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
// Generate and serve the site, rebuilding when the source files change
|
|
|
|
// and triggering a page refresh on clients browsing it.
|
2024-02-16 16:39:19 +01:00
|
|
|
func Serve(rootDir string) error {
|
2024-02-19 22:03:06 +01:00
|
|
|
config, err := config.LoadDev(rootDir)
|
2024-02-16 16:39:19 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-14 17:16:41 +01:00
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
if err := rebuildSite(config); err != nil {
|
2024-02-14 17:16:41 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// watch for changes in src and layouts, and trigger a rebuild
|
2024-02-19 22:03:06 +01:00
|
|
|
watcher, broker, err := setupWatcher(config)
|
2024-02-14 17:16:41 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer watcher.Close()
|
|
|
|
|
|
|
|
// serve the target dir with a file server
|
2024-02-19 22:03:06 +01:00
|
|
|
fs := http.FileServer(HTMLFileSystem{http.Dir(config.TargetDir)})
|
2024-02-14 17:16:41 +01:00
|
|
|
http.Handle("/", http.StripPrefix("/", fs))
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
if config.LiveReload {
|
|
|
|
// handle client requests to listen to server-sent events
|
|
|
|
http.Handle("/_events/", makeServerEventsHandler(broker))
|
|
|
|
}
|
|
|
|
|
2024-02-17 21:09:19 +01:00
|
|
|
addr := fmt.Sprintf("%s:%d", config.ServerHost, config.ServerPort)
|
|
|
|
fmt.Printf("server listening at http://%s\n", addr)
|
|
|
|
return http.ListenAndServe(addr, nil)
|
2024-02-14 17:16:41 +01:00
|
|
|
}
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
// Return an http.HandlerFunc that establishes a server-sent event stream with clients,
|
|
|
|
// subscribes to site rebuild events received through the given event broker
|
|
|
|
// and forwards them to the client.
|
|
|
|
func makeServerEventsHandler(broker *EventBroker) http.HandlerFunc {
|
|
|
|
return func(res http.ResponseWriter, req *http.Request) {
|
|
|
|
res.Header().Set("Content-Type", "text/event-stream")
|
|
|
|
res.Header().Set("Connection", "keep-alive")
|
|
|
|
res.Header().Set("Cache-Control", "no-cache")
|
|
|
|
res.Header().Set("Access-Control-Allow-Origin", "*")
|
|
|
|
|
|
|
|
id, events := broker.subscribe()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-events:
|
|
|
|
// send an event to the connected client.
|
|
|
|
// data\n\n just means send an empty, unnamed event
|
|
|
|
// since we only need to support the single reload operation.
|
|
|
|
fmt.Fprint(res, "data\n\n")
|
|
|
|
res.(http.Flusher).Flush()
|
|
|
|
case <-req.Context().Done():
|
|
|
|
broker.unsubscribe(id)
|
|
|
|
return
|
|
|
|
}
|
2024-02-14 17:16:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
// Sets up a watcher that will publish changes in the site source files
|
|
|
|
// to the returned event broker.
|
|
|
|
func setupWatcher(config *config.Config) (*fsnotify.Watcher, *EventBroker, error) {
|
2024-02-14 17:16:41 +01:00
|
|
|
watcher, err := fsnotify.NewWatcher()
|
|
|
|
if err != nil {
|
2024-02-19 22:03:06 +01:00
|
|
|
return nil, nil, err
|
2024-02-14 17:16:41 +01:00
|
|
|
}
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
broker := newEventBroker()
|
|
|
|
|
2024-02-14 17:16:41 +01:00
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case event, ok := <-watcher.Events:
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-02-18 01:13:50 +01:00
|
|
|
// chmod events are noisy, ignore them. also skip create events
|
|
|
|
// which we assume meaningless until the write that comes next
|
|
|
|
if event.Has(fsnotify.Chmod) || event.Has(fsnotify.Create) {
|
2024-02-15 22:02:24 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
fmt.Printf("\nFile %s changed, rebuilding site.\n", event.Name)
|
2024-02-15 22:02:24 +01:00
|
|
|
|
|
|
|
// since new nested directories could be triggering this change, and we need to watch those too
|
|
|
|
// and since re-watching files is a noop, I just re-add the entire src everytime there's a change
|
2024-02-16 16:39:19 +01:00
|
|
|
if err := addAll(watcher, config); err != nil {
|
2024-02-18 01:13:50 +01:00
|
|
|
fmt.Println("couldn't add watchers:", err)
|
|
|
|
continue
|
2024-02-15 22:02:24 +01:00
|
|
|
}
|
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
if err := rebuildSite(config); err != nil {
|
2024-02-18 01:13:50 +01:00
|
|
|
fmt.Println("build error:", err)
|
|
|
|
continue
|
2024-02-14 17:16:41 +01:00
|
|
|
}
|
2024-02-19 22:03:06 +01:00
|
|
|
broker.publish("rebuild")
|
2024-02-14 17:16:41 +01:00
|
|
|
|
2024-02-18 01:13:50 +01:00
|
|
|
fmt.Println("done\nserver listening at", config.SiteUrl)
|
|
|
|
|
2024-02-14 17:16:41 +01:00
|
|
|
case err, ok := <-watcher.Errors:
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
fmt.Println("error:", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
err = addAll(watcher, config)
|
2024-02-14 17:16:41 +01:00
|
|
|
|
2024-02-19 22:03:06 +01:00
|
|
|
return watcher, broker, err
|
2024-02-14 17:16:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Add the layouts and all source directories to the given watcher
|
2024-02-16 16:39:19 +01:00
|
|
|
func addAll(watcher *fsnotify.Watcher, config *config.Config) error {
|
|
|
|
err := watcher.Add(config.LayoutsDir)
|
|
|
|
err = watcher.Add(config.DataDir)
|
|
|
|
err = watcher.Add(config.IncludesDir)
|
2024-02-14 17:16:41 +01:00
|
|
|
// fsnotify watches all files within a dir, but non recursively
|
|
|
|
// this walks through the src dir and adds watches for each found directory
|
2024-02-16 16:39:19 +01:00
|
|
|
filepath.WalkDir(config.SrcDir, func(path string, entry fs.DirEntry, err error) error {
|
2024-02-14 17:16:41 +01:00
|
|
|
if entry.IsDir() {
|
|
|
|
watcher.Add(path)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
}
|
2024-02-19 22:03:06 +01:00
|
|
|
|
|
|
|
func rebuildSite(config *config.Config) error {
|
|
|
|
site, err := site.Load(*config)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := site.Build(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Tweaks the http file system to construct a server that hides the .html suffix from requests.
|
|
|
|
// Based on https://stackoverflow.com/a/57281956/993769
|
|
|
|
type HTMLFileSystem struct {
|
|
|
|
d http.Dir
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d HTMLFileSystem) Open(name string) (http.File, error) {
|
|
|
|
// Try name as supplied
|
|
|
|
f, err := d.d.Open(name)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
// Not found, try with .html
|
|
|
|
if f, err := d.d.Open(name + ".html"); err == nil {
|
|
|
|
return f, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return f, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// The event broker allows the file watcher to publish site rebuild events
|
|
|
|
// and register http clients to listen for them, in order to trigger browser refresh
|
|
|
|
// events after the the site has been rebuilt.
|
|
|
|
type EventBroker struct {
|
|
|
|
inEvents chan string
|
|
|
|
inSubscriptions chan Subscription
|
|
|
|
subscribers map[uint64]chan string
|
|
|
|
idgen atomic.Uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
type Subscription struct {
|
|
|
|
id uint64
|
|
|
|
outEvents chan string
|
|
|
|
}
|
|
|
|
|
|
|
|
func newEventBroker() *EventBroker {
|
|
|
|
broker := EventBroker{
|
|
|
|
inEvents: make(chan string),
|
|
|
|
inSubscriptions: make(chan Subscription),
|
|
|
|
subscribers: map[uint64]chan string{},
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case msg := <-broker.inSubscriptions:
|
|
|
|
if msg.outEvents != nil {
|
|
|
|
// subscribe
|
|
|
|
broker.subscribers[msg.id] = msg.outEvents
|
|
|
|
} else {
|
|
|
|
// unsubscribe
|
|
|
|
close(broker.subscribers[msg.id])
|
|
|
|
delete(broker.subscribers, msg.id)
|
|
|
|
}
|
|
|
|
case msg := <-broker.inEvents:
|
|
|
|
// send the event to all the subscribers
|
|
|
|
for _, outEvents := range broker.subscribers {
|
|
|
|
outEvents <- msg
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return &broker
|
|
|
|
}
|
|
|
|
|
|
|
|
// Adds a subscription to this broker events, returning a subscriber id
|
|
|
|
// (useful for unsubscribing later) and a channel where events will be delivered.
|
|
|
|
func (broker *EventBroker) subscribe() (uint64, <-chan string) {
|
|
|
|
id := broker.idgen.Add(1)
|
|
|
|
outEvents := make(chan string)
|
|
|
|
broker.inSubscriptions <- Subscription{id, outEvents}
|
|
|
|
return id, outEvents
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the subscriber with the given id from the broker,
|
|
|
|
// closing its associated channel.
|
|
|
|
func (broker *EventBroker) unsubscribe(id uint64) {
|
|
|
|
broker.inSubscriptions <- Subscription{id: id, outEvents: nil}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Publish an event to all the broker subscribers.
|
|
|
|
func (broker *EventBroker) publish(event string) {
|
|
|
|
broker.inEvents <- event
|
|
|
|
}
|