mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-29 19:12:09 -05:00
eada236f87
This commit introduces a new data structure to store pages and their resources. This data structure is backed by radix trees. This simplies tree operations, makes all pages a bundle, and paves the way for #6310. It also solves a set of annoying issues (see list below). Not a motivation behind this, but this commit also makes Hugo in general a little bit faster and more memory effective (see benchmarks). Especially for partial rebuilds on content edits, but also when taxonomies is in use. ``` name old time/op new time/op delta SiteNew/Bundle_with_image/Edit-16 1.32ms ± 8% 1.00ms ± 9% -24.42% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file/Edit-16 1.28ms ± 0% 0.94ms ± 0% -26.26% (p=0.029 n=4+4) SiteNew/Tags_and_categories/Edit-16 33.9ms ± 2% 21.8ms ± 1% -35.67% (p=0.029 n=4+4) SiteNew/Canonify_URLs/Edit-16 40.6ms ± 1% 37.7ms ± 3% -7.20% (p=0.029 n=4+4) SiteNew/Deep_content_tree/Edit-16 56.7ms ± 0% 51.7ms ± 1% -8.82% (p=0.029 n=4+4) SiteNew/Many_HTML_templates/Edit-16 19.9ms ± 2% 18.3ms ± 3% -7.64% (p=0.029 n=4+4) SiteNew/Page_collections/Edit-16 37.9ms ± 4% 34.0ms ± 2% -10.28% (p=0.029 n=4+4) SiteNew/Bundle_with_image-16 10.7ms ± 0% 10.6ms ± 0% -1.15% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file-16 10.8ms ± 0% 10.7ms ± 0% -1.05% (p=0.029 n=4+4) SiteNew/Tags_and_categories-16 43.2ms ± 1% 39.6ms ± 1% -8.35% (p=0.029 n=4+4) SiteNew/Canonify_URLs-16 47.6ms ± 1% 47.3ms ± 0% ~ (p=0.057 n=4+4) SiteNew/Deep_content_tree-16 73.0ms ± 1% 74.2ms ± 1% ~ (p=0.114 n=4+4) SiteNew/Many_HTML_templates-16 37.9ms ± 0% 38.1ms ± 1% ~ (p=0.114 n=4+4) SiteNew/Page_collections-16 53.6ms ± 1% 54.7ms ± 1% +2.09% (p=0.029 n=4+4) name old alloc/op new alloc/op delta SiteNew/Bundle_with_image/Edit-16 486kB ± 0% 430kB ± 0% -11.47% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file/Edit-16 265kB ± 0% 209kB ± 0% -21.06% (p=0.029 n=4+4) SiteNew/Tags_and_categories/Edit-16 13.6MB ± 0% 8.8MB ± 0% -34.93% (p=0.029 n=4+4) SiteNew/Canonify_URLs/Edit-16 66.5MB ± 0% 63.9MB ± 0% -3.95% (p=0.029 n=4+4) SiteNew/Deep_content_tree/Edit-16 28.8MB ± 0% 25.8MB ± 0% -10.55% (p=0.029 n=4+4) SiteNew/Many_HTML_templates/Edit-16 6.16MB ± 0% 5.56MB ± 0% -9.86% (p=0.029 n=4+4) SiteNew/Page_collections/Edit-16 16.9MB ± 0% 16.0MB ± 0% -5.19% (p=0.029 n=4+4) SiteNew/Bundle_with_image-16 2.28MB ± 0% 2.29MB ± 0% +0.35% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file-16 2.07MB ± 0% 2.07MB ± 0% ~ (p=0.114 n=4+4) SiteNew/Tags_and_categories-16 14.3MB ± 0% 13.2MB ± 0% -7.30% (p=0.029 n=4+4) SiteNew/Canonify_URLs-16 69.1MB ± 0% 69.0MB ± 0% ~ (p=0.343 n=4+4) SiteNew/Deep_content_tree-16 31.3MB ± 0% 31.8MB ± 0% +1.49% (p=0.029 n=4+4) SiteNew/Many_HTML_templates-16 10.8MB ± 0% 10.9MB ± 0% +1.11% (p=0.029 n=4+4) SiteNew/Page_collections-16 21.4MB ± 0% 21.6MB ± 0% +1.15% (p=0.029 n=4+4) name old allocs/op new allocs/op delta SiteNew/Bundle_with_image/Edit-16 4.74k ± 0% 3.86k ± 0% -18.57% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file/Edit-16 4.73k ± 0% 3.85k ± 0% -18.58% (p=0.029 n=4+4) SiteNew/Tags_and_categories/Edit-16 301k ± 0% 198k ± 0% -34.14% (p=0.029 n=4+4) SiteNew/Canonify_URLs/Edit-16 389k ± 0% 373k ± 0% -4.07% (p=0.029 n=4+4) SiteNew/Deep_content_tree/Edit-16 338k ± 0% 262k ± 0% -22.63% (p=0.029 n=4+4) SiteNew/Many_HTML_templates/Edit-16 102k ± 0% 88k ± 0% -13.81% (p=0.029 n=4+4) SiteNew/Page_collections/Edit-16 176k ± 0% 152k ± 0% -13.32% (p=0.029 n=4+4) SiteNew/Bundle_with_image-16 26.8k ± 0% 26.8k ± 0% +0.05% (p=0.029 n=4+4) SiteNew/Bundle_with_JSON_file-16 26.8k ± 0% 26.8k ± 0% +0.05% (p=0.029 n=4+4) SiteNew/Tags_and_categories-16 273k ± 0% 245k ± 0% -10.36% (p=0.029 n=4+4) SiteNew/Canonify_URLs-16 396k ± 0% 398k ± 0% +0.39% (p=0.029 n=4+4) SiteNew/Deep_content_tree-16 317k ± 0% 325k ± 0% +2.53% (p=0.029 n=4+4) SiteNew/Many_HTML_templates-16 146k ± 0% 147k ± 0% +0.98% (p=0.029 n=4+4) SiteNew/Page_collections-16 210k ± 0% 215k ± 0% +2.44% (p=0.029 n=4+4) ``` Fixes #6312 Fixes #6087 Fixes #6738 Fixes #6412 Fixes #6743 Fixes #6875 Fixes #6034 Fixes #6902 Fixes #6173 Fixes #6590
662 lines
19 KiB
Go
662 lines
19 KiB
Go
// Copyright 2016 The Hugo Authors. All rights reserved.
|
|
//
|
|
// Licensed under the Apache 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://www.apache.org/licenses/LICENSE-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
|
|
|
|
import (
|
|
"fmt"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/gohugoio/hugo/identity"
|
|
|
|
qt "github.com/frankban/quicktest"
|
|
"github.com/gohugoio/hugo/deps"
|
|
"github.com/gohugoio/hugo/hugofs"
|
|
"github.com/gohugoio/hugo/tpl"
|
|
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
func TestTemplateLookupOrder(t *testing.T) {
|
|
var (
|
|
fs *hugofs.Fs
|
|
cfg *viper.Viper
|
|
th testHelper
|
|
)
|
|
|
|
// Variants base templates:
|
|
// 1. <current-path>/<template-name>-baseof.<suffix>, e.g. list-baseof.<suffix>.
|
|
// 2. <current-path>/baseof.<suffix>
|
|
// 3. _default/<template-name>-baseof.<suffix>, e.g. list-baseof.<suffix>.
|
|
// 4. _default/baseof.<suffix>
|
|
for _, this := range []struct {
|
|
name string
|
|
setup func(t *testing.T)
|
|
assert func(t *testing.T)
|
|
}{
|
|
{
|
|
"Variant 1",
|
|
func(t *testing.T) {
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect1-baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect1.html"), `{{define "main"}}sect{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base: sect")
|
|
},
|
|
},
|
|
{
|
|
"Variant 2",
|
|
func(t *testing.T) {
|
|
writeSource(t, fs, filepath.Join("layouts", "baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "index.html"), `{{define "main"}}index{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "index.html"), "Base: index")
|
|
},
|
|
},
|
|
{
|
|
"Variant 3",
|
|
func(t *testing.T) {
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "list-baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "list.html"), `{{define "main"}}list{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base: list")
|
|
},
|
|
},
|
|
{
|
|
"Variant 4",
|
|
func(t *testing.T) {
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "list.html"), `{{define "main"}}list{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base: list")
|
|
},
|
|
},
|
|
{
|
|
"Variant 1, theme, use site base",
|
|
func(t *testing.T) {
|
|
cfg.Set("theme", "mytheme")
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect1-baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "section", "sect-baseof.html"), `Base Theme: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect1.html"), `{{define "main"}}sect{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base: sect")
|
|
},
|
|
},
|
|
{
|
|
"Variant 1, theme, use theme base",
|
|
func(t *testing.T) {
|
|
cfg.Set("theme", "mytheme")
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "section", "sect1-baseof.html"), `Base Theme: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect1.html"), `{{define "main"}}sect{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base Theme: sect")
|
|
},
|
|
},
|
|
{
|
|
"Variant 4, theme, use site base",
|
|
func(t *testing.T) {
|
|
cfg.Set("theme", "mytheme")
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "baseof.html"), `Base Theme: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "list.html"), `{{define "main"}}list{{ end }}`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "index.html"), `{{define "main"}}index{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base: list")
|
|
th.assertFileContent(filepath.Join("public", "index.html"), "Base: index") // Issue #3505
|
|
},
|
|
},
|
|
{
|
|
"Variant 4, theme, use themes base",
|
|
func(t *testing.T) {
|
|
cfg.Set("theme", "mytheme")
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "baseof.html"), `Base Theme: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "list.html"), `{{define "main"}}list{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base Theme: list")
|
|
},
|
|
},
|
|
{
|
|
// Issue #3116
|
|
"Test section list and single template selection",
|
|
func(t *testing.T) {
|
|
cfg.Set("theme", "mytheme")
|
|
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "baseof.html"), `Base: {{block "main" .}}block{{end}}`)
|
|
|
|
// Both single and list template in /SECTION/
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "sect1", "list.html"), `sect list`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "list.html"), `default list`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "sect1", "single.html"), `sect single`)
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "_default", "single.html"), `default single`)
|
|
|
|
// sect2 with list template in /section
|
|
writeSource(t, fs, filepath.Join("themes", "mytheme", "layouts", "section", "sect2.html"), `sect2 list`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "sect list")
|
|
th.assertFileContent(filepath.Join("public", "sect1", "page1", "index.html"), "sect single")
|
|
th.assertFileContent(filepath.Join("public", "sect2", "index.html"), "sect2 list")
|
|
},
|
|
},
|
|
{
|
|
// Issue #2995
|
|
"Test section list and single template selection with base template",
|
|
func(t *testing.T) {
|
|
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "baseof.html"), `Base Default: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "sect1", "baseof.html"), `Base Sect1: {{block "main" .}}block{{end}}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect2-baseof.html"), `Base Sect2: {{block "main" .}}block{{end}}`)
|
|
|
|
// Both single and list + base template in /SECTION/
|
|
writeSource(t, fs, filepath.Join("layouts", "sect1", "list.html"), `{{define "main"}}sect1 list{{ end }}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "list.html"), `{{define "main"}}default list{{ end }}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "sect1", "single.html"), `{{define "main"}}sect single{{ end }}`)
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "single.html"), `{{define "main"}}default single{{ end }}`)
|
|
|
|
// sect2 with list template in /section
|
|
writeSource(t, fs, filepath.Join("layouts", "section", "sect2.html"), `{{define "main"}}sect2 list{{ end }}`)
|
|
|
|
},
|
|
func(t *testing.T) {
|
|
th.assertFileContent(filepath.Join("public", "sect1", "index.html"), "Base Sect1", "sect1 list")
|
|
th.assertFileContent(filepath.Join("public", "sect1", "page1", "index.html"), "Base Sect1", "sect single")
|
|
th.assertFileContent(filepath.Join("public", "sect2", "index.html"), "Base Sect2", "sect2 list")
|
|
|
|
// Note that this will get the default base template and not the one in /sect2 -- because there are no
|
|
// single template defined in /sect2.
|
|
th.assertFileContent(filepath.Join("public", "sect2", "page2", "index.html"), "Base Default", "default single")
|
|
},
|
|
},
|
|
} {
|
|
|
|
this := this
|
|
t.Run(this.name, func(t *testing.T) {
|
|
// TODO(bep) there are some function vars need to pull down here to enable => t.Parallel()
|
|
cfg, fs = newTestCfg()
|
|
th = newTestHelper(cfg, fs, t)
|
|
|
|
for i := 1; i <= 3; i++ {
|
|
writeSource(t, fs, filepath.Join("content", fmt.Sprintf("sect%d", i), fmt.Sprintf("page%d.md", i)), `---
|
|
title: Template test
|
|
---
|
|
Some content
|
|
`)
|
|
}
|
|
|
|
this.setup(t)
|
|
|
|
buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{})
|
|
//helpers.PrintFs(s.BaseFs.Layouts.Fs, "", os.Stdout)
|
|
this.assert(t)
|
|
})
|
|
|
|
}
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/4895
|
|
func TestTemplateBOM(t *testing.T) {
|
|
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
bom := "\ufeff"
|
|
|
|
b.WithTemplatesAdded(
|
|
"_default/baseof.html", bom+`
|
|
Base: {{ block "main" . }}base main{{ end }}`,
|
|
"_default/single.html", bom+`{{ define "main" }}Hi!?{{ end }}`)
|
|
|
|
b.WithContent("page.md", `---
|
|
title: "Page"
|
|
---
|
|
|
|
Page Content
|
|
`)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/page/index.html", "Base: Hi!?")
|
|
|
|
}
|
|
|
|
func TestTemplateLateTemplates(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile().Running()
|
|
|
|
numPages := 500 // To get some parallelism
|
|
homeTempl := `
|
|
Len RegularPages: {{ len site.RegularPages }}
|
|
{{ range site.RegularPages }}
|
|
Link: {{ .RelPermalink }} Len Content: {{ len .Content }}
|
|
{{ end }}
|
|
`
|
|
pageTemplate := `<!doctype html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<title>{{ .RelPermalink }}</title>
|
|
<meta name="description" content="The HTML5 Herald">
|
|
<meta name="author" content="SitePoint">
|
|
<link rel="stylesheet" href="css/styles.css?v=1.0">
|
|
</head>
|
|
<body>
|
|
<h1>{{ .RelPermalink }}</h1>
|
|
<p>Shortcode: {{< shortcode >}}</p>
|
|
<p>Partial: {{ partial "mypartial.html" . }}</p>
|
|
<script src="js/scripts.js"></script>
|
|
</body>
|
|
</html>
|
|
`
|
|
|
|
b.WithTemplatesAdded(
|
|
"index.html", homeTempl,
|
|
"partials/mypartial.html", `this my partial`,
|
|
)
|
|
|
|
// Make sure we get some parallelism.
|
|
for i := 0; i < numPages; i++ {
|
|
b.WithContent(fmt.Sprintf("page%d.html", i+1), pageTemplate)
|
|
}
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/index.html", fmt.Sprintf(`
|
|
Len RegularPages: %d
|
|
Link: /page3/ Len Content: 0
|
|
Link: /page22/ Len Content: 0
|
|
`, numPages))
|
|
|
|
for i := 0; i < numPages; i++ {
|
|
b.AssertFileContent(fmt.Sprintf("public/page%d/index.html", i+1),
|
|
fmt.Sprintf(`<title>/page%d/</title>`, i+1),
|
|
`<p>Shortcode: Shortcode: Hello</p>`,
|
|
"<p>Partial: this my partial</p>",
|
|
)
|
|
}
|
|
|
|
b.EditFiles(
|
|
"layouts/partials/mypartial.html", `this my changed partial`,
|
|
"layouts/index.html", (homeTempl + "CHANGED"),
|
|
)
|
|
for i := 0; i < 5; i++ {
|
|
b.EditFiles(fmt.Sprintf("content/page%d.html", i+1), pageTemplate+"CHANGED")
|
|
}
|
|
|
|
b.Build(BuildCfg{})
|
|
b.AssertFileContent("public/index.html", fmt.Sprintf(`
|
|
Len RegularPages: %d
|
|
Link: /page3/ Len Content: 0
|
|
Link: /page2/ Len Content: 0
|
|
CHANGED
|
|
`, numPages))
|
|
for i := 0; i < 5; i++ {
|
|
b.AssertFileContent(fmt.Sprintf("public/page%d/index.html", i+1),
|
|
fmt.Sprintf(`<title>/page%d/</title>`, i+1),
|
|
`<p>Shortcode: Shortcode: Hello</p>`,
|
|
"<p>Partial: this my changed partial</p>",
|
|
"CHANGED",
|
|
)
|
|
}
|
|
|
|
}
|
|
|
|
func TestTemplateManyBaseTemplates(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
|
|
numPages := 100 // To get some parallelism
|
|
|
|
pageTemplate := `---
|
|
title: "Page %d"
|
|
layout: "layout%d"
|
|
---
|
|
|
|
Content.
|
|
`
|
|
|
|
singleTemplate := `
|
|
{{ define "main" }}%d{{ end }}
|
|
`
|
|
baseTemplate := `
|
|
Base %d: {{ block "main" . }}FOO{{ end }}
|
|
`
|
|
|
|
for i := 0; i < numPages; i++ {
|
|
id := i + 1
|
|
b.WithContent(fmt.Sprintf("page%d.md", id), fmt.Sprintf(pageTemplate, id, id))
|
|
b.WithTemplates(fmt.Sprintf("_default/layout%d.html", id), fmt.Sprintf(singleTemplate, id))
|
|
b.WithTemplates(fmt.Sprintf("_default/layout%d-baseof.html", id), fmt.Sprintf(baseTemplate, id))
|
|
}
|
|
|
|
b.Build(BuildCfg{})
|
|
for i := 0; i < numPages; i++ {
|
|
id := i + 1
|
|
b.AssertFileContent(fmt.Sprintf("public/page%d/index.html", id), fmt.Sprintf(`Base %d: %d`, id, id))
|
|
}
|
|
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/6790
|
|
func TestTemplateNoBasePlease(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
|
|
b.WithTemplates("_default/list.html", `
|
|
{{ define "main" }}
|
|
Bonjour
|
|
{{ end }}
|
|
|
|
{{ printf "list" }}
|
|
|
|
|
|
`)
|
|
|
|
b.WithTemplates(
|
|
"_default/single.html", `
|
|
{{ printf "single" }}
|
|
{{ define "main" }}
|
|
Bonjour
|
|
{{ end }}
|
|
|
|
|
|
`)
|
|
|
|
b.WithContent("blog/p1.md", `---
|
|
title: The Page
|
|
---
|
|
`)
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/blog/p1/index.html", `single`)
|
|
b.AssertFileContent("public/blog/index.html", `list`)
|
|
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/6816
|
|
func TestTemplateBaseWithComment(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
b.WithTemplatesAdded(
|
|
"baseof.html", `Base: {{ block "main" . }}{{ end }}`,
|
|
"index.html", `
|
|
{{/* A comment */}}
|
|
{{ define "main" }}
|
|
Bonjour
|
|
{{ end }}
|
|
|
|
|
|
`)
|
|
|
|
b.Build(BuildCfg{})
|
|
b.AssertFileContent("public/index.html", `Base:
|
|
Bonjour`)
|
|
|
|
}
|
|
|
|
func TestTemplateLookupSite(t *testing.T) {
|
|
t.Run("basic", func(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
b.WithTemplates(
|
|
"_default/single.html", `Single: {{ .Title }}`,
|
|
"_default/list.html", `List: {{ .Title }}`,
|
|
)
|
|
|
|
createContent := func(title string) string {
|
|
return fmt.Sprintf(`---
|
|
title: %s
|
|
---`, title)
|
|
}
|
|
|
|
b.WithContent(
|
|
"_index.md", createContent("Home Sweet Home"),
|
|
"p1.md", createContent("P1"))
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
b.AssertFileContent("public/index.html", `List: Home Sweet Home`)
|
|
b.AssertFileContent("public/p1/index.html", `Single: P1`)
|
|
})
|
|
|
|
t.Run("baseof", func(t *testing.T) {
|
|
t.Parallel()
|
|
b := newTestSitesBuilder(t).WithDefaultMultiSiteConfig()
|
|
|
|
b.WithTemplatesAdded(
|
|
"index.html", `{{ define "main" }}Main Home En{{ end }}`,
|
|
"index.fr.html", `{{ define "main" }}Main Home Fr{{ end }}`,
|
|
"baseof.html", `Baseof en: {{ block "main" . }}main block{{ end }}`,
|
|
"baseof.fr.html", `Baseof fr: {{ block "main" . }}main block{{ end }}`,
|
|
"mysection/baseof.html", `Baseof mysection: {{ block "main" . }}mysection block{{ end }}`,
|
|
"_default/single.html", `{{ define "main" }}Main Default Single{{ end }}`,
|
|
"_default/list.html", `{{ define "main" }}Main Default List{{ end }}`,
|
|
)
|
|
|
|
b.WithContent("mysection/p1.md", `---
|
|
title: My Page
|
|
---
|
|
|
|
`)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/en/index.html", `Baseof en: Main Home En`)
|
|
b.AssertFileContent("public/fr/index.html", `Baseof fr: Main Home Fr`)
|
|
b.AssertFileContent("public/en/mysection/index.html", `Baseof mysection: Main Default List`)
|
|
b.AssertFileContent("public/en/mysection/p1/index.html", `Baseof mysection: Main Default Single`)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
func TestTemplateFuncs(t *testing.T) {
|
|
|
|
b := newTestSitesBuilder(t).WithDefaultMultiSiteConfig()
|
|
|
|
homeTpl := `Site: {{ site.Language.Lang }} / {{ .Site.Language.Lang }} / {{ site.BaseURL }}
|
|
Sites: {{ site.Sites.First.Home.Language.Lang }}
|
|
Hugo: {{ hugo.Generator }}
|
|
`
|
|
|
|
b.WithTemplatesAdded(
|
|
"index.html", homeTpl,
|
|
"index.fr.html", homeTpl,
|
|
)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/en/index.html",
|
|
"Site: en / en / http://example.com/blog",
|
|
"Sites: en",
|
|
"Hugo: <meta name=\"generator\" content=\"Hugo")
|
|
b.AssertFileContent("public/fr/index.html",
|
|
"Site: fr / fr / http://example.com/blog",
|
|
"Sites: en",
|
|
"Hugo: <meta name=\"generator\" content=\"Hugo",
|
|
)
|
|
|
|
}
|
|
|
|
func TestPartialWithReturn(t *testing.T) {
|
|
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile()
|
|
|
|
b.WithTemplatesAdded(
|
|
"index.html", `
|
|
Test Partials With Return Values:
|
|
|
|
add42: 50: {{ partial "add42.tpl" 8 }}
|
|
dollarContext: 60: {{ partial "dollarContext.tpl" 18 }}
|
|
adder: 70: {{ partial "dict.tpl" (dict "adder" 28) }}
|
|
complex: 80: {{ partial "complex.tpl" 38 }}
|
|
`,
|
|
"partials/add42.tpl", `
|
|
{{ $v := add . 42 }}
|
|
{{ return $v }}
|
|
`,
|
|
"partials/dollarContext.tpl", `
|
|
{{ $v := add $ 42 }}
|
|
{{ return $v }}
|
|
`,
|
|
"partials/dict.tpl", `
|
|
{{ $v := add $.adder 42 }}
|
|
{{ return $v }}
|
|
`,
|
|
"partials/complex.tpl", `
|
|
{{ return add . 42 }}
|
|
`,
|
|
)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/index.html",
|
|
"add42: 50: 50",
|
|
"dollarContext: 60: 60",
|
|
"adder: 70: 70",
|
|
"complex: 80: 80",
|
|
)
|
|
|
|
}
|
|
|
|
func TestPartialCached(t *testing.T) {
|
|
b := newTestSitesBuilder(t)
|
|
|
|
b.WithTemplatesAdded(
|
|
"index.html", `
|
|
{{ $key1 := (dict "a" "av" ) }}
|
|
{{ $key2 := (dict "a" "av2" ) }}
|
|
Partial cached1: {{ partialCached "p1" "input1" $key1 }}
|
|
Partial cached2: {{ partialCached "p1" "input2" $key1 }}
|
|
Partial cached3: {{ partialCached "p1" "input3" $key2 }}
|
|
`,
|
|
|
|
"partials/p1.html", `partial: {{ . }}`,
|
|
)
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/index.html", `
|
|
Partial cached1: partial: input1
|
|
Partial cached2: partial: input1
|
|
Partial cached3: partial: input3
|
|
`)
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/6615
|
|
func TestTemplateTruth(t *testing.T) {
|
|
b := newTestSitesBuilder(t)
|
|
b.WithTemplatesAdded("index.html", `
|
|
{{ $p := index site.RegularPages 0 }}
|
|
{{ $zero := $p.ExpiryDate }}
|
|
{{ $notZero := time.Now }}
|
|
|
|
if: Zero: {{ if $zero }}FAIL{{ else }}OK{{ end }}
|
|
if: Not Zero: {{ if $notZero }}OK{{ else }}Fail{{ end }}
|
|
not: Zero: {{ if not $zero }}OK{{ else }}FAIL{{ end }}
|
|
not: Not Zero: {{ if not $notZero }}FAIL{{ else }}OK{{ end }}
|
|
|
|
with: Zero {{ with $zero }}FAIL{{ else }}OK{{ end }}
|
|
|
|
`)
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/index.html", `
|
|
if: Zero: OK
|
|
if: Not Zero: OK
|
|
not: Zero: OK
|
|
not: Not Zero: OK
|
|
with: Zero OK
|
|
`)
|
|
}
|
|
|
|
func TestTemplateDependencies(t *testing.T) {
|
|
b := newTestSitesBuilder(t).Running()
|
|
|
|
b.WithTemplates("index.html", `
|
|
{{ $p := site.GetPage "p1" }}
|
|
{{ partial "p1.html" $p }}
|
|
{{ partialCached "p2.html" "foo" }}
|
|
{{ partials.Include "p3.html" "data" }}
|
|
{{ partials.IncludeCached "p4.html" "foo" }}
|
|
{{ $p := partial "p5" }}
|
|
{{ partial "sub/p6.html" }}
|
|
{{ partial "P7.html" }}
|
|
{{ template "_default/foo.html" }}
|
|
Partial nested: {{ partial "p10" }}
|
|
|
|
`,
|
|
"partials/p1.html", `ps: {{ .Render "li" }}`,
|
|
"partials/p2.html", `p2`,
|
|
"partials/p3.html", `p3`,
|
|
"partials/p4.html", `p4`,
|
|
"partials/p5.html", `p5`,
|
|
"partials/sub/p6.html", `p6`,
|
|
"partials/P7.html", `p7`,
|
|
"partials/p8.html", `p8 {{ partial "p9.html" }}`,
|
|
"partials/p9.html", `p9`,
|
|
"partials/p10.html", `p10 {{ partial "p11.html" }}`,
|
|
"partials/p11.html", `p11`,
|
|
"_default/foo.html", `foo`,
|
|
"_default/li.html", `li {{ partial "p8.html" }}`,
|
|
)
|
|
|
|
b.WithContent("p1.md", `---
|
|
title: P1
|
|
---
|
|
|
|
|
|
`)
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
s := b.H.Sites[0]
|
|
|
|
templ, found := s.lookupTemplate("index.html")
|
|
b.Assert(found, qt.Equals, true)
|
|
|
|
idset := make(map[identity.Identity]bool)
|
|
collectIdentities(idset, templ.(tpl.Info))
|
|
b.Assert(idset, qt.HasLen, 10)
|
|
|
|
}
|
|
|
|
func collectIdentities(set map[identity.Identity]bool, provider identity.Provider) {
|
|
if ids, ok := provider.(identity.IdentitiesProvider); ok {
|
|
for _, id := range ids.GetIdentities() {
|
|
collectIdentities(set, id)
|
|
}
|
|
} else {
|
|
set[provider.GetIdentity()] = true
|
|
}
|
|
}
|
|
|
|
func ident(level int) string {
|
|
return strings.Repeat(" ", level)
|
|
}
|