hugolib: Fix regression of .Truncated evaluation in manual summaries
authorAnton Staaf <anton@socialhacker.com>
Thu, 26 Jan 2017 19:58:25 +0000 (11:58 -0800)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sun, 19 Feb 2017 09:11:49 +0000 (10:11 +0100)
This fixes the behavior of .Truncated that was introduced with commit
bef496b97e1dc2df8c52b080152d539a58752801 which was later broken.  The
desired behavior is that .Truncated would evaluate to false when there
was nothing after the user defined summary marker.

This also adds a simple unit test to ensure that this feature isn't
broken again.  The check for content after the user defined summary
marker is done on the raw content instead of the working copy because
some of the markup renderers add elements after the marker, making it
difficult to determine if there is actually any content.

The behavior (evaluating to false when there is no content, just
summary) is also now documented.

docs/content/content/summaries.md
hugolib/handler_page.go
hugolib/page.go
hugolib/page_test.go

index e64d43331941788be227ebabd482eaea8944a9a3..0f2e6433bee87eb8b008a7d7b7f8435dc3a1cf44 100644 (file)
@@ -33,6 +33,7 @@ Alternatively, you may add the <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</co
 
 Be careful to enter <code>&#60;&#33;&#45;&#45;more&#45;&#45;&#62;</code> exactly, i.e. all lowercase with no whitespace, otherwise it would be treated as regular comment and ignored.
 
+If there is nothing but spaces and newlines after the summary divider then `.Truncated` will be false.
 
 ## Showing Summaries
 
index 01c041ffc6b1d316edd45ce71d4fc89dd66d2404..a47758f84753d0d1d146a7aa0cb725af0afd1070 100644 (file)
@@ -14,7 +14,6 @@
 package hugolib
 
 import (
-       "bytes"
        "fmt"
 
        "github.com/spf13/hugo/helpers"
@@ -127,10 +126,7 @@ func commonConvert(p *Page) HandledResult {
                p.workContent = helpers.Emojify(p.workContent)
        }
 
-       // We have to replace the <!--more--> with something that survives all the
-       // rendering engines.
-       // TODO(bep) inline replace
-       p.workContent = bytes.Replace(p.workContent, []byte(helpers.SummaryDivider), internalSummaryDivider, 1)
+       p.workContent = p.replaceDivider(p.workContent)
        p.workContent = p.renderContent(p.workContent)
 
        return HandledResult{err: nil}
index f532758d8904de8a40207ebc823e8020a35ffd86..6af6a2da50b718dd83137c0d2a98471afca5cd0a 100644 (file)
@@ -414,6 +414,23 @@ var (
        internalSummaryDivider = []byte("HUGOMORE42")
 )
 
+// We have to replace the <!--more--> with something that survives all the
+// rendering engines.
+// TODO(bep) inline replace
+func (p *Page) replaceDivider(content []byte) []byte {
+       sections := bytes.Split(content, helpers.SummaryDivider)
+
+       // If the raw content has nothing but whitespace after the summary
+       // marker then the page shouldn't be marked as truncated.  This check
+       // is simplest against the raw content because different markup engines
+       // (rst and asciidoc in particular) add div and p elements after the
+       // summary marker.
+       p.Truncated = (len(sections) == 2 &&
+               len(bytes.Trim(sections[1], " \n\r")) > 0)
+
+       return bytes.Join(sections, internalSummaryDivider)
+}
+
 // Returns the page as summary and main if a user defined split is provided.
 func (p *Page) setUserDefinedSummaryIfProvided(rawContentCopy []byte) (*summaryContent, error) {
 
@@ -428,12 +445,6 @@ func (p *Page) setUserDefinedSummaryIfProvided(rawContentCopy []byte) (*summaryC
                return nil, nil
        }
 
-       p.Truncated = true
-       if len(sc.content) < 20 {
-               // only whitespace?
-               p.Truncated = len(bytes.Trim(sc.content, " \n\r")) > 0
-       }
-
        p.Summary = helpers.BytesToHTML(sc.summary)
 
        return sc, nil
@@ -441,9 +452,8 @@ func (p *Page) setUserDefinedSummaryIfProvided(rawContentCopy []byte) (*summaryC
 
 // Make this explicit so there is no doubt about what is what.
 type summaryContent struct {
-       summary               []byte
-       content               []byte
-       contentWithoutSummary []byte
+       summary []byte
+       content []byte
 }
 
 func splitUserDefinedSummaryAndContent(markup string, c []byte) (sc *summaryContent, err error) {
@@ -467,7 +477,6 @@ func splitUserDefinedSummaryAndContent(markup string, c []byte) (sc *summaryCont
                startMarkup []byte
                endMarkup   []byte
                addDiv      bool
-               divStart    = []byte("<div class=\"document\">")
        )
 
        switch markup {
@@ -499,20 +508,16 @@ func splitUserDefinedSummaryAndContent(markup string, c []byte) (sc *summaryCont
 
        withoutDivider := bytes.TrimSpace(append(c[:startDivider], c[endDivider:]...))
        var (
-               contentWithoutSummary []byte
-               summary               []byte
+               summary []byte
        )
 
        if len(withoutDivider) > 0 {
-               contentWithoutSummary = bytes.TrimSpace(withoutDivider[endSummary:])
                summary = bytes.TrimSpace(withoutDivider[:endSummary])
        }
 
        if addDiv {
                // For the rst
                summary = append(append([]byte(nil), summary...), []byte("</div>")...)
-               // TODO(bep) include the document class, maybe
-               contentWithoutSummary = append(divStart, contentWithoutSummary...)
        }
 
        if err != nil {
@@ -520,9 +525,8 @@ func splitUserDefinedSummaryAndContent(markup string, c []byte) (sc *summaryCont
        }
 
        sc = &summaryContent{
-               summary:               summary,
-               content:               withoutDivider,
-               contentWithoutSummary: contentWithoutSummary,
+               summary: summary,
+               content: withoutDivider,
        }
 
        return
index 0fa622e335da3236195b5aa6dea0bf84a1e5f16b..d86a7b2d2d48279dcc34bc38357c013d0ebd9dec 100644 (file)
@@ -164,6 +164,14 @@ title: Simple
 Summary Same Line<!--more-->
 
 Some more text
+`
+
+       simplePageWithSummaryDelimiterOnlySummary = `---
+title: Simple
+---
+Summary text
+
+<!--more-->
 `
 
        simplePageWithAllCJKRunes = `---
@@ -665,46 +673,42 @@ func TestCreateNewPage(t *testing.T) {
 func TestSplitSummaryAndContent(t *testing.T) {
        t.Parallel()
        for i, this := range []struct {
-               markup                        string
-               content                       string
-               expectedSummary               string
-               expectedContent               string
-               expectedContentWithoutSummary string
+               markup          string
+               content         string
+               expectedSummary string
+               expectedContent string
        }{
                {"markdown", `<p>Summary Same LineHUGOMORE42</p>
 
-<p>Some more text</p>`, "<p>Summary Same Line</p>", "<p>Summary Same Line</p>\n\n<p>Some more text</p>", "<p>Some more text</p>"},
+<p>Some more text</p>`, "<p>Summary Same Line</p>", "<p>Summary Same Line</p>\n\n<p>Some more text</p>"},
                {"asciidoc", `<div class="paragraph"><p>sn</p></div><div class="paragraph"><p>HUGOMORE42Some more text</p></div>`,
                        "<div class=\"paragraph\"><p>sn</p></div>",
-                       "<div class=\"paragraph\"><p>sn</p></div><div class=\"paragraph\"><p>Some more text</p></div>",
-                       "<div class=\"paragraph\"><p>Some more text</p></div>"},
+                       "<div class=\"paragraph\"><p>sn</p></div><div class=\"paragraph\"><p>Some more text</p></div>"},
                {"rst",
                        "<div class=\"document\"><p>Summary Next Line</p><p>HUGOMORE42Some more text</p></div>",
                        "<div class=\"document\"><p>Summary Next Line</p></div>",
-                       "<div class=\"document\"><p>Summary Next Line</p><p>Some more text</p></div>",
-                       "<div class=\"document\"><p>Some more text</p></div>"},
-               {"markdown", "<p>a</p><p>b</p><p>HUGOMORE42c</p>", "<p>a</p><p>b</p>", "<p>a</p><p>b</p><p>c</p>", "<p>c</p>"},
-               {"markdown", "<p>a</p><p>b</p><p>cHUGOMORE42</p>", "<p>a</p><p>b</p><p>c</p>", "<p>a</p><p>b</p><p>c</p>", ""},
-               {"markdown", "<p>a</p><p>bHUGOMORE42</p><p>c</p>", "<p>a</p><p>b</p>", "<p>a</p><p>b</p><p>c</p>", "<p>c</p>"},
-               {"markdown", "<p>aHUGOMORE42</p><p>b</p><p>c</p>", "<p>a</p>", "<p>a</p><p>b</p><p>c</p>", "<p>b</p><p>c</p>"},
-               {"markdown", "  HUGOMORE42 ", "", "", ""},
-               {"markdown", "HUGOMORE42", "", "", ""},
-               {"markdown", "<p>HUGOMORE42", "<p>", "<p>", ""},
-               {"markdown", "HUGOMORE42<p>", "", "<p>", "<p>"},
-               {"markdown", "\n\n<p>HUGOMORE42</p>\n", "<p></p>", "<p></p>", ""},
+                       "<div class=\"document\"><p>Summary Next Line</p><p>Some more text</p></div>"},
+               {"markdown", "<p>a</p><p>b</p><p>HUGOMORE42c</p>", "<p>a</p><p>b</p>", "<p>a</p><p>b</p><p>c</p>"},
+               {"markdown", "<p>a</p><p>b</p><p>cHUGOMORE42</p>", "<p>a</p><p>b</p><p>c</p>", "<p>a</p><p>b</p><p>c</p>"},
+               {"markdown", "<p>a</p><p>bHUGOMORE42</p><p>c</p>", "<p>a</p><p>b</p>", "<p>a</p><p>b</p><p>c</p>"},
+               {"markdown", "<p>aHUGOMORE42</p><p>b</p><p>c</p>", "<p>a</p>", "<p>a</p><p>b</p><p>c</p>"},
+               {"markdown", "  HUGOMORE42 ", "", ""},
+               {"markdown", "HUGOMORE42", "", ""},
+               {"markdown", "<p>HUGOMORE42", "<p>", "<p>"},
+               {"markdown", "HUGOMORE42<p>", "", "<p>"},
+               {"markdown", "\n\n<p>HUGOMORE42</p>\n", "<p></p>", "<p></p>"},
                // Issue #2586
                // Note: Hugo will not split mid-sentence but will look for the closest
                // paragraph end marker. This may be a change from Hugo 0.16, but it makes sense.
                {"markdown", `<p>this is an example HUGOMORE42of the issue.</p>`,
                        "<p>this is an example of the issue.</p>",
-                       "<p>this is an example of the issue.</p>", ""},
+                       "<p>this is an example of the issue.</p>"},
                // Issue: #2538
                {"markdown", fmt.Sprintf(` <p class="lead">%s</p>HUGOMORE42<p>%s</p>
 `,
                        strings.Repeat("A", 10), strings.Repeat("B", 31)),
                        fmt.Sprintf(`<p class="lead">%s</p>`, strings.Repeat("A", 10)),
                        fmt.Sprintf(`<p class="lead">%s</p><p>%s</p>`, strings.Repeat("A", 10), strings.Repeat("B", 31)),
-                       fmt.Sprintf(`<p>%s</p>`, strings.Repeat("B", 31)),
                },
        } {
 
@@ -714,7 +718,6 @@ func TestSplitSummaryAndContent(t *testing.T) {
                require.NotNil(t, sc, fmt.Sprintf("[%d] Nil %s", i, this.markup))
                require.Equal(t, this.expectedSummary, string(sc.summary), fmt.Sprintf("[%d] Summary markup %s", i, this.markup))
                require.Equal(t, this.expectedContent, string(sc.content), fmt.Sprintf("[%d] Content markup %s", i, this.markup))
-               require.Equal(t, this.expectedContentWithoutSummary, string(sc.contentWithoutSummary), fmt.Sprintf("[%d] Content without summary, markup %s", i, this.markup))
        }
 }
 
@@ -850,6 +853,16 @@ func TestPageWithMoreTag(t *testing.T) {
        testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithSummaryDelimiterSameLine)
 }
 
+func TestPageWithMoreTagOnlySummary(t *testing.T) {
+
+       assertFunc := func(t *testing.T, ext string, pages Pages) {
+               p := pages[0]
+               checkTruncation(t, p, false, "page with summary delimiter at end")
+       }
+
+       testAllMarkdownEnginesForPages(t, assertFunc, nil, simplePageWithSummaryDelimiterOnlySummary)
+}
+
 func TestPageWithDate(t *testing.T) {
        t.Parallel()
        cfg, fs := newTestCfg()