2015-07-21 08:58:34 +03:00
|
|
|
package markdown
|
|
|
|
|
|
|
|
import (
|
2015-08-05 01:35:09 +03:00
|
|
|
"bufio"
|
2015-07-30 00:43:26 +03:00
|
|
|
"log"
|
2015-07-21 08:58:34 +03:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2015-07-29 19:00:08 +03:00
|
|
|
"os"
|
2015-07-21 18:45:32 +03:00
|
|
|
"strings"
|
2015-07-29 14:15:02 +03:00
|
|
|
"sync"
|
2015-07-21 08:58:34 +03:00
|
|
|
"testing"
|
2015-07-29 19:00:08 +03:00
|
|
|
"time"
|
2015-07-21 08:58:34 +03:00
|
|
|
|
2015-07-23 10:35:46 +03:00
|
|
|
"github.com/mholt/caddy/middleware"
|
2015-07-21 08:58:34 +03:00
|
|
|
"github.com/russross/blackfriday"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMarkdown(t *testing.T) {
|
|
|
|
templates := make(map[string]string)
|
2015-07-23 10:35:46 +03:00
|
|
|
templates[DefaultTemplate] = "testdata/markdown_tpl.html"
|
2015-07-21 08:58:34 +03:00
|
|
|
md := Markdown{
|
2015-07-23 10:35:46 +03:00
|
|
|
Root: "./testdata",
|
|
|
|
FileSys: http.Dir("./testdata"),
|
2015-09-11 01:12:46 +03:00
|
|
|
Configs: []*Config{
|
2015-10-10 01:35:34 +03:00
|
|
|
{
|
2015-07-29 14:15:02 +03:00
|
|
|
Renderer: blackfriday.HtmlRenderer(0, "", ""),
|
|
|
|
PathScope: "/blog",
|
|
|
|
Extensions: []string{".md"},
|
|
|
|
Styles: []string{},
|
|
|
|
Scripts: []string{},
|
|
|
|
Templates: templates,
|
|
|
|
StaticDir: DefaultStaticDir,
|
|
|
|
StaticFiles: make(map[string]string),
|
2015-07-21 08:58:34 +03:00
|
|
|
},
|
2016-02-22 13:53:47 +03:00
|
|
|
{
|
|
|
|
Renderer: blackfriday.HtmlRenderer(0, "", ""),
|
|
|
|
PathScope: "/docflags",
|
|
|
|
Extensions: []string{".md"},
|
|
|
|
Styles: []string{},
|
|
|
|
Scripts: []string{},
|
|
|
|
Templates: map[string]string{
|
|
|
|
DefaultTemplate: "testdata/docflags/template.txt",
|
|
|
|
},
|
|
|
|
StaticDir: DefaultStaticDir,
|
|
|
|
StaticFiles: make(map[string]string),
|
|
|
|
},
|
2015-10-10 01:35:34 +03:00
|
|
|
{
|
2015-07-29 14:15:02 +03:00
|
|
|
Renderer: blackfriday.HtmlRenderer(0, "", ""),
|
|
|
|
PathScope: "/log",
|
|
|
|
Extensions: []string{".md"},
|
|
|
|
Styles: []string{"/resources/css/log.css", "/resources/css/default.css"},
|
|
|
|
Scripts: []string{"/resources/js/log.js", "/resources/js/default.js"},
|
|
|
|
Templates: make(map[string]string),
|
|
|
|
StaticDir: DefaultStaticDir,
|
|
|
|
StaticFiles: make(map[string]string),
|
2015-07-21 18:45:32 +03:00
|
|
|
},
|
2015-10-10 01:35:34 +03:00
|
|
|
{
|
2015-07-29 17:21:35 +03:00
|
|
|
Renderer: blackfriday.HtmlRenderer(0, "", ""),
|
|
|
|
PathScope: "/og",
|
|
|
|
Extensions: []string{".md"},
|
|
|
|
Styles: []string{},
|
|
|
|
Scripts: []string{},
|
|
|
|
Templates: templates,
|
2015-07-29 18:12:32 +03:00
|
|
|
StaticDir: "testdata/og_static",
|
2015-07-29 17:21:35 +03:00
|
|
|
StaticFiles: map[string]string{"/og/first.md": "testdata/og_static/og/first.md/index.html"},
|
2015-07-30 00:43:26 +03:00
|
|
|
Links: []PageLink{
|
2015-10-10 01:35:34 +03:00
|
|
|
{
|
2015-07-30 00:43:26 +03:00
|
|
|
Title: "first",
|
|
|
|
Summary: "",
|
|
|
|
Date: time.Now(),
|
2015-07-30 02:47:33 +03:00
|
|
|
URL: "/og/first.md",
|
2015-07-30 00:43:26 +03:00
|
|
|
},
|
|
|
|
},
|
2015-07-29 17:21:35 +03:00
|
|
|
},
|
2015-07-21 08:58:34 +03:00
|
|
|
},
|
|
|
|
IndexFiles: []string{"index.html"},
|
2015-07-23 10:35:46 +03:00
|
|
|
Next: middleware.HandlerFunc(func(w http.ResponseWriter, r *http.Request) (int, error) {
|
|
|
|
t.Fatalf("Next shouldn't be called")
|
|
|
|
return 0, nil
|
|
|
|
}),
|
2015-07-21 08:58:34 +03:00
|
|
|
}
|
|
|
|
|
2015-08-05 11:55:04 +03:00
|
|
|
for i := range md.Configs {
|
2015-09-11 01:12:46 +03:00
|
|
|
c := md.Configs[i]
|
2015-08-05 11:55:04 +03:00
|
|
|
if err := GenerateStatic(md, c); err != nil {
|
|
|
|
t.Fatalf("Error: %v", err)
|
|
|
|
}
|
|
|
|
Watch(md, c, time.Millisecond*100)
|
|
|
|
}
|
|
|
|
|
2015-07-21 08:58:34 +03:00
|
|
|
req, err := http.NewRequest("GET", "/blog/test.md", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not create HTTP request: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
rec := httptest.NewRecorder()
|
|
|
|
|
|
|
|
md.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
|
|
t.Fatalf("Wrong status, expected: %d and got %d", http.StatusOK, rec.Code)
|
|
|
|
}
|
|
|
|
|
|
|
|
respBody := rec.Body.String()
|
|
|
|
expectedBody := `<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
2015-10-29 18:59:14 +03:00
|
|
|
<title>Markdown test 1</title>
|
2015-07-21 08:58:34 +03:00
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<h1>Header</h1>
|
|
|
|
|
|
|
|
Welcome to A Caddy website!
|
|
|
|
<h2>Welcome on the blog</h2>
|
|
|
|
|
|
|
|
<p>Body</p>
|
|
|
|
|
2015-10-29 18:59:14 +03:00
|
|
|
<pre><code class="language-go">func getTrue() bool {
|
2015-07-21 08:58:34 +03:00
|
|
|
return true
|
|
|
|
}
|
2015-10-29 18:59:14 +03:00
|
|
|
</code></pre>
|
2015-07-21 08:58:34 +03:00
|
|
|
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
`
|
2015-08-05 01:35:09 +03:00
|
|
|
if !equalStrings(respBody, expectedBody) {
|
2015-07-21 18:45:32 +03:00
|
|
|
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
|
|
|
}
|
|
|
|
|
2016-02-22 13:53:47 +03:00
|
|
|
req, err = http.NewRequest("GET", "/docflags/test.md", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not create HTTP request: %v", err)
|
|
|
|
}
|
|
|
|
rec = httptest.NewRecorder()
|
|
|
|
|
|
|
|
md.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
|
|
t.Fatalf("Wrong status, expected: %d and got %d", http.StatusOK, rec.Code)
|
|
|
|
}
|
|
|
|
respBody = rec.Body.String()
|
|
|
|
expectedBody = `Doc.var_string hello
|
|
|
|
Doc.var_bool <no value>
|
|
|
|
DocFlags.var_string <no value>
|
|
|
|
DocFlags.var_bool true`
|
|
|
|
|
|
|
|
if !equalStrings(respBody, expectedBody) {
|
|
|
|
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
|
|
|
}
|
|
|
|
|
2015-07-21 18:45:32 +03:00
|
|
|
req, err = http.NewRequest("GET", "/log/test.md", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not create HTTP request: %v", err)
|
|
|
|
}
|
|
|
|
rec = httptest.NewRecorder()
|
|
|
|
|
|
|
|
md.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
|
|
t.Fatalf("Wrong status, expected: %d and got %d", http.StatusOK, rec.Code)
|
|
|
|
}
|
|
|
|
respBody = rec.Body.String()
|
|
|
|
expectedBody = `<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
2015-10-29 18:59:14 +03:00
|
|
|
<title>Markdown test 2</title>
|
2015-07-21 18:45:32 +03:00
|
|
|
<meta charset="utf-8">
|
|
|
|
<link rel="stylesheet" href="/resources/css/log.css">
|
|
|
|
<link rel="stylesheet" href="/resources/css/default.css">
|
|
|
|
|
|
|
|
<script src="/resources/js/log.js"></script>
|
|
|
|
<script src="/resources/js/default.js"></script>
|
|
|
|
|
|
|
|
</head>
|
|
|
|
<body>
|
|
|
|
<h2>Welcome on the blog</h2>
|
|
|
|
|
|
|
|
<p>Body</p>
|
|
|
|
|
2015-10-29 18:59:14 +03:00
|
|
|
<pre><code class="language-go">func getTrue() bool {
|
2015-07-21 18:45:32 +03:00
|
|
|
return true
|
|
|
|
}
|
2015-10-29 18:59:14 +03:00
|
|
|
</code></pre>
|
2015-07-21 18:45:32 +03:00
|
|
|
|
|
|
|
</body>
|
|
|
|
</html>`
|
2015-07-21 08:58:34 +03:00
|
|
|
|
2015-08-05 01:35:09 +03:00
|
|
|
if !equalStrings(respBody, expectedBody) {
|
2015-07-21 08:58:34 +03:00
|
|
|
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
|
|
|
}
|
2015-07-29 17:21:35 +03:00
|
|
|
|
|
|
|
req, err = http.NewRequest("GET", "/og/first.md", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not create HTTP request: %v", err)
|
|
|
|
}
|
|
|
|
rec = httptest.NewRecorder()
|
2015-07-29 19:00:08 +03:00
|
|
|
currenttime := time.Now().Local().Add(-time.Second)
|
|
|
|
err = os.Chtimes("testdata/og/first.md", currenttime, currenttime)
|
|
|
|
currenttime = time.Now().Local()
|
|
|
|
err = os.Chtimes("testdata/og_static/og/first.md/index.html", currenttime, currenttime)
|
2015-08-05 11:55:04 +03:00
|
|
|
time.Sleep(time.Millisecond * 200)
|
2015-07-29 17:21:35 +03:00
|
|
|
|
|
|
|
md.ServeHTTP(rec, req)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
|
|
t.Fatalf("Wrong status, expected: %d and got %d", http.StatusOK, rec.Code)
|
|
|
|
}
|
|
|
|
respBody = rec.Body.String()
|
|
|
|
expectedBody = `<!DOCTYPE html>
|
|
|
|
<html>
|
|
|
|
<head>
|
|
|
|
<title>first_post</title>
|
|
|
|
</head>
|
|
|
|
<body>
|
2015-08-05 11:55:04 +03:00
|
|
|
<h1>Header</h1>
|
2015-07-29 17:21:35 +03:00
|
|
|
|
2015-08-05 11:55:04 +03:00
|
|
|
Welcome to title!
|
2015-07-29 17:21:35 +03:00
|
|
|
<h1>Test h1</h1>
|
|
|
|
|
|
|
|
</body>
|
|
|
|
</html>`
|
2015-08-05 01:35:09 +03:00
|
|
|
|
|
|
|
if !equalStrings(respBody, expectedBody) {
|
2015-07-29 17:21:35 +03:00
|
|
|
t.Fatalf("Expected body: %v got: %v", expectedBody, respBody)
|
|
|
|
}
|
2015-07-30 00:43:26 +03:00
|
|
|
|
2015-07-29 14:15:02 +03:00
|
|
|
expectedLinks := []string{
|
|
|
|
"/blog/test.md",
|
2016-02-22 13:53:47 +03:00
|
|
|
"/docflags/test.md",
|
2015-07-29 14:15:02 +03:00
|
|
|
"/log/test.md",
|
|
|
|
}
|
|
|
|
|
2015-08-05 01:35:09 +03:00
|
|
|
for i, c := range md.Configs[:2] {
|
2015-07-30 00:43:26 +03:00
|
|
|
log.Printf("Test number: %d, configuration links: %v, config: %v", i, c.Links, c)
|
2015-07-30 02:47:33 +03:00
|
|
|
if c.Links[0].URL != expectedLinks[i] {
|
|
|
|
t.Fatalf("Expected %v got %v", expectedLinks[i], c.Links[0].URL)
|
2015-07-29 14:15:02 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-05 02:41:04 +03:00
|
|
|
// attempt to trigger race conditions
|
2015-07-29 14:15:02 +03:00
|
|
|
var w sync.WaitGroup
|
|
|
|
f := func() {
|
|
|
|
req, err := http.NewRequest("GET", "/log/test.md", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not create HTTP request: %v", err)
|
|
|
|
}
|
|
|
|
rec := httptest.NewRecorder()
|
|
|
|
|
|
|
|
md.ServeHTTP(rec, req)
|
|
|
|
w.Done()
|
|
|
|
}
|
|
|
|
for i := 0; i < 5; i++ {
|
|
|
|
w.Add(1)
|
|
|
|
go f()
|
|
|
|
}
|
|
|
|
w.Wait()
|
|
|
|
|
2015-08-05 02:41:04 +03:00
|
|
|
f = func() {
|
2015-09-11 01:12:46 +03:00
|
|
|
GenerateStatic(md, md.Configs[0])
|
2015-08-05 02:41:04 +03:00
|
|
|
w.Done()
|
|
|
|
}
|
|
|
|
for i := 0; i < 5; i++ {
|
|
|
|
w.Add(1)
|
|
|
|
go f()
|
|
|
|
}
|
|
|
|
w.Wait()
|
|
|
|
|
2015-07-30 00:43:26 +03:00
|
|
|
if err = os.RemoveAll(DefaultStaticDir); err != nil {
|
|
|
|
t.Errorf("Error while removing the generated static files: %v", err)
|
|
|
|
}
|
|
|
|
|
2015-07-21 08:58:34 +03:00
|
|
|
}
|
2015-08-05 01:35:09 +03:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|