node to page: Fix the Page copy()
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 16 Nov 2016 19:32:38 +0000 (20:32 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 22 Nov 2016 08:57:03 +0000 (09:57 +0100)
By embeding the init sync.Once var in a pointer so we can reset it when we copy the page.

Updates #2297

hugolib/hugo_sites.go
hugolib/page.go
hugolib/pageSort_test.go
hugolib/page_permalink_test.go
hugolib/page_test.go
hugolib/pagination_test.go

index 42e6cb503c7baeeacbfdc69e875a4d15cb741464..e4ee4ff9f6256e6dda54062f96db23cbec753313 100644 (file)
@@ -304,6 +304,7 @@ func (h *HugoSites) createMissingPages() error {
 // Move the new* methods after cleanup in site.go
 func (s *Site) newNodePage(typ string) *Page {
        return &Page{
+               pageInit: &pageInit{},
                Kind:     typ,
                Data:     make(map[string]interface{}),
                Site:     &s.Info,
index aa08c66f7e733d2e19b1e81d1db3b8d9ed8b6177..96a78ee736a7ab730929209cc22d66e1d2f4ae73 100644 (file)
@@ -70,6 +70,7 @@ const (
 )
 
 type Page struct {
+       *pageInit
 
        // Kind is the discriminator that identifies the different page types
        // in the different page collections. This can, as an example, be used
@@ -145,18 +146,14 @@ type Page struct {
        shortcodes        map[string]shortcode
 
        // the content stripped for HTML
-       plain          string // TODO should be []byte
-       plainWords     []string
-       plainInit      sync.Once
-       plainWordsInit sync.Once
+       plain      string // TODO should be []byte
+       plainWords []string
 
        // rendering configuration
-       renderingConfig     *helpers.Blackfriday
-       renderingConfigInit sync.Once
+       renderingConfig *helpers.Blackfriday
 
        // menus
-       pageMenus     PageMenus
-       pageMenusInit sync.Once
+       pageMenus PageMenus
 
        Source
 
@@ -196,14 +193,24 @@ type Page struct {
 
        URLPath
 
-       paginator     *Pager
-       paginatorInit sync.Once
+       paginator *Pager
 
        scratch *Scratch
 
-       language     *helpers.Language
-       languageInit sync.Once
-       lang         string
+       language *helpers.Language
+       lang     string
+}
+
+// pageInit lazy initializes different parts of the page. It is extracted
+// into its own type so we can easily create a copy of a given page.
+type pageInit struct {
+       languageInit        sync.Once
+       pageMenusInit       sync.Once
+       pageMetaInit        sync.Once
+       paginatorInit       sync.Once
+       plainInit           sync.Once
+       plainWordsInit      sync.Once
+       renderingConfigInit sync.Once
 }
 
 // IsNode returns whether this is an item of one of the list types in Hugo,
@@ -231,7 +238,6 @@ type PageMeta struct {
        wordCount      int
        fuzzyWordCount int
        readingTime    int
-       pageMetaInit   sync.Once
        Weight         int
 }
 
@@ -532,6 +538,7 @@ func (p *Page) getRenderingConfig() *helpers.Blackfriday {
 
 func newPage(filename string) *Page {
        page := Page{
+               pageInit:    &pageInit{},
                Kind:        kindFromFilename(filename),
                contentType: "",
                Source:      Source{File: *source.NewFile(filename)},
@@ -1538,20 +1545,12 @@ func (p *Page) updatePageDates() {
        }
 }
 
-// Page constains some sync.Once which have a mutex, so we cannot just
-// copy the Page by value. So for the situations where we need a copy,
-// the paginators etc., we do it manually here.
-// TODO(bep) np do better
+// copy creates a copy of this page with the lazy sync.Once vars reset
+// so they will be evaluated again, for word count calculations etc.
 func (p *Page) copy() *Page {
-       c := &Page{Kind: p.Kind, Site: p.Site}
-       c.Title = p.Title
-       c.Data = p.Data
-       c.Date = p.Date
-       c.Lastmod = p.Lastmod
-       c.language = p.language
-       c.lang = p.lang
-       c.URLPath = p.URLPath
-       return c
+       c := *p
+       c.pageInit = &pageInit{}
+       return &c
 }
 
 // TODO(bep) np these are pulled over from Node. Needs regrouping / embed
index 7ecbee98a20df1e0d37f6770b9070435da6e9bed..126c1b4adcf8e24f718f0399cbf813dfab93af0e 100644 (file)
@@ -147,7 +147,7 @@ func createSortTestPages(num int) Pages {
 
        for i := 0; i < num; i++ {
                pages[i] = &Page{
-
+                       pageInit: &pageInit{},
                        URLPath: URLPath{
                                Section: "z",
                                URL:     fmt.Sprintf("http://base/x/y/p%d.html", i),
index 341639529ec955b0bfaccf4f7eb6376ba3048b33..007f9eefccfb34b93c56adb1909333ff7af1d35c 100644 (file)
@@ -66,7 +66,8 @@ func TestPermalink(t *testing.T) {
                info := newSiteInfo(siteBuilderCfg{baseURL: string(test.base), language: helpers.NewDefaultLanguage()})
 
                p := &Page{
-                       Kind: KindPage,
+                       pageInit: &pageInit{},
+                       Kind:     KindPage,
                        URLPath: URLPath{
                                Section: "z",
                                URL:     test.url,
index 45ca241fdb3cc022085b8d81a94a9832392d2b93..a2e75c7ecefb7910f32d5f45d3cc3398963771f8 100644 (file)
@@ -1269,7 +1269,7 @@ func TestIndexPageSimpleMethods(t *testing.T) {
                {func(n *Page) bool { return n.Now().Unix() == time.Now().Unix() }},
        } {
 
-               n := &Page{Kind: KindHome}
+               n := &Page{pageInit: &pageInit{}, Kind: KindHome}
                n.RSSLink = "rssLink"
 
                if !this.assertFunc(n) {
index 591d090472e188041a13284131ae1c6d78e28959..2b3b18a5bef2dc02c78a5aca67c9d38fc4615b09 100644 (file)
@@ -457,6 +457,7 @@ func createTestPages(num int) Pages {
        info := newSiteInfo(siteBuilderCfg{baseURL: "http://base/", language: helpers.NewDefaultLanguage()})
        for i := 0; i < num; i++ {
                pages[i] = &Page{
+                       pageInit: &pageInit{},
                        URLPath: URLPath{
                                Section: "z",
                                URL:     fmt.Sprintf("http://base/x/y/p%d.html", i),