Revert "Use Node.ID for anchor ID"
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 12 Apr 2016 16:11:24 +0000 (18:11 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 12 Apr 2016 16:11:24 +0000 (18:11 +0200)
This reverts commit cd558958a0c0ecd06f7560a38e27334fe983e0de.

helpers/content.go
helpers/content_test.go
hugolib/handler_test.go
hugolib/hugolib_test.go [deleted file]
hugolib/node.go
hugolib/node_test.go
hugolib/page.go
hugolib/page_test.go
hugolib/shortcode.go
hugolib/shortcode_test.go
hugolib/site_test.go

index 3ac91b3606b6ccb219deb1c794077b1d73db4fad..f603490552ecb1f1ce11692179db7f849a0b1f25 100644 (file)
@@ -23,10 +23,6 @@ import (
        "os/exec"
        "unicode/utf8"
 
-       "fmt"
-       "strings"
-       "sync"
-
        "github.com/miekg/mmark"
        "github.com/mitchellh/mapstructure"
        "github.com/russross/blackfriday"
@@ -34,6 +30,9 @@ import (
        bp "github.com/spf13/hugo/bufferpool"
        jww "github.com/spf13/jwalterweatherman"
        "github.com/spf13/viper"
+
+       "strings"
+       "sync"
 )
 
 // SummaryLength is the length of the summary that Hugo extracts from a content.
@@ -168,11 +167,11 @@ func getHTMLRenderer(defaultFlags int, ctx *RenderingContext) blackfriday.Render
                FootnoteReturnLinkContents: viper.GetString("FootnoteReturnLinkContents"),
        }
 
-       b := ctx.DocumentID != 0
+       b := len(ctx.DocumentID) != 0
 
        if b && !ctx.getConfig().PlainIDAnchors {
-               renderParameters.FootnoteAnchorPrefix = fmt.Sprintf("%d:%s", ctx.DocumentID, renderParameters.FootnoteAnchorPrefix)
-               renderParameters.HeaderIDSuffix = fmt.Sprintf(":%d", ctx.DocumentID)
+               renderParameters.FootnoteAnchorPrefix = ctx.DocumentID + ":" + renderParameters.FootnoteAnchorPrefix
+               renderParameters.HeaderIDSuffix = ":" + ctx.DocumentID
        }
 
        htmlFlags := defaultFlags
@@ -259,10 +258,10 @@ func getMmarkHTMLRenderer(defaultFlags int, ctx *RenderingContext) mmark.Rendere
                FootnoteReturnLinkContents: viper.GetString("FootnoteReturnLinkContents"),
        }
 
-       b := ctx.DocumentID != 0
+       b := len(ctx.DocumentID) != 0
 
        if b && !ctx.getConfig().PlainIDAnchors {
-               renderParameters.FootnoteAnchorPrefix = fmt.Sprintf("%d:%s", ctx.DocumentID, renderParameters.FootnoteAnchorPrefix)
+               renderParameters.FootnoteAnchorPrefix = ctx.DocumentID + ":" + renderParameters.FootnoteAnchorPrefix
                // renderParameters.HeaderIDSuffix = ":" + ctx.DocumentId
        }
 
@@ -344,7 +343,7 @@ func ExtractTOC(content []byte) (newcontent []byte, toc []byte) {
 type RenderingContext struct {
        Content      []byte
        PageFmt      string
-       DocumentID   int
+       DocumentID   string
        Config       *Blackfriday
        FileResolver FileResolverFunc
        LinkResolver LinkResolverFunc
index 347884f0ac316ac02155c0a4085cf172768654a3..a89b4992e81c44d72cb45893bf594e31ff64a936 100644 (file)
@@ -172,15 +172,15 @@ func TestGetHTMLRendererAllFlags(t *testing.T) {
 
 func TestGetHTMLRendererAnchors(t *testing.T) {
        ctx := &RenderingContext{}
-       ctx.DocumentID = 123
+       ctx.DocumentID = "testid"
        ctx.Config = ctx.getConfig()
        ctx.Config.PlainIDAnchors = false
 
        actualRenderer := getHTMLRenderer(0, ctx)
        headerBuffer := &bytes.Buffer{}
        footnoteBuffer := &bytes.Buffer{}
-       expectedFootnoteHref := []byte("href=\"#fn:123:href\"")
-       expectedHeaderID := []byte("<h1 id=\"id:123\"></h1>\n")
+       expectedFootnoteHref := []byte("href=\"#fn:testid:href\"")
+       expectedHeaderID := []byte("<h1 id=\"id:testid\"></h1>\n")
 
        actualRenderer.Header(headerBuffer, func() bool { return true }, 1, "id")
        actualRenderer.FootnoteRef(footnoteBuffer, []byte("href"), 1)
@@ -196,14 +196,14 @@ func TestGetHTMLRendererAnchors(t *testing.T) {
 
 func TestGetMmarkHTMLRenderer(t *testing.T) {
        ctx := &RenderingContext{}
-       ctx.DocumentID = 321
+       ctx.DocumentID = "testid"
        ctx.Config = ctx.getConfig()
        ctx.Config.PlainIDAnchors = false
        actualRenderer := getMmarkHTMLRenderer(0, ctx)
 
        headerBuffer := &bytes.Buffer{}
        footnoteBuffer := &bytes.Buffer{}
-       expectedFootnoteHref := []byte("href=\"#fn:321:href\"")
+       expectedFootnoteHref := []byte("href=\"#fn:testid:href\"")
        expectedHeaderID := []byte("<h1 id=\"id\"></h1>")
 
        actualRenderer.FootnoteRef(footnoteBuffer, []byte("href"), 1)
index e48d2693254bc2fd455b8bccd4374e9b73f0b287..29b1161e4dc928100553185cb299dba997c6aa04 100644 (file)
@@ -25,8 +25,8 @@ import (
 )
 
 func TestDefaultHandler(t *testing.T) {
-       setUp()
-       defer tearDown()
+       viper.Reset()
+       defer viper.Reset()
 
        hugofs.InitMemFs()
        sources := []source.ByteSource{
@@ -63,14 +63,14 @@ func TestDefaultHandler(t *testing.T) {
                doc      string
                expected string
        }{
-               {filepath.FromSlash("sect/doc1.html"), "\n\n<h1 id=\"title:42\">title</h1>\n\n<p>some <em>content</em></p>\n"},
+               {filepath.FromSlash("sect/doc1.html"), "\n\n<h1 id=\"title:5d74edbb89ef198cd37882b687940cda\">title</h1>\n\n<p>some <em>content</em></p>\n"},
                {filepath.FromSlash("sect/doc2.html"), "<!doctype html><html><body>more content</body></html>"},
-               {filepath.FromSlash("sect/doc3.html"), "\n\n<h1 id=\"doc3:42\">doc3</h1>\n\n<p><em>some</em> content</p>\n"},
+               {filepath.FromSlash("sect/doc3.html"), "\n\n<h1 id=\"doc3:28c75a9e2162b8eccda73a1ab9ce80b4\">doc3</h1>\n\n<p><em>some</em> content</p>\n"},
                {filepath.FromSlash("sect/doc3/img1.png"), string([]byte("‰PNG \1a ��� IHDR���\ 1���\ 1\b����:~›U��� IDAT\18Wcø\ f\ 1\ 1\ 1�ZMoñ����IEND®B`‚"))},
                {filepath.FromSlash("sect/img2.gif"), string([]byte("GIF89a\ 1\ 1�€��ÿÿÿ���,����\ 1\ 1��\ 2\ 2D\ 1�;"))},
                {filepath.FromSlash("sect/img2.spf"), string([]byte("****FAKE-FILETYPE****"))},
                {filepath.FromSlash("doc7.html"), "<html><body>doc7 content</body></html>"},
-               {filepath.FromSlash("sect/doc8.html"), "\n\n<h1 id=\"title:42\">title</h1>\n\n<p>some <em>content</em></p>\n"},
+               {filepath.FromSlash("sect/doc8.html"), "\n\n<h1 id=\"title:0ae308ad73e2f37bd09874105281b5d8\">title</h1>\n\n<p>some <em>content</em></p>\n"},
        }
 
        for _, test := range tests {
diff --git a/hugolib/hugolib_test.go b/hugolib/hugolib_test.go
deleted file mode 100644 (file)
index 8f17553..0000000
+++ /dev/null
@@ -1,34 +0,0 @@
-// 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 (
-       "github.com/spf13/viper"
-)
-
-var stableNodeIDProvider nodeIDProviderFunc = func(n *Node) int {
-       return 42
-}
-
-// common test setup.
-func setUp() {
-       viper.Reset()
-       nodeIDProvider = stableNodeIDProvider
-}
-
-// common test cleanup.
-func tearDown() {
-       viper.Reset()
-       nodeIDProvider = defaultNodeIDProvider
-}
index dde1bd690b2fd3ce4664adad13f86496b8b1d457..7e9ad74587a9fee3b94bcbc0e2ee8d6725eb9a6d 100644 (file)
@@ -47,15 +47,6 @@ type Node struct {
 // but that would lead to massive changes; do it simple for now.
 var nodeIDCounter uint64
 
-type nodeIDProviderFunc func(n *Node) int
-
-var defaultNodeIDProvider nodeIDProviderFunc = func(n *Node) int {
-       n.idInit.Do(func() { n.id = nextNodeID() })
-       return n.id
-}
-
-var nodeIDProvider nodeIDProviderFunc = defaultNodeIDProvider
-
 func nextNodeID() int {
        return int(atomic.AddUint64(&nodeIDCounter, 1))
 }
@@ -64,7 +55,8 @@ func nextNodeID() int {
 // This is unique for a given Hugo build, but must not be considered stable.
 // See UniqueID on Page for an identify that is stable for repeated builds.
 func (n *Node) ID() int {
-       return nodeIDProvider(n)
+       n.idInit.Do(func() { n.id = nextNodeID() })
+       return n.id
 }
 
 func (n *Node) Now() time.Time {
index a0d08ed07826f47d442c456e367f85fe2976d326..5b83cc0ad5fa64567a9641f5c842cfdb98ae2950 100644 (file)
@@ -43,6 +43,14 @@ func TestNodeSimpleMethods(t *testing.T) {
 }
 
 func TestNodeID(t *testing.T) {
+       t.Parallel()
+
+       n1 := &Node{}
+       n2 := &Node{}
+
+       assert.True(t, n1.ID() > 0)
+       assert.Equal(t, n1.ID(), n1.ID())
+       assert.True(t, n2.ID() > n1.ID())
 
        var wg sync.WaitGroup
 
@@ -50,13 +58,8 @@ func TestNodeID(t *testing.T) {
                wg.Add(1)
                go func(j int) {
                        for k := 0; k < 10; k++ {
-                               n1 := &Node{}
-                               n2 := &Node{}
-
-                               assert.True(t, n1.ID() > 0)
-                               assert.Equal(t, n1.ID(), n1.ID())
-                               assert.True(t, n2.ID() > n1.ID())
-
+                               n := &Node{}
+                               assert.True(t, n.ID() > 0)
                        }
                        wg.Done()
                }(i)
index 9a9deecc197e6ae36b81af35a15cf729aa44dc26..cff84737b7ec5f5c61680dd7076472ce2c8ca941 100644 (file)
@@ -265,7 +265,7 @@ func (p *Page) renderBytes(content []byte) []byte {
        }
        return helpers.RenderBytes(
                &helpers.RenderingContext{Content: content, PageFmt: p.determineMarkupType(),
-                       DocumentID: p.ID(), Config: p.getRenderingConfig(), LinkResolver: fn, FileResolver: fileFn})
+                       DocumentID: p.UniqueID(), Config: p.getRenderingConfig(), LinkResolver: fn, FileResolver: fileFn})
 }
 
 func (p *Page) renderContent(content []byte) []byte {
@@ -280,7 +280,7 @@ func (p *Page) renderContent(content []byte) []byte {
                }
        }
        return helpers.RenderBytesWithTOC(&helpers.RenderingContext{Content: content, PageFmt: p.determineMarkupType(),
-               DocumentID: p.ID(), Config: p.getRenderingConfig(), LinkResolver: fn, FileResolver: fileFn})
+               DocumentID: p.UniqueID(), Config: p.getRenderingConfig(), LinkResolver: fn, FileResolver: fileFn})
 }
 
 func (p *Page) getRenderingConfig() *helpers.Blackfriday {
index 6e636261d037983f877348a35ec02a7715107b6c..b492bab2d328ee4a29eae3f07ceb88ce2cb38025 100644 (file)
@@ -605,17 +605,14 @@ func TestPageWithAdditionalExtension(t *testing.T) {
 }
 
 func TestTableOfContents(t *testing.T) {
-       setUp()
-       defer tearDown()
-
        p, _ := NewPage("tocpage.md")
        _, err := p.ReadFrom(strings.NewReader(pageWithToC))
        p.Convert()
        if err != nil {
                t.Fatalf("Unable to create a page with frontmatter and body content: %s", err)
        }
-       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:42\">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:42\">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&ndash;as swift as the passage of light&ndash;would leap after me from the pit about\nthe cylinder and strike me down. ## BB</p>\n\n<h3 id=\"bbb:42\">BBB</h3>\n\n<p>&ldquo;You&rsquo;re a great Granser,&rdquo; he cried delightedly, &ldquo;always making believe them little marks mean something.&rdquo;</p>\n")
-       checkPageTOC(t, p, "<nav id=\"TableOfContents\">\n<ul>\n<li>\n<ul>\n<li><a href=\"#aa:42\">AA</a>\n<ul>\n<li><a href=\"#aaa:42\">AAA</a></li>\n<li><a href=\"#bbb:42\">BBB</a></li>\n</ul></li>\n</ul></li>\n</ul>\n</nav>")
+       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:90b9174a5bdb091a9625b04adac96ca6\">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:90b9174a5bdb091a9625b04adac96ca6\">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&ndash;as swift as the passage of light&ndash;would leap after me from the pit about\nthe cylinder and strike me down. ## BB</p>\n\n<h3 id=\"bbb:90b9174a5bdb091a9625b04adac96ca6\">BBB</h3>\n\n<p>&ldquo;You&rsquo;re a great Granser,&rdquo; he cried delightedly, &ldquo;always making believe them little marks mean something.&rdquo;</p>\n")
+       checkPageTOC(t, p, "<nav id=\"TableOfContents\">\n<ul>\n<li>\n<ul>\n<li><a href=\"#aa:90b9174a5bdb091a9625b04adac96ca6\">AA</a>\n<ul>\n<li><a href=\"#aaa:90b9174a5bdb091a9625b04adac96ca6\">AAA</a></li>\n<li><a href=\"#bbb:90b9174a5bdb091a9625b04adac96ca6\">BBB</a></li>\n</ul></li>\n</ul></li>\n</ul>\n</nav>")
 }
 
 func TestPageWithMoreTag(t *testing.T) {
index dfe26d4d7991390de952450c110c7d7653095fb4..876e9293f53aefc78f33520b7851213da8427747 100644 (file)
@@ -235,7 +235,7 @@ func renderShortcode(sc shortcode, parent *ShortcodeWithPage, p *Page, t tpl.Tem
                if sc.doMarkup {
                        newInner := helpers.RenderBytes(&helpers.RenderingContext{
                                Content: []byte(inner), PageFmt: p.determineMarkupType(),
-                               DocumentID: p.ID(), Config: p.getRenderingConfig()})
+                               DocumentID: p.UniqueID(), Config: p.getRenderingConfig()})
 
                        // If the type is “unknown” or “markdown”, we assume the markdown
                        // generation has been performed. Given the input: `a line`, markdown
index 68c7f922071f3ae611abceded017e504b6bf1ca0..ab764b845693ee86aeb7d57d6dd4143e94f9f5bc 100644 (file)
@@ -168,9 +168,6 @@ func TestInnerSC(t *testing.T) {
 }
 
 func TestInnerSCWithMarkdown(t *testing.T) {
-       setUp()
-       defer tearDown()
-
        tem := tpl.New()
        tem.AddInternalShortcode("inside.html", `<div{{with .Get "class"}} class="{{.}}"{{end}}>{{ .Inner }}</div>`)
 
@@ -179,12 +176,10 @@ func TestInnerSCWithMarkdown(t *testing.T) {
 
 [link](http://spf13.com) and text
 
-{{% /inside %}}`, "<div><h1 id=\"more-here:42\">More Here</h1>\n\n<p><a href=\"http://spf13.com\">link</a> and text</p>\n</div>", tem)
+{{% /inside %}}`, "<div><h1 id=\"more-here:bec3ed8ba720b9073ab75abcf3ba5d97\">More Here</h1>\n\n<p><a href=\"http://spf13.com\">link</a> and text</p>\n</div>", tem)
 }
 
 func TestInnerSCWithAndWithoutMarkdown(t *testing.T) {
-       setUp()
-       defer tearDown()
        tem := tpl.New()
        tem.AddInternalShortcode("inside.html", `<div{{with .Get "class"}} class="{{.}}"{{end}}>{{ .Inner }}</div>`)
 
@@ -203,7 +198,7 @@ And then:
 This is **plain** text.
 
 {{< /inside >}}
-`, "<div><h1 id=\"more-here:42\">More Here</h1>\n\n<p><a href=\"http://spf13.com\">link</a> and text</p>\n</div>\n\nAnd then:\n\n<div>\n# More Here\n\nThis is **plain** text.\n\n</div>\n", tem)
+`, "<div><h1 id=\"more-here:bec3ed8ba720b9073ab75abcf3ba5d97\">More Here</h1>\n\n<p><a href=\"http://spf13.com\">link</a> and text</p>\n</div>\n\nAnd then:\n\n<div>\n# More Here\n\nThis is **plain** text.\n\n</div>\n", tem)
 }
 
 func TestEmbeddedSC(t *testing.T) {
index 55e7b529941912422a1151b6bc6d070f1fd58203..9f0abc98cc14fbf63dddf554e8fa6d1394b099b6 100644 (file)
@@ -142,9 +142,6 @@ func NopCloser(w io.Writer) io.WriteCloser {
 }
 
 func TestRenderThing(t *testing.T) {
-       setUp()
-       defer tearDown()
-
        tests := []struct {
                content  string
                template string
@@ -152,7 +149,7 @@ func TestRenderThing(t *testing.T) {
        }{
                {pageSimpleTitle, templateTitle, "simple template"},
                {pageSimpleTitle, templateFunc, "simple-template"},
-               {pageWithMd, templateContent, "\n\n<h1 id=\"heading-1:42\">heading 1</h1>\n\n<p>text</p>\n\n<h2 id=\"heading-2:42\">heading 2</h2>\n\n<p>more text</p>\n"},
+               {pageWithMd, templateContent, "\n\n<h1 id=\"heading-1:91b5c4a22fc6103c73bb91e4a40568f8\">heading 1</h1>\n\n<p>text</p>\n\n<h2 id=\"heading-2:91b5c4a22fc6103c73bb91e4a40568f8\">heading 2</h2>\n\n<p>more text</p>\n"},
                {simplePageRFC3339Date, templateDate, "2013-05-17 16:59:30 &#43;0000 UTC"},
        }
 
@@ -559,8 +556,8 @@ func doTestSectionNaming(t *testing.T, canonify, uglify, pluralize bool) {
 
 }
 func TestSkipRender(t *testing.T) {
-       setUp()
-       defer tearDown()
+       viper.Reset()
+       defer viper.Reset()
 
        hugofs.InitMemFs()
        sources := []source.ByteSource{
@@ -596,14 +593,14 @@ func TestSkipRender(t *testing.T) {
                doc      string
                expected string
        }{
-               {filepath.FromSlash("sect/doc1.html"), "\n\n<h1 id=\"title:42\">title</h1>\n\n<p>some <em>content</em></p>\n"},
+               {filepath.FromSlash("sect/doc1.html"), "\n\n<h1 id=\"title:5d74edbb89ef198cd37882b687940cda\">title</h1>\n\n<p>some <em>content</em></p>\n"},
                {filepath.FromSlash("sect/doc2.html"), "<!doctype html><html><body>more content</body></html>"},
-               {filepath.FromSlash("sect/doc3.html"), "\n\n<h1 id=\"doc3:42\">doc3</h1>\n\n<p><em>some</em> content</p>\n"},
-               {filepath.FromSlash("sect/doc4.html"), "\n\n<h1 id=\"doc4:42\">doc4</h1>\n\n<p><em>some content</em></p>\n"},
+               {filepath.FromSlash("sect/doc3.html"), "\n\n<h1 id=\"doc3:28c75a9e2162b8eccda73a1ab9ce80b4\">doc3</h1>\n\n<p><em>some</em> content</p>\n"},
+               {filepath.FromSlash("sect/doc4.html"), "\n\n<h1 id=\"doc4:f8e6806123f341b8975509637645a4d3\">doc4</h1>\n\n<p><em>some content</em></p>\n"},
                {filepath.FromSlash("sect/doc5.html"), "<!doctype html><html><head><script src=\"script.js\"></script></head><body>body5</body></html>"},
                {filepath.FromSlash("sect/doc6.html"), "<!doctype html><html><head><script src=\"http://auth/bub/script.js\"></script></head><body>body5</body></html>"},
                {filepath.FromSlash("doc7.html"), "<html><body>doc7 content</body></html>"},
-               {filepath.FromSlash("sect/doc8.html"), "\n\n<h1 id=\"title:42\">title</h1>\n\n<p>some <em>content</em></p>\n"},
+               {filepath.FromSlash("sect/doc8.html"), "\n\n<h1 id=\"title:0ae308ad73e2f37bd09874105281b5d8\">title</h1>\n\n<p>some <em>content</em></p>\n"},
        }
 
        for _, test := range tests {