1
0
mirror of https://github.com/danog/gojekyll.git synced 2024-12-02 16:37:48 +01:00
gojekyll/server/watcher.go
2017-06-24 13:30:01 -04:00

86 lines
1.7 KiB
Go

package server
import (
"log"
"path/filepath"
"time"
"github.com/fsnotify/fsnotify"
)
func (s *Server) watchFiles() error {
var (
site = s.Site
events = make(chan string)
debounced = debounce(time.Second, events)
)
watcher, err := fsnotify.NewWatcher()
if err != nil {
return err
}
go func() {
for {
select {
case event := <-watcher.Events:
events <- event.Name
case err := <-watcher.Errors:
log.Println("error:", err)
}
}
}()
go func() {
for {
filenames := <-debounced
// Resolve to URLS *before* reloading the site, in case the latter
// remaps permalinks.
urls := map[string]bool{}
for _, filename := range filenames {
relpath, err := filepath.Rel(site.Source, filename)
if err != nil {
log.Println("error:", err)
continue
}
url, found := site.RelativeFilenameToURL(relpath)
if !found {
// TODO don't warn re config and excluded files
log.Println("error:", filename, "does not match a site URL")
}
urls[url] = true
}
s.reloadSite()
for url := range urls {
s.lr.Reload(url)
}
}
}()
return watcher.Add(site.Source)
}
// debounce relays values from input to output, merging successive values within interval
// TODO consider https://github.com/ReactiveX/RxGo
func debounce(interval time.Duration, input chan string) (output chan []string) {
output = make(chan []string)
var (
pending = []string{}
ticker = time.Tick(interval) // nolint: staticcheck
)
go func() {
for {
select {
case value := <-input:
pending = append(pending, value)
case <-ticker:
if len(pending) > 0 {
output <- pending
pending = []string{}
}
}
}
}()
return
}