From: spf13 Date: Wed, 23 Apr 2014 06:52:01 +0000 (-0400) Subject: Add Disqus support out of the box. Move template/bundle into hugolib. X-Git-Tag: v0.11~74 X-Git-Url: http://git.maquefel.me/?a=commitdiff_plain;h=4a8de8ea;p=brevno-suite%2Fhugo Add Disqus support out of the box. Move template/bundle into hugolib. --- diff --git a/hugolib/page.go b/hugolib/page.go index 0c66c215..20665233 100644 --- a/hugolib/page.go +++ b/hugolib/page.go @@ -28,7 +28,6 @@ import ( "github.com/spf13/cast" "github.com/spf13/hugo/helpers" "github.com/spf13/hugo/parser" - "github.com/spf13/hugo/template/bundle" jww "github.com/spf13/jwalterweatherman" "github.com/spf13/viper" "github.com/theplant/blackfriday" @@ -49,7 +48,7 @@ type Page struct { contentType string Draft bool Aliases []string - Tmpl bundle.Template + Tmpl Template Markup string renderable bool layout string @@ -519,7 +518,7 @@ func (page *Page) parse(reader io.Reader) error { return nil } -func (p *Page) ProcessShortcodes(t bundle.Template) { +func (p *Page) ProcessShortcodes(t Template) { p.rawContent = []byte(ShortcodesHandle(string(p.rawContent), p, t)) p.Summary = template.HTML(ShortcodesHandle(string(p.Summary), p, t)) } diff --git a/hugolib/path_seperators_windows_test.go b/hugolib/path_seperators_windows_test.go new file mode 100644 index 00000000..5cdd7c5f --- /dev/null +++ b/hugolib/path_seperators_windows_test.go @@ -0,0 +1,17 @@ +package hugolib + +import ( + "testing" +) + +const ( + win_base = "c:\\a\\windows\\path\\layout" + win_path = "c:\\a\\windows\\path\\layout\\sub1\\index.html" +) + +func TestTemplatePathSeperator(t *testing.T) { + tmpl := new(GoHtmlTemplate) + if name := tmpl.generateTemplateNameFrom(win_base, win_path); name != "sub1/index.html" { + t.Fatalf("Template name incorrect. Expected: %s, Got: %s", "sub1/index.html", name) + } +} diff --git a/hugolib/shortcode.go b/hugolib/shortcode.go index 92e81195..67204b74 100644 --- a/hugolib/shortcode.go +++ b/hugolib/shortcode.go @@ -20,7 +20,6 @@ import ( "strings" "unicode" - "github.com/spf13/hugo/template/bundle" jww "github.com/spf13/jwalterweatherman" ) @@ -78,7 +77,7 @@ func (scp *ShortcodeWithPage) Get(key interface{}) interface{} { type Shortcodes map[string]ShortcodeFunc -func ShortcodesHandle(stringToParse string, p *Page, t bundle.Template) string { +func ShortcodesHandle(stringToParse string, p *Page, t Template) string { leadStart := strings.Index(stringToParse, `{{%`) if leadStart >= 0 { leadEnd := strings.Index(stringToParse[leadStart:], `%}}`) + leadStart @@ -147,7 +146,7 @@ func FindEnd(str string, name string) (int, int) { return startPos, endPos } -func GetTemplate(name string, t bundle.Template) *template.Template { +func GetTemplate(name string, t Template) *template.Template { if x := t.Lookup("shortcodes/" + name + ".html"); x != nil { return x } diff --git a/hugolib/site.go b/hugolib/site.go index 050eaafb..20d03cb0 100644 --- a/hugolib/site.go +++ b/hugolib/site.go @@ -27,7 +27,6 @@ import ( "github.com/spf13/hugo/helpers" "github.com/spf13/hugo/source" "github.com/spf13/hugo/target" - "github.com/spf13/hugo/template/bundle" "github.com/spf13/hugo/transform" jww "github.com/spf13/jwalterweatherman" "github.com/spf13/nitro" @@ -57,7 +56,7 @@ var DefaultTimer *nitro.B // 5. The entire collection of files is written to disk. type Site struct { Pages Pages - Tmpl bundle.Template + Tmpl Template Taxonomies TaxonomyList Source source.Input Sections Taxonomy @@ -72,19 +71,20 @@ type Site struct { } type SiteInfo struct { - BaseUrl template.URL - Taxonomies TaxonomyList - Indexes *TaxonomyList // legacy, should be identical to Taxonomies - Recent *Pages - Title string - Author string - AuthorEmail string - LanguageCode string - Copyright string - LastChange time.Time - ConfigGet func(key string) interface{} - Permalinks PermalinkOverrides - Params map[string]interface{} + BaseUrl template.URL + Taxonomies TaxonomyList + Indexes *TaxonomyList // legacy, should be identical to Taxonomies + Recent *Pages + Title string + Author map[string]string + LanguageCode string + DisqusShortname string + Copyright string + LastChange time.Time + ConfigGet func(key string) interface{} + Permalinks PermalinkOverrides + Params map[string]interface{} +} } type runmode struct { @@ -130,7 +130,7 @@ func (s *Site) Analyze() { } func (s *Site) prepTemplates() { - s.Tmpl = bundle.NewTemplate() + s.Tmpl = NewTemplate() s.Tmpl.LoadTemplates(s.absLayoutDir()) if s.hasTheme() { s.Tmpl.LoadTemplatesWithPrefix(s.absThemeDir()+"/layouts", "theme") @@ -234,16 +234,16 @@ func (s *Site) initializeSiteInfo() { permalinks = make(PermalinkOverrides) } - s.Info = SiteInfo{ - BaseUrl: template.URL(helpers.SanitizeUrl(viper.GetString("BaseUrl"))), - Title: viper.GetString("Title"), - Author: viper.GetString("author"), - AuthorEmail: viper.GetString("authoremail"), - LanguageCode: viper.GetString("languagecode"), - Copyright: viper.GetString("copyright"), - Recent: &s.Pages, - Params: params, - Permalinks: permalinks, + s.Info = &SiteInfo{ + BaseUrl: template.URL(helpers.SanitizeUrl(viper.GetString("BaseUrl"))), + Title: viper.GetString("Title"), + Author: viper.GetStringMapString("author"), + LanguageCode: viper.GetString("languagecode"), + Copyright: viper.GetString("copyright"), + DisqusShortname: viper.GetString("DisqusShortname"), + Recent: &s.Pages, + Params: params, + Permalinks: permalinks, } } diff --git a/hugolib/template.go b/hugolib/template.go new file mode 100644 index 00000000..f7bd5281 --- /dev/null +++ b/hugolib/template.go @@ -0,0 +1,275 @@ +package hugolib + +import ( + "errors" + "html" + "html/template" + "io" + "io/ioutil" + "os" + "path/filepath" + "reflect" + "strconv" + "strings" + + "github.com/eknkc/amber" + "github.com/spf13/hugo/helpers" +) + +func Gt(a interface{}, b interface{}) bool { + var left, right int64 + av := reflect.ValueOf(a) + + switch av.Kind() { + case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: + left = int64(av.Len()) + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + left = av.Int() + case reflect.String: + left, _ = strconv.ParseInt(av.String(), 10, 64) + } + + bv := reflect.ValueOf(b) + + switch bv.Kind() { + case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: + right = int64(bv.Len()) + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + right = bv.Int() + case reflect.String: + right, _ = strconv.ParseInt(bv.String(), 10, 64) + } + + return left > right +} + +// First is exposed to templates, to iterate over the first N items in a +// rangeable list. +func First(limit int, seq interface{}) (interface{}, error) { + if limit < 1 { + return nil, errors.New("can't return negative/empty count of items from sequence") + } + + seqv := reflect.ValueOf(seq) + // this is better than my first pass; ripped from text/template/exec.go indirect(): + for ; seqv.Kind() == reflect.Ptr || seqv.Kind() == reflect.Interface; seqv = seqv.Elem() { + if seqv.IsNil() { + return nil, errors.New("can't iterate over a nil value") + } + if seqv.Kind() == reflect.Interface && seqv.NumMethod() > 0 { + break + } + } + + switch seqv.Kind() { + case reflect.Array, reflect.Slice, reflect.String: + // okay + default: + return nil, errors.New("can't iterate over " + reflect.ValueOf(seq).Type().String()) + } + if limit > seqv.Len() { + limit = seqv.Len() + } + return seqv.Slice(0, limit).Interface(), nil +} + +func IsSet(a interface{}, key interface{}) bool { + av := reflect.ValueOf(a) + kv := reflect.ValueOf(key) + + switch av.Kind() { + case reflect.Array, reflect.Chan, reflect.Slice: + if int64(av.Len()) > kv.Int() { + return true + } + case reflect.Map: + if kv.Type() == av.Type().Key() { + return av.MapIndex(kv).IsValid() + } + } + + return false +} + +func ReturnWhenSet(a interface{}, index int) interface{} { + av := reflect.ValueOf(a) + + switch av.Kind() { + case reflect.Array, reflect.Slice: + if av.Len() > index { + + avv := av.Index(index) + switch avv.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return avv.Int() + case reflect.String: + return avv.String() + } + } + } + + return "" +} + +func Highlight(in interface{}, lang string) template.HTML { + var str string + av := reflect.ValueOf(in) + switch av.Kind() { + case reflect.String: + str = av.String() + } + + if strings.HasPrefix(strings.TrimSpace(str), "
") {
+		str = str[strings.Index(str, "
")+11:]
+	}
+	if strings.HasSuffix(strings.TrimSpace(str), "
") { + str = str[:strings.LastIndex(str, "
")] + } + return template.HTML(helpers.Highlight(html.UnescapeString(str), lang)) +} + +func SafeHtml(text string) template.HTML { + return template.HTML(text) +} + +type Template interface { + ExecuteTemplate(wr io.Writer, name string, data interface{}) error + Lookup(name string) *template.Template + Templates() []*template.Template + New(name string) *template.Template + LoadTemplates(absPath string) + LoadTemplatesWithPrefix(absPath, prefix string) + AddTemplate(name, tpl string) error + AddInternalTemplate(prefix, name, tpl string) error + AddInternalShortcode(name, tpl string) error +} + +type templateErr struct { + name string + err error +} + +type GoHtmlTemplate struct { + template.Template + errors []*templateErr +} + +func NewTemplate() Template { + var templates = &GoHtmlTemplate{ + Template: *template.New(""), + errors: make([]*templateErr, 0), + } + + funcMap := template.FuncMap{ + "urlize": helpers.Urlize, + "sanitizeurl": helpers.SanitizeUrl, + "gt": Gt, + "isset": IsSet, + "echoParam": ReturnWhenSet, + "safeHtml": SafeHtml, + "first": First, + "highlight": Highlight, + "add": func(a, b int) int { return a + b }, + "sub": func(a, b int) int { return a - b }, + "div": func(a, b int) int { return a / b }, + "mod": func(a, b int) int { return a % b }, + "mul": func(a, b int) int { return a * b }, + "modBool": func(a, b int) bool { return a%b == 0 }, + "lower": func(a string) string { return strings.ToLower(a) }, + "upper": func(a string) string { return strings.ToUpper(a) }, + "title": func(a string) string { return strings.Title(a) }, + } + + templates.Funcs(funcMap) + + templates.LoadEmbedded() + return templates +} + +func (t *GoHtmlTemplate) LoadEmbedded() { + t.EmbedShortcodes() + t.EmbedTemplates() +} + +func (t *GoHtmlTemplate) AddInternalTemplate(prefix, name, tpl string) error { + if prefix != "" { + return t.AddTemplate("_internal/"+prefix+"/"+name, tpl) + } else { + return t.AddTemplate("_internal/"+name, tpl) + } +} + +func (t *GoHtmlTemplate) AddInternalShortcode(name, content string) error { + return t.AddInternalTemplate("shortcodes", name, content) +} + +func (t *GoHtmlTemplate) AddTemplate(name, tpl string) error { + _, err := t.New(name).Parse(tpl) + if err != nil { + t.errors = append(t.errors, &templateErr{name: name, err: err}) + } + return err +} + +func (t *GoHtmlTemplate) AddTemplateFile(name, path string) error { + b, err := ioutil.ReadFile(path) + if err != nil { + return err + } + return t.AddTemplate(name, string(b)) +} + +func (t *GoHtmlTemplate) generateTemplateNameFrom(base, path string) string { + return filepath.ToSlash(path[len(base)+1:]) +} + +func ignoreDotFile(path string) bool { + return filepath.Base(path)[0] == '.' +} + +func (t *GoHtmlTemplate) loadTemplates(absPath string, prefix string) { + walker := func(path string, fi os.FileInfo, err error) error { + if err != nil { + return nil + } + + if !fi.IsDir() { + if ignoreDotFile(path) { + return nil + } + + tplName := t.generateTemplateNameFrom(absPath, path) + + if prefix != "" { + tplName = strings.Trim(prefix, "/") + "/" + tplName + } + + // TODO move this into the AddTemplateFile function + if strings.HasSuffix(path, ".amber") { + compiler := amber.New() + // Parse the input file + if err := compiler.ParseFile(path); err != nil { + return nil + } + + if _, err := compiler.CompileWithTemplate(t.New(tplName)); err != nil { + return err + } + + } else { + t.AddTemplateFile(tplName, path) + } + } + return nil + } + + filepath.Walk(absPath, walker) +} + +func (t *GoHtmlTemplate) LoadTemplatesWithPrefix(absPath string, prefix string) { + t.loadTemplates(absPath, prefix) +} + +func (t *GoHtmlTemplate) LoadTemplates(absPath string) { + t.loadTemplates(absPath, "") +} diff --git a/hugolib/template_embedded.go b/hugolib/template_embedded.go new file mode 100644 index 00000000..2555f9a2 --- /dev/null +++ b/hugolib/template_embedded.go @@ -0,0 +1,84 @@ +// Copyright © 2013 Steve Francia . +// +// Licensed under the Simple Public License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// http://opensource.org/licenses/Simple-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package hugolib + +type Tmpl struct { + Name string + Data string +} + +func (t *GoHtmlTemplate) EmbedShortcodes() { + t.AddInternalShortcode("highlight.html", `{{ .Get 0 | highlight .Inner }}`) + t.AddInternalShortcode("test.html", `This is a simple Test`) + t.AddInternalShortcode("figure.html", ` +
+ {{ with .Get "link"}}{{ end }} + + {{ if .Get "link"}}{{ end }} + {{ if or (or (.Get "title") (.Get "caption")) (.Get "attr")}} +
{{ if isset .Params "title" }} +

{{ .Get "title" }}

{{ end }} + {{ if or (.Get "caption") (.Get "attr")}}

+ {{ .Get "caption" }} + {{ with .Get "attrlink"}} {{ end }} + {{ .Get "attr" }} + {{ if .Get "attrlink"}} {{ end }} +

{{ end }} +
+ {{ end }} +
+`) +} + +func (t *GoHtmlTemplate) EmbedTemplates() { + + t.AddInternalTemplate("_default", "rss.xml", ` + + {{ .Title }} on {{ .Site.Title }} + Hugo + {{ .Permalink }} + {{ with .Site.LanguageCode }}{{.}}{{end}} + {{ with .Site.Author.name }}{{.}}{{end}} + {{ with .Site.Copyright }}{{.}}{{end}} + {{ .Date.Format "Mon, 02 Jan 2006 15:04:05 MST" }} + {{ range first 15 .Data.Pages }} + + {{ .Title }} + {{ .Permalink }} + {{ .Date.Format "Mon, 02 Jan 2006 15:04:05 MST" }} + {{with .Site.Author.name}}{{.}}{{end}} + {{ .Permalink }} + {{ .Content | html }} + + {{ end }} + +`) + + t.AddInternalTemplate("", "disqus.html", `{{ if .Site.DisqusShortname }}
+ + +comments powered by Disqus{{end}}`) + +} diff --git a/hugolib/template_test.go b/hugolib/template_test.go new file mode 100644 index 00000000..ffc417b4 --- /dev/null +++ b/hugolib/template_test.go @@ -0,0 +1,55 @@ +package hugolib + +import ( + "reflect" + "testing" +) + +func TestGt(t *testing.T) { + for i, this := range []struct { + left interface{} + right interface{} + leftShouldWin bool + }{ + {5, 8, false}, + {8, 5, true}, + {5, 5, false}, + {-2, 1, false}, + {2, -5, true}, + {"8", "5", true}, + {"5", "0001", true}, + {[]int{100, 99}, []int{1, 2, 3, 4}, false}, + } { + leftIsBigger := Gt(this.left, this.right) + if leftIsBigger != this.leftShouldWin { + var which string + if leftIsBigger { + which = "expected right to be bigger, but left was" + } else { + which = "expected left to be bigger, but right was" + } + t.Errorf("[%d] %v compared to %v: %s", i, this.left, this.right, which) + } + } +} + +func TestFirst(t *testing.T) { + for i, this := range []struct { + count int + sequence interface{} + expect interface{} + }{ + {2, []string{"a", "b", "c"}, []string{"a", "b"}}, + {3, []string{"a", "b"}, []string{"a", "b"}}, + {2, []int{100, 200, 300}, []int{100, 200}}, + } { + results, err := First(this.count, this.sequence) + if err != nil { + t.Errorf("[%d] failed: %s", i, err) + continue + } + if !reflect.DeepEqual(results, this.expect) { + t.Errorf("[%d] First %d items, got %v but expected %v", i, this.count, results, this.expect) + } + } +} diff --git a/template/bundle/embedded.go b/template/bundle/embedded.go deleted file mode 100644 index 45e182b7..00000000 --- a/template/bundle/embedded.go +++ /dev/null @@ -1,68 +0,0 @@ -// Copyright © 2013 Steve Francia . -// -// Licensed under the Simple Public License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// http://opensource.org/licenses/Simple-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -package bundle - -type Tmpl struct { - Name string - Data string -} - -func (t *GoHtmlTemplate) EmbedShortcodes() { - t.AddInternalShortcode("highlight.html", `{{ .Get 0 | highlight .Inner }}`) - t.AddInternalShortcode("test.html", `This is a simple Test`) - t.AddInternalShortcode("figure.html", ` -
- {{ with .Get "link"}}{{ end }} - - {{ if .Get "link"}}{{ end }} - {{ if or (or (.Get "title") (.Get "caption")) (.Get "attr")}} -
{{ if isset .Params "title" }} -

{{ .Get "title" }}

{{ end }} - {{ if or (.Get "caption") (.Get "attr")}}

- {{ .Get "caption" }} - {{ with .Get "attrlink"}} {{ end }} - {{ .Get "attr" }} - {{ if .Get "attrlink"}} {{ end }} -

{{ end }} -
- {{ end }} -
-`) -} - -func (t *GoHtmlTemplate) EmbedTemplates() { - - t.AddInternalTemplate("_default", "rss.xml", ` - - {{ .Title }} on {{ .Site.Title }} - Hugo - {{ .Permalink }} - {{ with .Site.LanguageCode }}{{.}}{{end}} - {{ with .Site.Author }}{{.}}{{end}} - {{ with .Site.Copyright }}{{.}}{{end}} - {{ .Date.Format "Mon, 02 Jan 2006 15:04:05 MST" }} - {{ range first 15 .Data.Pages }} - - {{ .Title }} - {{ .Permalink }} - {{ .Date.Format "Mon, 02 Jan 2006 15:04:05 MST" }} - {{with .Site.Author}}{{.}}{{end}} - {{ .Permalink }} - {{ .Content | html }} - - {{ end }} - -`) - -} diff --git a/template/bundle/path_seperators_windows_test.go b/template/bundle/path_seperators_windows_test.go deleted file mode 100644 index e5f168b4..00000000 --- a/template/bundle/path_seperators_windows_test.go +++ /dev/null @@ -1,17 +0,0 @@ -package bundle - -import ( - "testing" -) - -const ( - win_base = "c:\\a\\windows\\path\\layout" - win_path = "c:\\a\\windows\\path\\layout\\sub1\\index.html" -) - -func TestTemplatePathSeperator(t *testing.T) { - tmpl := new(GoHtmlTemplate) - if name := tmpl.generateTemplateNameFrom(win_base, win_path); name != "sub1/index.html" { - t.Fatalf("Template name incorrect. Expected: %s, Got: %s", "sub1/index.html", name) - } -} diff --git a/template/bundle/template.go b/template/bundle/template.go deleted file mode 100644 index 30d24144..00000000 --- a/template/bundle/template.go +++ /dev/null @@ -1,271 +0,0 @@ -package bundle - -import ( - "errors" - "html" - "html/template" - "io" - "io/ioutil" - "os" - "path/filepath" - "reflect" - "strconv" - "strings" - - "github.com/eknkc/amber" - "github.com/spf13/hugo/helpers" -) - -func Gt(a interface{}, b interface{}) bool { - var left, right int64 - av := reflect.ValueOf(a) - - switch av.Kind() { - case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: - left = int64(av.Len()) - case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: - left = av.Int() - case reflect.String: - left, _ = strconv.ParseInt(av.String(), 10, 64) - } - - bv := reflect.ValueOf(b) - - switch bv.Kind() { - case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: - right = int64(bv.Len()) - case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: - right = bv.Int() - case reflect.String: - right, _ = strconv.ParseInt(bv.String(), 10, 64) - } - - return left > right -} - -// First is exposed to templates, to iterate over the first N items in a -// rangeable list. -func First(limit int, seq interface{}) (interface{}, error) { - if limit < 1 { - return nil, errors.New("can't return negative/empty count of items from sequence") - } - - seqv := reflect.ValueOf(seq) - // this is better than my first pass; ripped from text/template/exec.go indirect(): - for ; seqv.Kind() == reflect.Ptr || seqv.Kind() == reflect.Interface; seqv = seqv.Elem() { - if seqv.IsNil() { - return nil, errors.New("can't iterate over a nil value") - } - if seqv.Kind() == reflect.Interface && seqv.NumMethod() > 0 { - break - } - } - - switch seqv.Kind() { - case reflect.Array, reflect.Slice, reflect.String: - // okay - default: - return nil, errors.New("can't iterate over " + reflect.ValueOf(seq).Type().String()) - } - if limit > seqv.Len() { - limit = seqv.Len() - } - return seqv.Slice(0, limit).Interface(), nil -} - -func IsSet(a interface{}, key interface{}) bool { - av := reflect.ValueOf(a) - kv := reflect.ValueOf(key) - - switch av.Kind() { - case reflect.Array, reflect.Chan, reflect.Slice: - if int64(av.Len()) > kv.Int() { - return true - } - case reflect.Map: - if kv.Type() == av.Type().Key() { - return av.MapIndex(kv).IsValid() - } - } - - return false -} - -func ReturnWhenSet(a interface{}, index int) interface{} { - av := reflect.ValueOf(a) - - switch av.Kind() { - case reflect.Array, reflect.Slice: - if av.Len() > index { - - avv := av.Index(index) - switch avv.Kind() { - case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: - return avv.Int() - case reflect.String: - return avv.String() - } - } - } - - return "" -} - -func Highlight(in interface{}, lang string) template.HTML { - var str string - av := reflect.ValueOf(in) - switch av.Kind() { - case reflect.String: - str = av.String() - } - - if strings.HasPrefix(strings.TrimSpace(str), "
") {
-		str = str[strings.Index(str, "
")+11:]
-	}
-	if strings.HasSuffix(strings.TrimSpace(str), "
") { - str = str[:strings.LastIndex(str, "
")] - } - return template.HTML(helpers.Highlight(html.UnescapeString(str), lang)) -} - -func SafeHtml(text string) template.HTML { - return template.HTML(text) -} - -type Template interface { - ExecuteTemplate(wr io.Writer, name string, data interface{}) error - Lookup(name string) *template.Template - Templates() []*template.Template - New(name string) *template.Template - LoadTemplates(absPath string) - LoadTemplatesWithPrefix(absPath, prefix string) - AddTemplate(name, tpl string) error - AddInternalTemplate(prefix, name, tpl string) error - AddInternalShortcode(name, tpl string) error -} - -type templateErr struct { - name string - err error -} - -type GoHtmlTemplate struct { - template.Template - errors []*templateErr -} - -func NewTemplate() Template { - var templates = &GoHtmlTemplate{ - Template: *template.New(""), - errors: make([]*templateErr, 0), - } - - funcMap := template.FuncMap{ - "urlize": helpers.Urlize, - "sanitizeurl": helpers.SanitizeUrl, - "gt": Gt, - "isset": IsSet, - "echoParam": ReturnWhenSet, - "safeHtml": SafeHtml, - "first": First, - "highlight": Highlight, - "add": func(a, b int) int { return a + b }, - "sub": func(a, b int) int { return a - b }, - "div": func(a, b int) int { return a / b }, - "mod": func(a, b int) int { return a % b }, - "mul": func(a, b int) int { return a * b }, - "modBool": func(a, b int) bool { return a%b == 0 }, - "lower": func(a string) string { return strings.ToLower(a) }, - "upper": func(a string) string { return strings.ToUpper(a) }, - "title": func(a string) string { return strings.Title(a) }, - } - - templates.Funcs(funcMap) - - templates.LoadEmbedded() - return templates -} - -func (t *GoHtmlTemplate) LoadEmbedded() { - t.EmbedShortcodes() - t.EmbedTemplates() -} - -func (t *GoHtmlTemplate) AddInternalTemplate(prefix, name, tpl string) error { - return t.AddTemplate("_internal/"+prefix+"/"+name, tpl) -} - -func (t *GoHtmlTemplate) AddInternalShortcode(name, content string) error { - return t.AddInternalTemplate("shortcodes", name, content) -} - -func (t *GoHtmlTemplate) AddTemplate(name, tpl string) error { - _, err := t.New(name).Parse(tpl) - if err != nil { - t.errors = append(t.errors, &templateErr{name: name, err: err}) - } - return err -} - -func (t *GoHtmlTemplate) AddTemplateFile(name, path string) error { - b, err := ioutil.ReadFile(path) - if err != nil { - return err - } - return t.AddTemplate(name, string(b)) -} - -func (t *GoHtmlTemplate) generateTemplateNameFrom(base, path string) string { - return filepath.ToSlash(path[len(base)+1:]) -} - -func ignoreDotFile(path string) bool { - return filepath.Base(path)[0] == '.' -} - -func (t *GoHtmlTemplate) loadTemplates(absPath string, prefix string) { - walker := func(path string, fi os.FileInfo, err error) error { - if err != nil { - return nil - } - - if !fi.IsDir() { - if ignoreDotFile(path) { - return nil - } - - tplName := t.generateTemplateNameFrom(absPath, path) - - if prefix != "" { - tplName = strings.Trim(prefix, "/") + "/" + tplName - } - - // TODO move this into the AddTemplateFile function - if strings.HasSuffix(path, ".amber") { - compiler := amber.New() - // Parse the input file - if err := compiler.ParseFile(path); err != nil { - return nil - } - - if _, err := compiler.CompileWithTemplate(t.New(tplName)); err != nil { - return err - } - - } else { - t.AddTemplateFile(tplName, path) - } - } - return nil - } - - filepath.Walk(absPath, walker) -} - -func (t *GoHtmlTemplate) LoadTemplatesWithPrefix(absPath string, prefix string) { - t.loadTemplates(absPath, prefix) -} - -func (t *GoHtmlTemplate) LoadTemplates(absPath string) { - t.loadTemplates(absPath, "") -} diff --git a/template/bundle/template_test.go b/template/bundle/template_test.go deleted file mode 100644 index 019816f6..00000000 --- a/template/bundle/template_test.go +++ /dev/null @@ -1,55 +0,0 @@ -package bundle - -import ( - "reflect" - "testing" -) - -func TestGt(t *testing.T) { - for i, this := range []struct { - left interface{} - right interface{} - leftShouldWin bool - }{ - {5, 8, false}, - {8, 5, true}, - {5, 5, false}, - {-2, 1, false}, - {2, -5, true}, - {"8", "5", true}, - {"5", "0001", true}, - {[]int{100, 99}, []int{1, 2, 3, 4}, false}, - } { - leftIsBigger := Gt(this.left, this.right) - if leftIsBigger != this.leftShouldWin { - var which string - if leftIsBigger { - which = "expected right to be bigger, but left was" - } else { - which = "expected left to be bigger, but right was" - } - t.Errorf("[%d] %v compared to %v: %s", i, this.left, this.right, which) - } - } -} - -func TestFirst(t *testing.T) { - for i, this := range []struct { - count int - sequence interface{} - expect interface{} - }{ - {2, []string{"a", "b", "c"}, []string{"a", "b"}}, - {3, []string{"a", "b"}, []string{"a", "b"}}, - {2, []int{100, 200, 300}, []int{100, 200}}, - } { - results, err := First(this.count, this.sequence) - if err != nil { - t.Errorf("[%d] failed: %s", i, err) - continue - } - if !reflect.DeepEqual(results, this.expect) { - t.Errorf("[%d] First %d items, got %v but expected %v", i, this.count, results, this.expect) - } - } -}