mirror of
https://github.com/caddyserver/caddy.git
synced 2025-01-19 09:05:41 +03:00
Merge pull request #213 from abiosoft/master
markdown: Watch for file changes for links. Removed sitegen requirement for link index.
This commit is contained in:
commit
9669363504
8 changed files with 202 additions and 26 deletions
|
@ -29,13 +29,20 @@ func Markdown(c *Controller) (middleware.Middleware, error) {
|
|||
|
||||
// For any configs that enabled static site gen, sweep the whole path at startup
|
||||
c.Startup = append(c.Startup, func() error {
|
||||
for _, cfg := range mdconfigs {
|
||||
if cfg.StaticDir == "" {
|
||||
continue
|
||||
for i := range mdconfigs {
|
||||
cfg := &mdconfigs[i]
|
||||
|
||||
// Links generation.
|
||||
if err := markdown.GenerateLinks(md, cfg); err != nil {
|
||||
return err
|
||||
}
|
||||
// Watch file changes for links generation if not in development mode.
|
||||
if !cfg.Development {
|
||||
markdown.Watch(md, cfg, markdown.DefaultInterval)
|
||||
}
|
||||
|
||||
if err := markdown.GenerateLinks(md, &cfg); err != nil {
|
||||
return err
|
||||
if cfg.StaticDir == "" {
|
||||
continue
|
||||
}
|
||||
|
||||
// If generated site already exists, clear it out
|
||||
|
@ -68,7 +75,7 @@ func Markdown(c *Controller) (middleware.Middleware, error) {
|
|||
|
||||
// Generate the static file
|
||||
ctx := middleware.Context{Root: md.FileSys}
|
||||
_, err = md.Process(cfg, reqPath, body, ctx)
|
||||
_, err = md.Process(*cfg, reqPath, body, ctx)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -155,6 +162,16 @@ func markdownParse(c *Controller) ([]markdown.Config, error) {
|
|||
// only 1 argument allowed
|
||||
return mdconfigs, c.ArgErr()
|
||||
}
|
||||
case "dev":
|
||||
if c.NextArg() {
|
||||
md.Development = strings.ToLower(c.Val()) == "true"
|
||||
} else {
|
||||
md.Development = true
|
||||
}
|
||||
if c.NextArg() {
|
||||
// only 1 argument allowed
|
||||
return mdconfigs, c.ArgErr()
|
||||
}
|
||||
default:
|
||||
return mdconfigs, c.Err("Expected valid markdown configuration property")
|
||||
}
|
||||
|
|
|
@ -69,12 +69,29 @@ type Config struct {
|
|||
// Links to all markdown pages ordered by date.
|
||||
Links []PageLink
|
||||
|
||||
// Stores a directory hash to check for changes.
|
||||
linksHash string
|
||||
|
||||
// Directory to store static files
|
||||
StaticDir string
|
||||
|
||||
// If in development mode. i.e. Actively editing markdown files.
|
||||
Development bool
|
||||
|
||||
sync.RWMutex
|
||||
}
|
||||
|
||||
// IsValidExt checks to see if an extension is a valid markdown extension
|
||||
// for config.
|
||||
func (c Config) IsValidExt(ext string) bool {
|
||||
for _, e := range c.Extensions {
|
||||
if e == ext {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
// ServeHTTP implements the http.Handler interface.
|
||||
func (md Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error) {
|
||||
for i := range md.Configs {
|
||||
|
@ -103,6 +120,13 @@ func (md Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error
|
|||
return http.StatusNotFound, nil
|
||||
}
|
||||
|
||||
// if development is set, scan directory for file changes for links.
|
||||
if m.Development {
|
||||
if err := GenerateLinks(md, m); err != nil {
|
||||
log.Println(err)
|
||||
}
|
||||
}
|
||||
|
||||
// if static site is generated, attempt to use it
|
||||
if filepath, ok := m.StaticFiles[fpath]; ok {
|
||||
if fs1, err := os.Stat(filepath); err == nil {
|
||||
|
@ -122,13 +146,6 @@ func (md Markdown) ServeHTTP(w http.ResponseWriter, r *http.Request) (int, error
|
|||
}
|
||||
}
|
||||
|
||||
if m.StaticDir != "" {
|
||||
// Markdown modified or new. Update links.
|
||||
if err := GenerateLinks(md, m); err != nil {
|
||||
log.Println(err)
|
||||
}
|
||||
}
|
||||
|
||||
body, err := ioutil.ReadAll(f)
|
||||
if err != nil {
|
||||
return http.StatusInternalServerError, err
|
||||
|
|
|
@ -1,6 +1,7 @@
|
|||
package markdown
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"log"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
|
@ -102,7 +103,7 @@ func getTrue() bool {
|
|||
</body>
|
||||
</html>
|
||||
`
|
||||
if respBody != expectedBody {
|
||||
if !equalStrings(respBody, expectedBody) {
|
||||
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
||||
}
|
||||
|
||||
|
@ -143,10 +144,7 @@ func getTrue() bool {
|
|||
</body>
|
||||
</html>`
|
||||
|
||||
replacer := strings.NewReplacer("\r", "", "\n", "")
|
||||
respBody = replacer.Replace(respBody)
|
||||
expectedBody = replacer.Replace(expectedBody)
|
||||
if respBody != expectedBody {
|
||||
if !equalStrings(respBody, expectedBody) {
|
||||
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
||||
}
|
||||
|
||||
|
@ -177,26 +175,31 @@ func getTrue() bool {
|
|||
|
||||
</body>
|
||||
</html>`
|
||||
respBody = replacer.Replace(respBody)
|
||||
expectedBody = replacer.Replace(expectedBody)
|
||||
if respBody != expectedBody {
|
||||
|
||||
if !equalStrings(respBody, expectedBody) {
|
||||
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
||||
}
|
||||
|
||||
expectedLinks := []string{
|
||||
"/blog/test.md",
|
||||
"/log/test.md",
|
||||
"/og/first.md",
|
||||
}
|
||||
|
||||
for i, c := range md.Configs {
|
||||
for i := range md.Configs {
|
||||
c := &md.Configs[i]
|
||||
if err := GenerateLinks(md, c); err != nil {
|
||||
t.Fatalf("Error: %v", err)
|
||||
}
|
||||
}
|
||||
|
||||
for i, c := range md.Configs[:2] {
|
||||
log.Printf("Test number: %d, configuration links: %v, config: %v", i, c.Links, c)
|
||||
if c.Links[0].URL != expectedLinks[i] {
|
||||
t.Fatalf("Expected %v got %v", expectedLinks[i], c.Links[0].URL)
|
||||
}
|
||||
}
|
||||
|
||||
// attempt to trigger race condition
|
||||
// attempt to trigger race conditions
|
||||
var w sync.WaitGroup
|
||||
f := func() {
|
||||
req, err := http.NewRequest("GET", "/log/test.md", nil)
|
||||
|
@ -214,8 +217,32 @@ func getTrue() bool {
|
|||
}
|
||||
w.Wait()
|
||||
|
||||
f = func() {
|
||||
GenerateLinks(md, &md.Configs[0])
|
||||
w.Done()
|
||||
}
|
||||
for i := 0; i < 5; i++ {
|
||||
w.Add(1)
|
||||
go f()
|
||||
}
|
||||
w.Wait()
|
||||
|
||||
if err = os.RemoveAll(DefaultStaticDir); err != nil {
|
||||
t.Errorf("Error while removing the generated static files: %v", err)
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
func equalStrings(s1, s2 string) bool {
|
||||
s1 = strings.TrimSpace(s1)
|
||||
s2 = strings.TrimSpace(s2)
|
||||
in := bufio.NewScanner(strings.NewReader(s1))
|
||||
for in.Scan() {
|
||||
txt := strings.TrimSpace(in.Text())
|
||||
if !strings.HasPrefix(strings.TrimSpace(s2), txt) {
|
||||
return false
|
||||
}
|
||||
s2 = strings.Replace(s2, txt, "", 1)
|
||||
}
|
||||
return true
|
||||
}
|
||||
|
|
|
@ -2,7 +2,11 @@ package markdown
|
|||
|
||||
import (
|
||||
"bytes"
|
||||
"crypto/sha1"
|
||||
"encoding/hex"
|
||||
"fmt"
|
||||
"io/ioutil"
|
||||
"log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"sort"
|
||||
|
@ -75,10 +79,23 @@ func (l *linkGen) generateLinks(md Markdown, cfg *Config) {
|
|||
if _, err := os.Stat(fp); os.IsNotExist(err) {
|
||||
l.Lock()
|
||||
l.lastErr = err
|
||||
l.generating = false
|
||||
l.Unlock()
|
||||
return
|
||||
}
|
||||
|
||||
hash, err := computeDirHash(md, *cfg)
|
||||
|
||||
// same hash, return.
|
||||
if err == nil && hash == cfg.linksHash {
|
||||
l.Lock()
|
||||
l.generating = false
|
||||
l.Unlock()
|
||||
return
|
||||
} else if err != nil {
|
||||
log.Println("Error:", err)
|
||||
}
|
||||
|
||||
cfg.Links = []PageLink{}
|
||||
|
||||
cfg.Lock()
|
||||
|
@ -138,6 +155,8 @@ func (l *linkGen) generateLinks(md Markdown, cfg *Config) {
|
|||
|
||||
// sort by newest date
|
||||
sort.Sort(byDate(cfg.Links))
|
||||
|
||||
cfg.linksHash = hash
|
||||
cfg.Unlock()
|
||||
|
||||
l.Lock()
|
||||
|
@ -176,3 +195,25 @@ func GenerateLinks(md Markdown, cfg *Config) error {
|
|||
g.discardWaiters()
|
||||
return g.lastErr
|
||||
}
|
||||
|
||||
// computeDirHash computes an hash on static directory of c.
|
||||
func computeDirHash(md Markdown, c Config) (string, error) {
|
||||
dir := filepath.Join(md.Root, c.PathScope)
|
||||
if _, err := os.Stat(dir); err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
hashString := ""
|
||||
err := filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||
if !info.IsDir() && c.IsValidExt(filepath.Ext(path)) {
|
||||
hashString += fmt.Sprintf("%v%v%v%v", info.ModTime(), info.Name(), info.Size(), path)
|
||||
}
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
sum := sha1.Sum([]byte(hashString))
|
||||
return hex.EncodeToString(sum[:]), nil
|
||||
}
|
||||
|
|
4
middleware/markdown/testdata/og/first.md
vendored
4
middleware/markdown/testdata/og/first.md
vendored
|
@ -1 +1,5 @@
|
|||
---
|
||||
title: first_post
|
||||
sitename: title
|
||||
---
|
||||
# Test h1
|
||||
|
|
|
@ -1,7 +1,8 @@
|
|||
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<title>first_post</title>
|
||||
<title>first_post</title>
|
||||
</head>
|
||||
<body>
|
||||
<h1>Header title</h1>
|
||||
|
|
35
middleware/markdown/watcher.go
Normal file
35
middleware/markdown/watcher.go
Normal file
|
@ -0,0 +1,35 @@
|
|||
package markdown
|
||||
|
||||
import "time"
|
||||
|
||||
const DefaultInterval = time.Second * 60
|
||||
|
||||
// Watch monitors the configured markdown directory for changes. It calls GenerateLinks
|
||||
// when there are changes.
|
||||
func Watch(md Markdown, c *Config, interval time.Duration) (stopChan chan struct{}) {
|
||||
return TickerFunc(interval, func() {
|
||||
GenerateLinks(md, c)
|
||||
})
|
||||
}
|
||||
|
||||
// TickerFunc runs f at interval. If interval is <= 0, it loops f. A message to the
|
||||
// returned channel will stop the executing goroutine.
|
||||
func TickerFunc(interval time.Duration, f func()) chan struct{} {
|
||||
stopChan := make(chan struct{})
|
||||
|
||||
ticker := time.NewTicker(interval)
|
||||
go func() {
|
||||
loop:
|
||||
for {
|
||||
select {
|
||||
case <-ticker.C:
|
||||
f()
|
||||
case <-stopChan:
|
||||
ticker.Stop()
|
||||
break loop
|
||||
}
|
||||
}
|
||||
}()
|
||||
|
||||
return stopChan
|
||||
}
|
34
middleware/markdown/watcher_test.go
Normal file
34
middleware/markdown/watcher_test.go
Normal file
|
@ -0,0 +1,34 @@
|
|||
package markdown
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
func TestWatcher(t *testing.T) {
|
||||
expected := "12345678"
|
||||
interval := time.Millisecond * 100
|
||||
i := 0
|
||||
out := ""
|
||||
stopChan := TickerFunc(interval, func() {
|
||||
i++
|
||||
out += fmt.Sprint(i)
|
||||
})
|
||||
time.Sleep(interval * 8)
|
||||
stopChan <- struct{}{}
|
||||
if expected != out {
|
||||
t.Fatalf("Expected %v, found %v", expected, out)
|
||||
}
|
||||
out = ""
|
||||
i = 0
|
||||
stopChan = TickerFunc(interval, func() {
|
||||
i++
|
||||
out += fmt.Sprint(i)
|
||||
})
|
||||
time.Sleep(interval * 10)
|
||||
if !strings.HasPrefix(out, expected) || out == expected {
|
||||
t.Fatalf("expected (%v) must be a proper prefix of out(%v).", expected, out)
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue