2024-02-14 17:16:41 +01:00
|
|
|
package commands
|
|
|
|
|
|
|
|
import (
|
2024-02-15 22:02:24 +01:00
|
|
|
"errors"
|
2024-02-14 17:16:41 +01:00
|
|
|
"fmt"
|
|
|
|
"io/fs"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
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"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Generate and serve the site, rebuilding when the source files change.
|
2024-02-16 16:39:19 +01:00
|
|
|
func Serve(rootDir string) error {
|
|
|
|
config, err := config.LoadDevServer(rootDir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-02-14 17:16:41 +01:00
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
if err := rebuild(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-16 16:39:19 +01:00
|
|
|
watcher, 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
|
|
|
|
fs := http.FileServer(HTMLDir{http.Dir("target/")})
|
|
|
|
http.Handle("/", http.StripPrefix("/", fs))
|
|
|
|
fmt.Println("server listening at http://localhost:4001/")
|
|
|
|
http.ListenAndServe(":4001", nil)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
func rebuild(config *config.Config) error {
|
|
|
|
|
|
|
|
site, err := site.Load(*config)
|
2024-02-15 18:41:02 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
if err := site.Build(); err != nil {
|
2024-02-15 18:41:02 +01:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-02-14 17:16:41 +01:00
|
|
|
// 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 HTMLDir struct {
|
|
|
|
d http.Dir
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d HTMLDir) 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
|
|
|
|
}
|
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
func setupWatcher(config *config.Config) (*fsnotify.Watcher, error) {
|
2024-02-14 17:16:41 +01:00
|
|
|
watcher, err := fsnotify.NewWatcher()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case event, ok := <-watcher.Events:
|
|
|
|
if !ok {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-02-15 22:02:24 +01:00
|
|
|
// some events can be received for temporary files, e.g. .#backup files generated by emacs while editing .org
|
|
|
|
// when events regarding such files arrive, discard them here instead of triggering a faulty rebuild
|
|
|
|
if _, err := os.Stat(event.Name); errors.Is(err, os.ErrNotExist) {
|
2024-02-16 00:33:19 +01:00
|
|
|
// FIXME change to log debug
|
2024-02-15 22:02:24 +01:00
|
|
|
fmt.Println("ignoring temporary file", event.Name)
|
|
|
|
continue
|
|
|
|
}
|
2024-02-14 17:16:41 +01:00
|
|
|
// chmod events are noisy, ignore them
|
2024-02-15 21:10:21 +01:00
|
|
|
if !event.Has(fsnotify.Chmod) {
|
2024-02-15 22:02:24 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Printf("\nFile %s changed, triggering rebuild.\n", event.Name)
|
|
|
|
|
|
|
|
// 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-15 22:02:24 +01:00
|
|
|
fmt.Println("error:", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-02-16 16:39:19 +01:00
|
|
|
if err := rebuild(config); err != nil {
|
2024-02-15 22:02:24 +01:00
|
|
|
fmt.Println("error:", err)
|
|
|
|
return
|
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
|
|
|
|
|
|
|
return watcher, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
}
|