mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-07 20:30:36 -05:00
bceda1b288
See #5784
1448 lines
42 KiB
Go
1448 lines
42 KiB
Go
// Copyright 2019 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"
|
|
"html/template"
|
|
"os"
|
|
|
|
"github.com/gohugoio/hugo/common/loggers"
|
|
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/gohugoio/hugo/hugofs"
|
|
|
|
"github.com/gohugoio/hugo/resources/page"
|
|
"github.com/gohugoio/hugo/resources/resource"
|
|
|
|
"github.com/spf13/afero"
|
|
"github.com/spf13/viper"
|
|
|
|
"github.com/gohugoio/hugo/deps"
|
|
"github.com/gohugoio/hugo/helpers"
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
const (
|
|
homePage = "---\ntitle: Home\n---\nHome Page Content\n"
|
|
simplePage = "---\ntitle: Simple\n---\nSimple Page\n"
|
|
|
|
simplePageRFC3339Date = "---\ntitle: RFC3339 Date\ndate: \"2013-05-17T16:59:30Z\"\n---\nrfc3339 content"
|
|
|
|
simplePageWithSummaryDelimiter = `---
|
|
title: Simple
|
|
---
|
|
Summary Next Line
|
|
|
|
<!--more-->
|
|
Some more text
|
|
`
|
|
|
|
simplePageWithSummaryDelimiterAndMarkdownThatCrossesBorder = `---
|
|
title: Simple
|
|
---
|
|
The [best static site generator][hugo].[^1]
|
|
<!--more-->
|
|
[hugo]: http://gohugo.io/
|
|
[^1]: Many people say so.
|
|
`
|
|
simplePageWithShortcodeInSummary = `---
|
|
title: Simple
|
|
---
|
|
Summary Next Line. {{<figure src="/not/real" >}}.
|
|
More text here.
|
|
|
|
Some more text
|
|
`
|
|
|
|
simplePageWithEmbeddedScript = `---
|
|
title: Simple
|
|
---
|
|
<script type='text/javascript'>alert('the script tags are still there, right?');</script>
|
|
`
|
|
|
|
simplePageWithSummaryDelimiterSameLine = `---
|
|
title: Simple
|
|
---
|
|
Summary Same Line<!--more-->
|
|
|
|
Some more text
|
|
`
|
|
|
|
simplePageWithAllCJKRunes = `---
|
|
title: Simple
|
|
---
|
|
|
|
|
|
€ € € € €
|
|
你好
|
|
도형이
|
|
カテゴリー
|
|
|
|
|
|
`
|
|
|
|
simplePageWithMainEnglishWithCJKRunes = `---
|
|
title: Simple
|
|
---
|
|
|
|
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
In Chinese, 好 means good. In Chinese, 好 means good.
|
|
More then 70 words.
|
|
|
|
|
|
`
|
|
simplePageWithMainEnglishWithCJKRunesSummary = "In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good. " +
|
|
"In Chinese, 好 means good. In Chinese, 好 means good."
|
|
|
|
simplePageWithIsCJKLanguageFalse = `---
|
|
title: Simple
|
|
isCJKLanguage: false
|
|
---
|
|
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀 means good.
|
|
In Chinese, 好的啊 means good. In Chinese, 好的呀呀 means good enough.
|
|
More then 70 words.
|
|
|
|
|
|
`
|
|
simplePageWithIsCJKLanguageFalseSummary = "In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀 means good. " +
|
|
"In Chinese, 好的啊 means good. In Chinese, 好的呀呀 means good enough."
|
|
|
|
simplePageWithLongContent = `---
|
|
title: Simple
|
|
---
|
|
|
|
Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor
|
|
incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis
|
|
nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.
|
|
Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu
|
|
fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in
|
|
culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit
|
|
amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore
|
|
et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation
|
|
ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor
|
|
in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla
|
|
pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui
|
|
officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet,
|
|
consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et
|
|
dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco
|
|
laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in
|
|
reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.
|
|
Excepteur sint occaecat cupidatat non proident, sunt in culpa qui officia
|
|
deserunt mollit anim id est laborum. Lorem ipsum dolor sit amet, consectetur
|
|
adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna
|
|
aliqua. Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi
|
|
ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit in
|
|
voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint
|
|
occaecat cupidatat non proident, sunt in culpa qui officia deserunt mollit anim
|
|
id est laborum. Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed
|
|
do eiusmod tempor incididunt ut labore et dolore magna aliqua. Ut enim ad minim
|
|
veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo
|
|
consequat. Duis aute irure dolor in reprehenderit in voluptate velit esse
|
|
cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat non
|
|
proident, sunt in culpa qui officia deserunt mollit anim id est laborum. Lorem
|
|
ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor
|
|
incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis
|
|
nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.
|
|
Duis aute irure dolor in reprehenderit in voluptate velit esse cillum dolore eu
|
|
fugiat nulla pariatur. Excepteur sint occaecat cupidatat non proident, sunt in
|
|
culpa qui officia deserunt mollit anim id est laborum. Lorem ipsum dolor sit
|
|
amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore
|
|
et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation
|
|
ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor
|
|
in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla
|
|
pariatur. Excepteur sint occaecat cupidatat non proident, sunt in culpa qui
|
|
officia deserunt mollit anim id est laborum.`
|
|
|
|
pageWithToC = `---
|
|
title: TOC
|
|
---
|
|
For some moments the old man did not reply. He stood with bowed head, buried in deep thought. But at last he spoke.
|
|
|
|
## AA
|
|
|
|
I have no idea, of course, how long it took me to reach the limit of the plain,
|
|
but at last I entered the foothills, following a pretty little canyon upward
|
|
toward the mountains. Beside me frolicked a laughing brooklet, hurrying upon
|
|
its noisy way down to the silent sea. In its quieter pools I discovered many
|
|
small fish, of four-or five-pound weight I should imagine. In appearance,
|
|
except as to size and color, they were not unlike the whale of our own seas. As
|
|
I watched them playing about I discovered, not only that they suckled their
|
|
young, but that at intervals they rose to the surface to breathe as well as to
|
|
feed upon certain grasses and a strange, scarlet lichen which grew upon the
|
|
rocks just above the water line.
|
|
|
|
### AAA
|
|
|
|
I remember I felt an extraordinary persuasion that I was being played with,
|
|
that presently, when I was upon the very verge of safety, this mysterious
|
|
death--as swift as the passage of light--would leap after me from the pit about
|
|
the cylinder and strike me down. ## BB
|
|
|
|
### BBB
|
|
|
|
"You're a great Granser," he cried delightedly, "always making believe them little marks mean something."
|
|
`
|
|
|
|
simplePageWithAdditionalExtension = `+++
|
|
[blackfriday]
|
|
extensions = ["hardLineBreak"]
|
|
+++
|
|
first line.
|
|
second line.
|
|
|
|
fourth line.
|
|
`
|
|
|
|
simplePageWithURL = `---
|
|
title: Simple
|
|
url: simple/url/
|
|
---
|
|
Simple Page With URL`
|
|
|
|
simplePageWithSlug = `---
|
|
title: Simple
|
|
slug: simple-slug
|
|
---
|
|
Simple Page With Slug`
|
|
|
|
simplePageWithDate = `---
|
|
title: Simple
|
|
date: '2013-10-15T06:16:13'
|
|
---
|
|
Simple Page With Date`
|
|
|
|
UTF8Page = `---
|
|
title: ラーメン
|
|
---
|
|
UTF8 Page`
|
|
|
|
UTF8PageWithURL = `---
|
|
title: ラーメン
|
|
url: ラーメン/url/
|
|
---
|
|
UTF8 Page With URL`
|
|
|
|
UTF8PageWithSlug = `---
|
|
title: ラーメン
|
|
slug: ラーメン-slug
|
|
---
|
|
UTF8 Page With Slug`
|
|
|
|
UTF8PageWithDate = `---
|
|
title: ラーメン
|
|
date: '2013-10-15T06:16:13'
|
|
---
|
|
UTF8 Page With Date`
|
|
)
|
|
|
|
func checkPageTitle(t *testing.T, page page.Page, title string) {
|
|
if page.Title() != title {
|
|
t.Fatalf("Page title is: %s. Expected %s", page.Title(), title)
|
|
}
|
|
}
|
|
|
|
func checkPageContent(t *testing.T, page page.Page, expected string, msg ...interface{}) {
|
|
a := normalizeContent(expected)
|
|
b := normalizeContent(content(page))
|
|
if a != b {
|
|
t.Log(stackTrace())
|
|
t.Fatalf("Page content is:\n%q\nExpected:\n%q (%q)", b, a, msg)
|
|
}
|
|
}
|
|
|
|
func normalizeContent(c string) string {
|
|
norm := c
|
|
norm = strings.Replace(norm, "\n", " ", -1)
|
|
norm = strings.Replace(norm, " ", " ", -1)
|
|
norm = strings.Replace(norm, " ", " ", -1)
|
|
norm = strings.Replace(norm, " ", " ", -1)
|
|
norm = strings.Replace(norm, "p> ", "p>", -1)
|
|
norm = strings.Replace(norm, "> <", "> <", -1)
|
|
return strings.TrimSpace(norm)
|
|
}
|
|
|
|
func checkPageTOC(t *testing.T, page page.Page, toc string) {
|
|
if page.TableOfContents() != template.HTML(toc) {
|
|
t.Fatalf("Page TableOfContents is: %q.\nExpected %q", page.TableOfContents(), toc)
|
|
}
|
|
}
|
|
|
|
func checkPageSummary(t *testing.T, page page.Page, summary string, msg ...interface{}) {
|
|
a := normalizeContent(string(page.Summary()))
|
|
b := normalizeContent(summary)
|
|
if a != b {
|
|
t.Fatalf("Page summary is:\n%q.\nExpected\n%q (%q)", a, b, msg)
|
|
}
|
|
}
|
|
|
|
func checkPageType(t *testing.T, page page.Page, pageType string) {
|
|
if page.Type() != pageType {
|
|
t.Fatalf("Page type is: %s. Expected: %s", page.Type(), pageType)
|
|
}
|
|
}
|
|
|
|
func checkPageDate(t *testing.T, page page.Page, time time.Time) {
|
|
if page.Date() != time {
|
|
t.Fatalf("Page date is: %s. Expected: %s", page.Date(), time)
|
|
}
|
|
}
|
|
|
|
func normalizeExpected(ext, str string) string {
|
|
str = normalizeContent(str)
|
|
switch ext {
|
|
default:
|
|
return str
|
|
case "html":
|
|
return strings.Trim(helpers.StripHTML(str), " ")
|
|
case "ad":
|
|
paragraphs := strings.Split(str, "</p>")
|
|
expected := ""
|
|
for _, para := range paragraphs {
|
|
if para == "" {
|
|
continue
|
|
}
|
|
expected += fmt.Sprintf("<div class=\"paragraph\">\n%s</p></div>\n", para)
|
|
}
|
|
|
|
return expected
|
|
case "rst":
|
|
return fmt.Sprintf("<div class=\"document\">\n\n\n%s</div>", str)
|
|
}
|
|
}
|
|
|
|
func testAllMarkdownEnginesForPages(t *testing.T,
|
|
assertFunc func(t *testing.T, ext string, pages page.Pages), settings map[string]interface{}, pageSources ...string) {
|
|
|
|
engines := []struct {
|
|
ext string
|
|
shouldExecute func() bool
|
|
}{
|
|
{"md", func() bool { return true }},
|
|
{"mmark", func() bool { return true }},
|
|
{"ad", func() bool { return helpers.HasAsciidoc() }},
|
|
{"rst", func() bool { return helpers.HasRst() }},
|
|
}
|
|
|
|
for _, e := range engines {
|
|
if !e.shouldExecute() {
|
|
continue
|
|
}
|
|
|
|
cfg, fs := newTestCfg()
|
|
|
|
for k, v := range settings {
|
|
cfg.Set(k, v)
|
|
}
|
|
|
|
contentDir := "content"
|
|
|
|
if s := cfg.GetString("contentDir"); s != "" {
|
|
contentDir = s
|
|
}
|
|
|
|
var fileSourcePairs []string
|
|
|
|
for i, source := range pageSources {
|
|
fileSourcePairs = append(fileSourcePairs, fmt.Sprintf("p%d.%s", i, e.ext), source)
|
|
}
|
|
|
|
for i := 0; i < len(fileSourcePairs); i += 2 {
|
|
writeSource(t, fs, filepath.Join(contentDir, fileSourcePairs[i]), fileSourcePairs[i+1])
|
|
}
|
|
|
|
// Add a content page for the home page
|
|
homePath := fmt.Sprintf("_index.%s", e.ext)
|
|
writeSource(t, fs, filepath.Join(contentDir, homePath), homePage)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), len(pageSources))
|
|
|
|
assertFunc(t, e.ext, s.RegularPages())
|
|
|
|
home, err := s.Info.Home()
|
|
require.NoError(t, err)
|
|
require.NotNil(t, home)
|
|
require.Equal(t, homePath, home.File().Path())
|
|
require.Contains(t, content(home), "Home Page Content")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Issue #1076
|
|
func TestPageWithDelimiterForMarkdownThatCrossesBorder(t *testing.T) {
|
|
t.Parallel()
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "simple.md"), simplePageWithSummaryDelimiterAndMarkdownThatCrossesBorder)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
p := s.RegularPages()[0]
|
|
|
|
if p.Summary() != template.HTML(
|
|
"<p>The <a href=\"http://gohugo.io/\">best static site generator</a>.<sup class=\"footnote-ref\" id=\"fnref:1\"><a href=\"#fn:1\">1</a></sup></p>") {
|
|
t.Fatalf("Got summary:\n%q", p.Summary())
|
|
}
|
|
|
|
c := content(p)
|
|
if c != "<p>The <a href=\"http://gohugo.io/\">best static site generator</a>.<sup class=\"footnote-ref\" id=\"fnref:1\"><a href=\"#fn:1\">1</a></sup></p>\n\n<div class=\"footnotes\">\n\n<hr />\n\n<ol>\n<li id=\"fn:1\">Many people say so.\n <a class=\"footnote-return\" href=\"#fnref:1\"><sup>[return]</sup></a></li>\n</ol>\n</div>" {
|
|
t.Fatalf("Got content:\n%q", c)
|
|
}
|
|
}
|
|
|
|
func TestPageDatesAllKinds(t *testing.T) {
|
|
t.Parallel()
|
|
assert := assert.New(t)
|
|
|
|
pageContent := `
|
|
---
|
|
title: Page
|
|
date: 2017-01-15
|
|
tags: ["hugo"]
|
|
categories: ["cool stuff"]
|
|
---
|
|
`
|
|
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile().WithContent("page.md", pageContent)
|
|
b.WithSimpleConfigFile().WithContent("blog/page.md", pageContent)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
assert.Equal(1, len(b.H.Sites))
|
|
s := b.H.Sites[0]
|
|
|
|
checkDate := func(t time.Time, msg string) {
|
|
assert.Equal(2017, t.Year(), msg)
|
|
}
|
|
|
|
checkDated := func(d resource.Dated, msg string) {
|
|
checkDate(d.Date(), "date: "+msg)
|
|
checkDate(d.Lastmod(), "lastmod: "+msg)
|
|
}
|
|
for _, p := range s.Pages() {
|
|
checkDated(p, p.Kind())
|
|
}
|
|
checkDate(s.Info.LastChange(), "site")
|
|
|
|
}
|
|
|
|
func TestCreateNewPage(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
|
|
// issue #2290: Path is relative to the content dir and will continue to be so.
|
|
require.Equal(t, filepath.FromSlash(fmt.Sprintf("p0.%s", ext)), p.File().Path())
|
|
assert.False(t, p.IsHome())
|
|
checkPageTitle(t, p, "Simple")
|
|
checkPageContent(t, p, normalizeExpected(ext, "<p>Simple Page</p>\n"))
|
|
checkPageSummary(t, p, "Simple Page")
|
|
checkPageType(t, p, "page")
|
|
}
|
|
|
|
settings := map[string]interface{}{
|
|
"contentDir": "mycontent",
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, settings, simplePage)
|
|
}
|
|
|
|
func TestPageWithDelimiter(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
checkPageTitle(t, p, "Simple")
|
|
checkPageContent(t, p, normalizeExpected(ext, "<p>Summary Next Line</p>\n\n<p>Some more text</p>\n"), ext)
|
|
checkPageSummary(t, p, normalizeExpected(ext, "<p>Summary Next Line</p>"), ext)
|
|
checkPageType(t, p, "page")
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithSummaryDelimiter)
|
|
}
|
|
|
|
// Issue #3854
|
|
// Also see https://github.com/gohugoio/hugo/issues/3977
|
|
func TestPageWithDateFields(t *testing.T) {
|
|
assert := require.New(t)
|
|
pageWithDate := `---
|
|
title: P%d
|
|
weight: %d
|
|
%s: 2017-10-13
|
|
---
|
|
Simple Page With Some Date`
|
|
|
|
hasDate := func(p page.Page) bool {
|
|
return p.Date().Year() == 2017
|
|
}
|
|
|
|
datePage := func(field string, weight int) string {
|
|
return fmt.Sprintf(pageWithDate, weight, weight, field)
|
|
}
|
|
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
assert.True(len(pages) > 0)
|
|
for _, p := range pages {
|
|
assert.True(hasDate(p))
|
|
}
|
|
|
|
}
|
|
|
|
fields := []string{"date", "publishdate", "pubdate", "published"}
|
|
pageContents := make([]string, len(fields))
|
|
for i, field := range fields {
|
|
pageContents[i] = datePage(field, i+1)
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, pageContents...)
|
|
}
|
|
|
|
// Issue #2601
|
|
func TestPageRawContent(t *testing.T) {
|
|
t.Parallel()
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "raw.md"), `---
|
|
title: Raw
|
|
---
|
|
**Raw**`)
|
|
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "single.html"), `{{ .RawContent }}`)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
p := s.RegularPages()[0]
|
|
|
|
require.Equal(t, p.RawContent(), "**Raw**")
|
|
|
|
}
|
|
|
|
func TestPageWithShortCodeInSummary(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
checkPageTitle(t, p, "Simple")
|
|
checkPageContent(t, p, normalizeExpected(ext, "<p>Summary Next Line. <figure> <img src=\"/not/real\"/> </figure> . More text here.</p><p>Some more text</p>"))
|
|
checkPageSummary(t, p, "Summary Next Line. . More text here. Some more text")
|
|
checkPageType(t, p, "page")
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithShortcodeInSummary)
|
|
}
|
|
|
|
func TestPageWithEmbeddedScriptTag(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if ext == "ad" || ext == "rst" {
|
|
// TOD(bep)
|
|
return
|
|
}
|
|
checkPageContent(t, p, "<script type='text/javascript'>alert('the script tags are still there, right?');</script>\n", ext)
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithEmbeddedScript)
|
|
}
|
|
|
|
func TestPageWithAdditionalExtension(t *testing.T) {
|
|
t.Parallel()
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "simple.md"), simplePageWithAdditionalExtension)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
p := s.RegularPages()[0]
|
|
|
|
checkPageContent(t, p, "<p>first line.<br />\nsecond line.</p>\n\n<p>fourth line.</p>\n")
|
|
}
|
|
|
|
func TestTableOfContents(t *testing.T) {
|
|
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "tocpage.md"), pageWithToC)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
p := s.RegularPages()[0]
|
|
|
|
checkPageContent(t, p, "\n\n<p>For some moments the old man did not reply. He stood with bowed head, buried in deep thought. But at last he spoke.</p>\n\n<h2 id=\"aa\">AA</h2>\n\n<p>I have no idea, of course, how long it took me to reach the limit of the plain,\nbut at last I entered the foothills, following a pretty little canyon upward\ntoward the mountains. Beside me frolicked a laughing brooklet, hurrying upon\nits noisy way down to the silent sea. In its quieter pools I discovered many\nsmall fish, of four-or five-pound weight I should imagine. In appearance,\nexcept as to size and color, they were not unlike the whale of our own seas. As\nI watched them playing about I discovered, not only that they suckled their\nyoung, but that at intervals they rose to the surface to breathe as well as to\nfeed upon certain grasses and a strange, scarlet lichen which grew upon the\nrocks just above the water line.</p>\n\n<h3 id=\"aaa\">AAA</h3>\n\n<p>I remember I felt an extraordinary persuasion that I was being played with,\nthat presently, when I was upon the very verge of safety, this mysterious\ndeath–as swift as the passage of light–would leap after me from the pit about\nthe cylinder and strike me down. ## BB</p>\n\n<h3 id=\"bbb\">BBB</h3>\n\n<p>“You’re a great Granser,” he cried delightedly, “always making believe them little marks mean something.”</p>\n")
|
|
checkPageTOC(t, p, "<nav id=\"TableOfContents\">\n<ul>\n<li>\n<ul>\n<li><a href=\"#aa\">AA</a>\n<ul>\n<li><a href=\"#aaa\">AAA</a></li>\n<li><a href=\"#bbb\">BBB</a></li>\n</ul></li>\n</ul></li>\n</ul>\n</nav>")
|
|
}
|
|
|
|
func TestPageWithMoreTag(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
checkPageTitle(t, p, "Simple")
|
|
checkPageContent(t, p, normalizeExpected(ext, "<p>Summary Same Line</p>\n\n<p>Some more text</p>\n"))
|
|
checkPageSummary(t, p, normalizeExpected(ext, "<p>Summary Same Line</p>"))
|
|
checkPageType(t, p, "page")
|
|
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithSummaryDelimiterSameLine)
|
|
}
|
|
|
|
// #2973
|
|
func TestSummaryWithHTMLTagsOnNextLine(t *testing.T) {
|
|
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
require.Contains(t, p.Summary(), "Happy new year everyone!")
|
|
require.NotContains(t, p.Summary(), "User interface")
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, `---
|
|
title: Simple
|
|
---
|
|
Happy new year everyone!
|
|
|
|
Here is the last report for commits in the year 2016. It covers hrev50718-hrev50829.
|
|
|
|
<!--more-->
|
|
|
|
<h3>User interface</h3>
|
|
|
|
`)
|
|
}
|
|
|
|
func TestPageWithDate(t *testing.T) {
|
|
t.Parallel()
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "simple.md"), simplePageRFC3339Date)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
p := s.RegularPages()[0]
|
|
d, _ := time.Parse(time.RFC3339, "2013-05-17T16:59:30Z")
|
|
|
|
checkPageDate(t, p, d)
|
|
}
|
|
|
|
func TestPageWithLastmodFromGitInfo(t *testing.T) {
|
|
assrt := require.New(t)
|
|
|
|
// We need to use the OS fs for this.
|
|
cfg := viper.New()
|
|
fs := hugofs.NewFrom(hugofs.Os, cfg)
|
|
fs.Destination = &afero.MemMapFs{}
|
|
|
|
cfg.Set("frontmatter", map[string]interface{}{
|
|
"lastmod": []string{":git", "lastmod"},
|
|
})
|
|
cfg.Set("defaultContentLanguage", "en")
|
|
|
|
langConfig := map[string]interface{}{
|
|
"en": map[string]interface{}{
|
|
"weight": 1,
|
|
"languageName": "English",
|
|
"contentDir": "content",
|
|
},
|
|
"nn": map[string]interface{}{
|
|
"weight": 2,
|
|
"languageName": "Nynorsk",
|
|
"contentDir": "content_nn",
|
|
},
|
|
}
|
|
|
|
cfg.Set("languages", langConfig)
|
|
cfg.Set("enableGitInfo", true)
|
|
|
|
assrt.NoError(loadDefaultSettingsFor(cfg))
|
|
assrt.NoError(loadLanguageSettings(cfg, nil))
|
|
|
|
wd, err := os.Getwd()
|
|
assrt.NoError(err)
|
|
cfg.Set("workingDir", filepath.Join(wd, "testsite"))
|
|
|
|
h, err := NewHugoSites(deps.DepsCfg{Fs: fs, Cfg: cfg})
|
|
|
|
assrt.NoError(err)
|
|
assrt.Len(h.Sites, 2)
|
|
|
|
require.NoError(t, h.Build(BuildCfg{SkipRender: true}))
|
|
|
|
enSite := h.Sites[0]
|
|
assrt.Len(enSite.RegularPages(), 1)
|
|
|
|
// 2018-03-11 is the Git author date for testsite/content/first-post.md
|
|
assrt.Equal("2018-03-11", enSite.RegularPages()[0].Lastmod().Format("2006-01-02"))
|
|
|
|
nnSite := h.Sites[1]
|
|
assrt.Len(nnSite.RegularPages(), 1)
|
|
|
|
// 2018-08-11 is the Git author date for testsite/content_nn/first-post.md
|
|
assrt.Equal("2018-08-11", nnSite.RegularPages()[0].Lastmod().Format("2006-01-02"))
|
|
|
|
}
|
|
|
|
func TestPageWithFrontMatterConfig(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
for _, dateHandler := range []string{":filename", ":fileModTime"} {
|
|
t.Run(fmt.Sprintf("dateHandler=%q", dateHandler), func(t *testing.T) {
|
|
assrt := require.New(t)
|
|
cfg, fs := newTestCfg()
|
|
|
|
pageTemplate := `
|
|
---
|
|
title: Page
|
|
weight: %d
|
|
lastMod: 2018-02-28
|
|
%s
|
|
---
|
|
Content
|
|
`
|
|
|
|
cfg.Set("frontmatter", map[string]interface{}{
|
|
"date": []string{dateHandler, "date"},
|
|
})
|
|
|
|
c1 := filepath.Join("content", "section", "2012-02-21-noslug.md")
|
|
c2 := filepath.Join("content", "section", "2012-02-22-slug.md")
|
|
|
|
writeSource(t, fs, c1, fmt.Sprintf(pageTemplate, 1, ""))
|
|
writeSource(t, fs, c2, fmt.Sprintf(pageTemplate, 2, "slug: aslug"))
|
|
|
|
c1fi, err := fs.Source.Stat(c1)
|
|
assrt.NoError(err)
|
|
c2fi, err := fs.Source.Stat(c2)
|
|
assrt.NoError(err)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
assrt.Len(s.RegularPages(), 2)
|
|
|
|
noSlug := s.RegularPages()[0]
|
|
slug := s.RegularPages()[1]
|
|
|
|
assrt.Equal(28, noSlug.Lastmod().Day())
|
|
|
|
switch strings.ToLower(dateHandler) {
|
|
case ":filename":
|
|
assrt.False(noSlug.Date().IsZero())
|
|
assrt.False(slug.Date().IsZero())
|
|
assrt.Equal(2012, noSlug.Date().Year())
|
|
assrt.Equal(2012, slug.Date().Year())
|
|
assrt.Equal("noslug", noSlug.Slug())
|
|
assrt.Equal("aslug", slug.Slug())
|
|
case ":filemodtime":
|
|
assrt.Equal(c1fi.ModTime().Year(), noSlug.Date().Year())
|
|
assrt.Equal(c2fi.ModTime().Year(), slug.Date().Year())
|
|
fallthrough
|
|
default:
|
|
assrt.Equal("", noSlug.Slug())
|
|
assrt.Equal("aslug", slug.Slug())
|
|
|
|
}
|
|
})
|
|
}
|
|
|
|
}
|
|
|
|
func TestWordCountWithAllCJKRunesWithoutHasCJKLanguage(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if p.WordCount() != 8 {
|
|
t.Fatalf("[%s] incorrect word count. expected %v, got %v", ext, 8, p.WordCount())
|
|
}
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithAllCJKRunes)
|
|
}
|
|
|
|
func TestWordCountWithAllCJKRunesHasCJKLanguage(t *testing.T) {
|
|
t.Parallel()
|
|
settings := map[string]interface{}{"hasCJKLanguage": true}
|
|
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if p.WordCount() != 15 {
|
|
t.Fatalf("[%s] incorrect word count, expected %v, got %v", ext, 15, p.WordCount())
|
|
}
|
|
}
|
|
testAllMarkdownEnginesForPages(t, assertFunc, settings, simplePageWithAllCJKRunes)
|
|
}
|
|
|
|
func TestWordCountWithMainEnglishWithCJKRunes(t *testing.T) {
|
|
t.Parallel()
|
|
settings := map[string]interface{}{"hasCJKLanguage": true}
|
|
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if p.WordCount() != 74 {
|
|
t.Fatalf("[%s] incorrect word count, expected %v, got %v", ext, 74, p.WordCount())
|
|
}
|
|
|
|
if p.Summary() != simplePageWithMainEnglishWithCJKRunesSummary {
|
|
t.Fatalf("[%s] incorrect Summary for content '%s'. expected %v, got %v", ext, p.Plain(),
|
|
simplePageWithMainEnglishWithCJKRunesSummary, p.Summary())
|
|
}
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, settings, simplePageWithMainEnglishWithCJKRunes)
|
|
}
|
|
|
|
func TestWordCountWithIsCJKLanguageFalse(t *testing.T) {
|
|
t.Parallel()
|
|
settings := map[string]interface{}{
|
|
"hasCJKLanguage": true,
|
|
}
|
|
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if p.WordCount() != 75 {
|
|
t.Fatalf("[%s] incorrect word count for content '%s'. expected %v, got %v", ext, p.Plain(), 74, p.WordCount())
|
|
}
|
|
|
|
if p.Summary() != simplePageWithIsCJKLanguageFalseSummary {
|
|
t.Fatalf("[%s] incorrect Summary for content '%s'. expected %v, got %v", ext, p.Plain(),
|
|
simplePageWithIsCJKLanguageFalseSummary, p.Summary())
|
|
}
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, settings, simplePageWithIsCJKLanguageFalse)
|
|
|
|
}
|
|
|
|
func TestWordCount(t *testing.T) {
|
|
t.Parallel()
|
|
assertFunc := func(t *testing.T, ext string, pages page.Pages) {
|
|
p := pages[0]
|
|
if p.WordCount() != 483 {
|
|
t.Fatalf("[%s] incorrect word count. expected %v, got %v", ext, 483, p.WordCount())
|
|
}
|
|
|
|
if p.FuzzyWordCount() != 500 {
|
|
t.Fatalf("[%s] incorrect word count. expected %v, got %v", ext, 500, p.FuzzyWordCount())
|
|
}
|
|
|
|
if p.ReadingTime() != 3 {
|
|
t.Fatalf("[%s] incorrect min read. expected %v, got %v", ext, 3, p.ReadingTime())
|
|
}
|
|
|
|
}
|
|
|
|
testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithLongContent)
|
|
}
|
|
|
|
func TestPagePaths(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
siteParmalinksSetting := map[string]string{
|
|
"post": ":year/:month/:day/:title/",
|
|
}
|
|
|
|
tests := []struct {
|
|
content string
|
|
path string
|
|
hasPermalink bool
|
|
expected string
|
|
}{
|
|
{simplePage, "post/x.md", false, "post/x.html"},
|
|
{simplePageWithURL, "post/x.md", false, "simple/url/index.html"},
|
|
{simplePageWithSlug, "post/x.md", false, "post/simple-slug.html"},
|
|
{simplePageWithDate, "post/x.md", true, "2013/10/15/simple/index.html"},
|
|
{UTF8Page, "post/x.md", false, "post/x.html"},
|
|
{UTF8PageWithURL, "post/x.md", false, "ラーメン/url/index.html"},
|
|
{UTF8PageWithSlug, "post/x.md", false, "post/ラーメン-slug.html"},
|
|
{UTF8PageWithDate, "post/x.md", true, "2013/10/15/ラーメン/index.html"},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
cfg, fs := newTestCfg()
|
|
|
|
if test.hasPermalink {
|
|
cfg.Set("permalinks", siteParmalinksSetting)
|
|
}
|
|
|
|
writeSource(t, fs, filepath.Join("content", filepath.FromSlash(test.path)), test.content)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
}
|
|
}
|
|
|
|
func TestTranslationKey(t *testing.T) {
|
|
t.Parallel()
|
|
assert := require.New(t)
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", filepath.FromSlash("sect/simple.no.md")), "---\ntitle: \"A1\"\ntranslationKey: \"k1\"\n---\nContent\n")
|
|
writeSource(t, fs, filepath.Join("content", filepath.FromSlash("sect/simple.en.md")), "---\ntitle: \"A2\"\n---\nContent\n")
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 2)
|
|
|
|
home, _ := s.Info.Home()
|
|
assert.NotNil(home)
|
|
assert.Equal("home", home.TranslationKey())
|
|
assert.Equal("page/k1", s.RegularPages()[0].TranslationKey())
|
|
p2 := s.RegularPages()[1]
|
|
|
|
assert.Equal("page/sect/simple", p2.TranslationKey())
|
|
|
|
}
|
|
|
|
func TestChompBOM(t *testing.T) {
|
|
t.Parallel()
|
|
const utf8BOM = "\xef\xbb\xbf"
|
|
|
|
cfg, fs := newTestCfg()
|
|
|
|
writeSource(t, fs, filepath.Join("content", "simple.md"), utf8BOM+simplePage)
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{SkipRender: true})
|
|
|
|
require.Len(t, s.RegularPages(), 1)
|
|
|
|
p := s.RegularPages()[0]
|
|
|
|
checkPageTitle(t, p, "Simple")
|
|
}
|
|
|
|
func TestPageWithEmoji(t *testing.T) {
|
|
for _, enableEmoji := range []bool{true, false} {
|
|
v := viper.New()
|
|
v.Set("enableEmoji", enableEmoji)
|
|
b := newTestSitesBuilder(t)
|
|
b.WithViper(v)
|
|
|
|
b.WithSimpleConfigFile()
|
|
|
|
b.WithContent("page-emoji.md", `---
|
|
title: "Hugo Smile"
|
|
---
|
|
This is a :smile:.
|
|
<!--more-->
|
|
|
|
Another :smile: This is :not: :an: :emoji:.
|
|
|
|
O :christmas_tree:
|
|
|
|
Write me an :e-mail: or :email:?
|
|
|
|
Too many colons: :: ::: :::: :?: :!: :.:
|
|
|
|
If you dislike this video, you can hit that :-1: button :stuck_out_tongue_winking_eye:,
|
|
but if you like it, hit :+1: and get subscribed!
|
|
`)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
if enableEmoji {
|
|
b.AssertFileContent("public/page-emoji/index.html",
|
|
"This is a 😄",
|
|
"Another 😄",
|
|
"This is :not: :an: :emoji:.",
|
|
"O 🎄",
|
|
"Write me an 📧 or ✉️?",
|
|
"Too many colons: :: ::: :::: :?: :!: :.:",
|
|
"you can hit that 👎 button 😜,",
|
|
"hit 👍 and get subscribed!",
|
|
)
|
|
} else {
|
|
b.AssertFileContent("public/page-emoji/index.html",
|
|
"This is a :smile:",
|
|
"Another :smile:",
|
|
"This is :not: :an: :emoji:.",
|
|
"O :christmas_tree:",
|
|
"Write me an :e-mail: or :email:?",
|
|
"Too many colons: :: ::: :::: :?: :!: :.:",
|
|
"you can hit that :-1: button :stuck_out_tongue_winking_eye:,",
|
|
"hit :+1: and get subscribed!",
|
|
)
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
func TestPageHTMLContent(t *testing.T) {
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile()
|
|
|
|
frontmatter := `---
|
|
title: "HTML Content"
|
|
---
|
|
`
|
|
b.WithContent("regular.html", frontmatter+`<h1>Hugo</h1>`)
|
|
b.WithContent("noblackfridayforyou.html", frontmatter+`**Hugo!**`)
|
|
b.WithContent("manualsummary.html", frontmatter+`
|
|
<p>This is summary</p>
|
|
<!--more-->
|
|
<p>This is the main content.</p>`)
|
|
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent(
|
|
"public/regular/index.html",
|
|
"Single: HTML Content|Hello|en|RelPermalink: /regular/|",
|
|
"Summary: Hugo|Truncated: false")
|
|
|
|
b.AssertFileContent(
|
|
"public/noblackfridayforyou/index.html",
|
|
"Permalink: http://example.com/noblackfridayforyou/|**Hugo!**|",
|
|
)
|
|
|
|
// https://github.com/gohugoio/hugo/issues/5723
|
|
b.AssertFileContent(
|
|
"public/manualsummary/index.html",
|
|
"Single: HTML Content|Hello|en|RelPermalink: /manualsummary/|",
|
|
"Summary: \n<p>This is summary</p>\n|Truncated: true",
|
|
"|<p>This is the main content.</p>|",
|
|
)
|
|
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/5381
|
|
func TestPageManualSummary(t *testing.T) {
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile()
|
|
|
|
b.WithContent("page-md-shortcode.md", `---
|
|
title: "Hugo"
|
|
---
|
|
This is a {{< sc >}}.
|
|
<!--more-->
|
|
Content.
|
|
`)
|
|
|
|
// https://github.com/gohugoio/hugo/issues/5464
|
|
b.WithContent("page-md-only-shortcode.md", `---
|
|
title: "Hugo"
|
|
---
|
|
{{< sc >}}
|
|
<!--more-->
|
|
{{< sc >}}
|
|
`)
|
|
|
|
b.WithContent("page-md-shortcode-same-line.md", `---
|
|
title: "Hugo"
|
|
---
|
|
This is a {{< sc >}}<!--more-->Same line.
|
|
`)
|
|
|
|
b.WithContent("page-md-shortcode-same-line-after.md", `---
|
|
title: "Hugo"
|
|
---
|
|
Summary<!--more-->{{< sc >}}
|
|
`)
|
|
|
|
b.WithContent("page-org-shortcode.org", `#+TITLE: T1
|
|
#+AUTHOR: A1
|
|
#+DESCRIPTION: D1
|
|
This is a {{< sc >}}.
|
|
# more
|
|
Content.
|
|
`)
|
|
|
|
b.WithContent("page-org-variant1.org", `#+TITLE: T1
|
|
Summary.
|
|
|
|
# more
|
|
|
|
Content.
|
|
`)
|
|
|
|
b.WithTemplatesAdded("layouts/shortcodes/sc.html", "a shortcode")
|
|
b.WithTemplatesAdded("layouts/_default/single.html", `
|
|
SUMMARY:{{ .Summary }}:END
|
|
--------------------------
|
|
CONTENT:{{ .Content }}
|
|
`)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/page-md-shortcode/index.html",
|
|
"SUMMARY:<p>This is a a shortcode.</p>:END",
|
|
"CONTENT:<p>This is a a shortcode.</p>\n\n<p>Content.</p>\n",
|
|
)
|
|
|
|
b.AssertFileContent("public/page-md-shortcode-same-line/index.html",
|
|
"SUMMARY:<p>This is a a shortcode</p>:END",
|
|
"CONTENT:<p>This is a a shortcode</p>\n\n<p>Same line.</p>\n",
|
|
)
|
|
|
|
b.AssertFileContent("public/page-md-shortcode-same-line-after/index.html",
|
|
"SUMMARY:<p>Summary</p>:END",
|
|
"CONTENT:<p>Summary</p>\n\na shortcode",
|
|
)
|
|
|
|
b.AssertFileContent("public/page-org-shortcode/index.html",
|
|
"SUMMARY:<p>This is a a shortcode.</p>:END",
|
|
"CONTENT:<p>This is a a shortcode.</p>\n\n<p>Content.\t</p>\n",
|
|
)
|
|
b.AssertFileContent("public/page-org-variant1/index.html",
|
|
"SUMMARY:<p>Summary.</p>:END",
|
|
"CONTENT:<p>Summary.</p>\n\n<p>Content.\t</p>\n",
|
|
)
|
|
|
|
b.AssertFileContent("public/page-md-only-shortcode/index.html",
|
|
"SUMMARY:a shortcode:END",
|
|
"CONTENT:a shortcode\n\na shortcode\n",
|
|
)
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/5478
|
|
func TestPageWithCommentedOutFrontMatter(t *testing.T) {
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile()
|
|
|
|
b.WithContent("page.md", `<!--
|
|
+++
|
|
title = "hello"
|
|
+++
|
|
-->
|
|
This is the content.
|
|
`)
|
|
|
|
b.WithTemplatesAdded("layouts/_default/single.html", `
|
|
Title: {{ .Title }}
|
|
Content:{{ .Content }}
|
|
`)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/page/index.html",
|
|
"Title: hello",
|
|
"Content:<p>This is the content.</p>",
|
|
)
|
|
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/5781
|
|
func TestPageWithZeroFile(t *testing.T) {
|
|
newTestSitesBuilder(t).WithLogger(loggers.NewWarningLogger()).WithSimpleConfigFile().
|
|
WithTemplatesAdded("index.html", "{{ .File.Filename }}{{ with .File }}{{ .Dir }}{{ end }}").Build(BuildCfg{})
|
|
}
|
|
|
|
func TestHomePageWithNoTitle(t *testing.T) {
|
|
b := newTestSitesBuilder(t).WithSimpleConfigFile().
|
|
WithTemplatesAdded("index.html", "Title|{{ with .Title }}{{ . }}{{ end }}|")
|
|
b.WithContent("_index.md", `---
|
|
description: "No title for you!"
|
|
---
|
|
|
|
Content.
|
|
`)
|
|
|
|
b.Build(BuildCfg{})
|
|
b.AssertFileContent("public/index.html", "Title||")
|
|
}
|
|
|
|
func TestShouldBuild(t *testing.T) {
|
|
t.Parallel()
|
|
var past = time.Date(2009, 11, 17, 20, 34, 58, 651387237, time.UTC)
|
|
var future = time.Date(2037, 11, 17, 20, 34, 58, 651387237, time.UTC)
|
|
var zero = time.Time{}
|
|
|
|
var publishSettings = []struct {
|
|
buildFuture bool
|
|
buildExpired bool
|
|
buildDrafts bool
|
|
draft bool
|
|
publishDate time.Time
|
|
expiryDate time.Time
|
|
out bool
|
|
}{
|
|
// publishDate and expiryDate
|
|
{false, false, false, false, zero, zero, true},
|
|
{false, false, false, false, zero, future, true},
|
|
{false, false, false, false, past, zero, true},
|
|
{false, false, false, false, past, future, true},
|
|
{false, false, false, false, past, past, false},
|
|
{false, false, false, false, future, future, false},
|
|
{false, false, false, false, future, past, false},
|
|
|
|
// buildFuture and buildExpired
|
|
{false, true, false, false, past, past, true},
|
|
{true, true, false, false, past, past, true},
|
|
{true, false, false, false, past, past, false},
|
|
{true, false, false, false, future, future, true},
|
|
{true, true, false, false, future, future, true},
|
|
{false, true, false, false, future, past, false},
|
|
|
|
// buildDrafts and draft
|
|
{true, true, false, true, past, future, false},
|
|
{true, true, true, true, past, future, true},
|
|
{true, true, true, true, past, future, true},
|
|
}
|
|
|
|
for _, ps := range publishSettings {
|
|
s := shouldBuild(ps.buildFuture, ps.buildExpired, ps.buildDrafts, ps.draft,
|
|
ps.publishDate, ps.expiryDate)
|
|
if s != ps.out {
|
|
t.Errorf("AssertShouldBuild unexpected output with params: %+v", ps)
|
|
}
|
|
}
|
|
}
|
|
|
|
// "dot" in path: #1885 and #2110
|
|
// disablePathToLower regression: #3374
|
|
func TestPathIssues(t *testing.T) {
|
|
t.Parallel()
|
|
for _, disablePathToLower := range []bool{false, true} {
|
|
for _, uglyURLs := range []bool{false, true} {
|
|
t.Run(fmt.Sprintf("disablePathToLower=%t,uglyURLs=%t", disablePathToLower, uglyURLs), func(t *testing.T) {
|
|
|
|
cfg, fs := newTestCfg()
|
|
th := testHelper{cfg, fs, t}
|
|
|
|
cfg.Set("permalinks", map[string]string{
|
|
"post": ":section/:title",
|
|
})
|
|
|
|
cfg.Set("uglyURLs", uglyURLs)
|
|
cfg.Set("disablePathToLower", disablePathToLower)
|
|
cfg.Set("paginate", 1)
|
|
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "single.html"), "<html><body>{{.Content}}</body></html>")
|
|
writeSource(t, fs, filepath.Join("layouts", "_default", "list.html"),
|
|
"<html><body>P{{.Paginator.PageNumber}}|URL: {{.Paginator.URL}}|{{ if .Paginator.HasNext }}Next: {{.Paginator.Next.URL }}{{ end }}</body></html>")
|
|
|
|
for i := 0; i < 3; i++ {
|
|
writeSource(t, fs, filepath.Join("content", "post", fmt.Sprintf("doc%d.md", i)),
|
|
fmt.Sprintf(`---
|
|
title: "test%d.dot"
|
|
tags:
|
|
- ".net"
|
|
---
|
|
# doc1
|
|
*some content*`, i))
|
|
}
|
|
|
|
writeSource(t, fs, filepath.Join("content", "Blog", "Blog1.md"),
|
|
fmt.Sprintf(`---
|
|
title: "testBlog"
|
|
tags:
|
|
- "Blog"
|
|
---
|
|
# doc1
|
|
*some blog content*`))
|
|
|
|
s := buildSingleSite(t, deps.DepsCfg{Fs: fs, Cfg: cfg}, BuildCfg{})
|
|
|
|
require.Len(t, s.RegularPages(), 4)
|
|
|
|
pathFunc := func(s string) string {
|
|
if uglyURLs {
|
|
return strings.Replace(s, "/index.html", ".html", 1)
|
|
}
|
|
return s
|
|
}
|
|
|
|
blog := "blog"
|
|
|
|
if disablePathToLower {
|
|
blog = "Blog"
|
|
}
|
|
|
|
th.assertFileContent(pathFunc("public/"+blog+"/"+blog+"1/index.html"), "some blog content")
|
|
|
|
th.assertFileContent(pathFunc("public/post/test0.dot/index.html"), "some content")
|
|
|
|
if uglyURLs {
|
|
th.assertFileContent("public/post/page/1.html", `canonical" href="/post.html"/`)
|
|
th.assertFileContent("public/post.html", `<body>P1|URL: /post.html|Next: /post/page/2.html</body>`)
|
|
th.assertFileContent("public/post/page/2.html", `<body>P2|URL: /post/page/2.html|Next: /post/page/3.html</body>`)
|
|
} else {
|
|
th.assertFileContent("public/post/page/1/index.html", `canonical" href="/post/"/`)
|
|
th.assertFileContent("public/post/index.html", `<body>P1|URL: /post/|Next: /post/page/2/</body>`)
|
|
th.assertFileContent("public/post/page/2/index.html", `<body>P2|URL: /post/page/2/|Next: /post/page/3/</body>`)
|
|
th.assertFileContent("public/tags/.net/index.html", `<body>P1|URL: /tags/.net/|Next: /tags/.net/page/2/</body>`)
|
|
|
|
}
|
|
|
|
p := s.RegularPages()[0]
|
|
if uglyURLs {
|
|
require.Equal(t, "/post/test0.dot.html", p.RelPermalink())
|
|
} else {
|
|
require.Equal(t, "/post/test0.dot/", p.RelPermalink())
|
|
}
|
|
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
// https://github.com/gohugoio/hugo/issues/4675
|
|
func TestWordCountAndSimilarVsSummary(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
assert := require.New(t)
|
|
|
|
single := []string{"_default/single.html", `
|
|
WordCount: {{ .WordCount }}
|
|
FuzzyWordCount: {{ .FuzzyWordCount }}
|
|
ReadingTime: {{ .ReadingTime }}
|
|
Len Plain: {{ len .Plain }}
|
|
Len PlainWords: {{ len .PlainWords }}
|
|
Truncated: {{ .Truncated }}
|
|
Len Summary: {{ len .Summary }}
|
|
Len Content: {{ len .Content }}
|
|
|
|
SUMMARY:{{ .Summary }}:{{ len .Summary }}:END
|
|
`}
|
|
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile().WithTemplatesAdded(single...).WithContent("p1.md", fmt.Sprintf(`---
|
|
title: p1
|
|
---
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("word ", 510)),
|
|
|
|
"p2.md", fmt.Sprintf(`---
|
|
title: p2
|
|
---
|
|
This is a summary.
|
|
|
|
<!--more-->
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("word ", 310)),
|
|
"p3.md", fmt.Sprintf(`---
|
|
title: p3
|
|
isCJKLanguage: true
|
|
---
|
|
Summary: In Chinese, 好 means good.
|
|
|
|
<!--more-->
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("好", 200)),
|
|
"p4.md", fmt.Sprintf(`---
|
|
title: p4
|
|
isCJKLanguage: false
|
|
---
|
|
Summary: In Chinese, 好 means good.
|
|
|
|
<!--more-->
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("好", 200)),
|
|
|
|
"p5.md", fmt.Sprintf(`---
|
|
title: p4
|
|
isCJKLanguage: true
|
|
---
|
|
Summary: In Chinese, 好 means good.
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("好", 200)),
|
|
"p6.md", fmt.Sprintf(`---
|
|
title: p4
|
|
isCJKLanguage: false
|
|
---
|
|
Summary: In Chinese, 好 means good.
|
|
|
|
%s
|
|
|
|
`, strings.Repeat("好", 200)),
|
|
)
|
|
|
|
b.CreateSites().Build(BuildCfg{})
|
|
|
|
assert.Equal(1, len(b.H.Sites))
|
|
require.Len(t, b.H.Sites[0].RegularPages(), 6)
|
|
|
|
b.AssertFileContent("public/p1/index.html", "WordCount: 510\nFuzzyWordCount: 600\nReadingTime: 3\nLen Plain: 2550\nLen PlainWords: 510\nTruncated: false\nLen Summary: 2549\nLen Content: 2557")
|
|
|
|
b.AssertFileContent("public/p2/index.html", "WordCount: 314\nFuzzyWordCount: 400\nReadingTime: 2\nLen Plain: 1569\nLen PlainWords: 314\nTruncated: true\nLen Summary: 25\nLen Content: 1583")
|
|
|
|
b.AssertFileContent("public/p3/index.html", "WordCount: 206\nFuzzyWordCount: 300\nReadingTime: 1\nLen Plain: 638\nLen PlainWords: 7\nTruncated: true\nLen Summary: 43\nLen Content: 652")
|
|
b.AssertFileContent("public/p4/index.html", "WordCount: 7\nFuzzyWordCount: 100\nReadingTime: 1\nLen Plain: 638\nLen PlainWords: 7\nTruncated: true\nLen Summary: 43\nLen Content: 652")
|
|
b.AssertFileContent("public/p5/index.html", "WordCount: 206\nFuzzyWordCount: 300\nReadingTime: 1\nLen Plain: 638\nLen PlainWords: 7\nTruncated: true\nLen Summary: 229\nLen Content: 653")
|
|
b.AssertFileContent("public/p6/index.html", "WordCount: 7\nFuzzyWordCount: 100\nReadingTime: 1\nLen Plain: 638\nLen PlainWords: 7\nTruncated: false\nLen Summary: 637\nLen Content: 653")
|
|
|
|
}
|
|
|
|
func TestScratchSite(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
b := newTestSitesBuilder(t)
|
|
b.WithSimpleConfigFile().WithTemplatesAdded("index.html", `
|
|
{{ .Scratch.Set "b" "bv" }}
|
|
B: {{ .Scratch.Get "b" }}
|
|
`,
|
|
"shortcodes/scratch.html", `
|
|
{{ .Scratch.Set "c" "cv" }}
|
|
C: {{ .Scratch.Get "c" }}
|
|
`,
|
|
)
|
|
|
|
b.WithContentAdded("scratchme.md", `
|
|
---
|
|
title: Scratch Me!
|
|
---
|
|
|
|
{{< scratch >}}
|
|
`)
|
|
b.Build(BuildCfg{})
|
|
|
|
b.AssertFileContent("public/index.html", "B: bv")
|
|
b.AssertFileContent("public/scratchme/index.html", "C: cv")
|
|
}
|