Fix multilingual reload when shortcode changes
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 12 Aug 2016 22:33:17 +0000 (00:33 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 6 Sep 2016 15:32:21 +0000 (18:32 +0300)
This commit also refines the partial rebuild logic, to make sure we do not do more work than needed.

Updates #2309

hugolib/handler_page.go
hugolib/hugo_sites.go
hugolib/hugo_sites_test.go
hugolib/page.go
hugolib/shortcode.go
hugolib/site.go

index fcc5b956107026cb10fddd9f5e682c216b8e6bc0..c1f83d7166daef2ccf4cd704e0b1733045d1be2c 100644 (file)
@@ -15,6 +15,7 @@ package hugolib
 
 import (
        "bytes"
+       "fmt"
 
        "github.com/spf13/hugo/helpers"
        "github.com/spf13/hugo/source"
@@ -67,6 +68,10 @@ type htmlHandler struct {
 
 func (h htmlHandler) Extensions() []string { return []string{"html", "htm"} }
 func (h htmlHandler) PageConvert(p *Page, t tpl.Template) HandledResult {
+       if p.rendered {
+               panic(fmt.Sprintf("Page %q already rendered, does not need conversion", p.BaseFileName()))
+       }
+
        p.ProcessShortcodes(t)
 
        return HandledResult{err: nil}
@@ -100,6 +105,11 @@ func (h mmarkHandler) PageConvert(p *Page, t tpl.Template) HandledResult {
 }
 
 func commonConvert(p *Page, t tpl.Template) HandledResult {
+
+       if p.rendered {
+               panic(fmt.Sprintf("Page %q already rendered, does not need conversion", p.BaseFileName()))
+       }
+
        p.ProcessShortcodes(t)
 
        // TODO(bep) these page handlers need to be re-evaluated, as it is hard to
index 8fe27e506840f7913b3a27c1972084f503ea6166..36b797e7ef1fdc40975e00746ee2755ce97c9180 100644 (file)
@@ -220,7 +220,7 @@ func (h *HugoSites) Build(config BuildCfg) error {
                }
        }
 
-       if err := h.preRender(); err != nil {
+       if err := h.preRender(config, whatChanged{source: true, other: true}); err != nil {
                return err
        }
 
@@ -261,6 +261,10 @@ func (h *HugoSites) Rebuild(config BuildCfg, events ...fsnotify.Event) error {
                return errors.New("Rebuild does not support 'ResetState'. Use Build.")
        }
 
+       if !config.Watching {
+               return errors.New("Rebuild called when not in watch mode")
+       }
+
        h.runMode.Watching = config.Watching
 
        firstSite := h.Sites[0]
@@ -270,7 +274,7 @@ func (h *HugoSites) Rebuild(config BuildCfg, events ...fsnotify.Event) error {
                s.resetBuildState()
        }
 
-       sourceChanged, err := firstSite.reBuild(events)
+       changed, err := firstSite.reBuild(events)
 
        if err != nil {
                return err
@@ -279,7 +283,7 @@ func (h *HugoSites) Rebuild(config BuildCfg, events ...fsnotify.Event) error {
        // Assign pages to sites per translation.
        h.setupTranslations(firstSite)
 
-       if sourceChanged {
+       if changed.source {
                for _, s := range h.Sites {
                        if err := s.postProcess(); err != nil {
                                return err
@@ -287,7 +291,7 @@ func (h *HugoSites) Rebuild(config BuildCfg, events ...fsnotify.Event) error {
                }
        }
 
-       if err := h.preRender(); err != nil {
+       if err := h.preRender(config, changed); err != nil {
                return err
        }
 
@@ -391,7 +395,7 @@ func (h *HugoSites) setupTranslations(master *Site) {
 // preRender performs build tasks that need to be done as late as possible.
 // Shortcode handling is the main task in here.
 // TODO(bep) We need to look at the whole handler-chain construct with he below in mind.
-func (h *HugoSites) preRender() error {
+func (h *HugoSites) preRender(cfg BuildCfg, changed whatChanged) error {
 
        for _, s := range h.Sites {
                if err := s.setCurrentLanguageConfig(); err != nil {
@@ -416,13 +420,13 @@ func (h *HugoSites) preRender() error {
                if err := s.setCurrentLanguageConfig(); err != nil {
                        return err
                }
-               renderShortcodesForSite(s)
+               s.preparePagesForRender(cfg, changed)
        }
 
        return nil
 }
 
-func renderShortcodesForSite(s *Site) {
+func (s *Site) preparePagesForRender(cfg BuildCfg, changed whatChanged) {
        pageChan := make(chan *Page)
        wg := &sync.WaitGroup{}
 
@@ -431,14 +435,37 @@ func renderShortcodesForSite(s *Site) {
                go func(pages <-chan *Page, wg *sync.WaitGroup) {
                        defer wg.Done()
                        for p := range pages {
+
+                               if !changed.other && p.rendered {
+                                       // No need to process it again.
+                                       continue
+                               }
+
+                               // If we got this far it means that this is either a new Page pointer
+                               // or a template or similar has changed so wee need to do a rerendering
+                               // of the shortcodes etc.
+
+                               // Mark it as rendered
+                               p.rendered = true
+
+                               // If in watch mode, we need to keep the original so we can
+                               // repeat this process on rebuild.
+                               if cfg.Watching {
+                                       p.rawContentCopy = make([]byte, len(p.rawContent))
+                                       copy(p.rawContentCopy, p.rawContent)
+                               } else {
+                                       // Just reuse the same slice.
+                                       p.rawContentCopy = p.rawContent
+                               }
+
                                if err := handleShortcodes(p, s.owner.tmpl); err != nil {
                                        jww.ERROR.Printf("Failed to handle shortcodes for page %s: %s", p.BaseFileName(), err)
                                }
 
                                if p.Markup == "markdown" {
-                                       tmpContent, tmpTableOfContents := helpers.ExtractTOC(p.rawContent)
+                                       tmpContent, tmpTableOfContents := helpers.ExtractTOC(p.rawContentCopy)
                                        p.TableOfContents = helpers.BytesToHTML(tmpTableOfContents)
-                                       p.rawContent = tmpContent
+                                       p.rawContentCopy = tmpContent
                                }
 
                                if p.Markup != "html" {
@@ -449,19 +476,25 @@ func renderShortcodesForSite(s *Site) {
                                        if err != nil {
                                                jww.ERROR.Printf("Failed to set use defined summary: %s", err)
                                        } else if summaryContent != nil {
-                                               p.rawContent = summaryContent.content
+                                               p.rawContentCopy = summaryContent.content
                                        }
 
-                                       p.Content = helpers.BytesToHTML(p.rawContent)
-                                       p.rendered = true
+                                       p.Content = helpers.BytesToHTML(p.rawContentCopy)
 
                                        if summaryContent == nil {
                                                p.setAutoSummary()
                                        }
+
+                               } else {
+                                       p.Content = helpers.BytesToHTML(p.rawContentCopy)
                                }
 
+                               // no need for this anymore
+                               p.rawContentCopy = nil
+
                                //analyze for raw stats
                                p.analyzePage()
+
                        }
                }(pageChan, wg)
        }
@@ -490,7 +523,7 @@ func handleShortcodes(p *Page, t tpl.Template) error {
                        return err
                }
 
-               p.rawContent, err = replaceShortcodeTokens(p.rawContent, shortcodePlaceholderPrefix, shortcodes)
+               p.rawContentCopy, err = replaceShortcodeTokens(p.rawContentCopy, shortcodePlaceholderPrefix, shortcodes)
 
                if err != nil {
                        jww.FATAL.Printf("Failed to replace short code tokens in %s:\n%s", p.BaseFileName(), err.Error())
index bfbcd914ef298a5b5970a1b365c5f27829d5c493..b19e9ea2e86d51bee9a009c3ed7b1bd3b5646e14 100644 (file)
@@ -51,7 +51,7 @@ func doTestMultiSitesMainLangInRoot(t *testing.T, defaultInSubDir bool) {
        testCommonResetState()
        viper.Set("DefaultContentLanguageInSubdir", defaultInSubDir)
 
-       sites := createMultiTestSites(t, multiSiteTomlConfig)
+       sites := createMultiTestSites(t, multiSiteTOMLConfig)
 
        err := sites.Build(BuildCfg{})
 
@@ -166,7 +166,7 @@ func TestMultiSitesBuild(t *testing.T) {
                content string
                suffix  string
        }{
-               {multiSiteTomlConfig, "toml"},
+               {multiSiteTOMLConfig, "toml"},
                {multiSiteYAMLConfig, "yml"},
                {multiSiteJSONConfig, "json"},
        } {
@@ -323,8 +323,8 @@ func doTestMultiSitesBuild(t *testing.T, configContent, configSuffix string) {
 
 func TestMultiSitesRebuild(t *testing.T) {
        testCommonResetState()
-       sites := createMultiTestSites(t, multiSiteTomlConfig)
-       cfg := BuildCfg{}
+       sites := createMultiTestSites(t, multiSiteTOMLConfig)
+       cfg := BuildCfg{Watching: true}
 
        err := sites.Build(cfg)
 
@@ -350,6 +350,10 @@ func TestMultiSitesRebuild(t *testing.T) {
        docFr := readDestination(t, "public/fr/sect/doc1/index.html")
        assert.True(t, strings.Contains(docFr, "Bonjour"), "No Bonjour")
 
+       // check single page content
+       assertFileContent(t, "public/fr/sect/doc1/index.html", true, "Single", "Shortcode: Bonjour")
+       assertFileContent(t, "public/en/sect/doc1-slug/index.html", true, "Single", "Shortcode: Hello")
+
        for i, this := range []struct {
                preFunc    func(t *testing.T)
                events     []fsnotify.Event
@@ -474,6 +478,22 @@ func TestMultiSitesRebuild(t *testing.T) {
 
                        },
                },
+               // Change a shortcode
+               {
+                       func(t *testing.T) {
+                               writeSource(t, "layouts/shortcodes/shortcode.html", "Modified Shortcode: {{ i18n \"hello\" }}")
+                       },
+                       []fsnotify.Event{
+                               {Name: "layouts/shortcodes/shortcode.html", Op: fsnotify.Write},
+                       },
+                       func(t *testing.T) {
+                               assert.Len(t, enSite.Pages, 4)
+                               assert.Len(t, enSite.AllPages, 10)
+                               assert.Len(t, frSite.Pages, 4)
+                               assertFileContent(t, "public/fr/sect/doc1/index.html", true, "Single", "Modified Shortcode: Salut")
+                               assertFileContent(t, "public/en/sect/doc1-slug/index.html", true, "Single", "Modified Shortcode: Hello")
+                       },
+               },
        } {
 
                if this.preFunc != nil {
@@ -516,7 +536,7 @@ func assertShouldNotBuild(t *testing.T, sites *HugoSites) {
 func TestAddNewLanguage(t *testing.T) {
        testCommonResetState()
 
-       sites := createMultiTestSites(t, multiSiteTomlConfig)
+       sites := createMultiTestSites(t, multiSiteTOMLConfig)
        cfg := BuildCfg{}
 
        err := sites.Build(cfg)
@@ -525,7 +545,7 @@ func TestAddNewLanguage(t *testing.T) {
                t.Fatalf("Failed to build sites: %s", err)
        }
 
-       newConfig := multiSiteTomlConfig + `
+       newConfig := multiSiteTOMLConfig + `
 
 [Languages.sv]
 weight = 15
@@ -573,7 +593,7 @@ title = "Svenska"
 
 }
 
-var multiSiteTomlConfig = `
+var multiSiteTOMLConfig = `
 DefaultExtension = "html"
 baseurl = "http://example.com/blog"
 DisableSitemap = false
index b09e2b1f7546bf3bcfab1425fd68b91721f93705..7892f222d8f8f2375b3e4128042747fdb2e84fc6 100644 (file)
@@ -48,28 +48,42 @@ var (
 )
 
 type Page struct {
-       Params              map[string]interface{}
-       Content             template.HTML
-       Summary             template.HTML
-       Aliases             []string
-       Status              string
-       Images              []Image
-       Videos              []Video
-       TableOfContents     template.HTML
-       Truncated           bool
-       Draft               bool
-       PublishDate         time.Time
-       ExpiryDate          time.Time
-       Markup              string
-       translations        Pages
-       extension           string
-       contentType         string
-       renderable          bool
-       Layout              string
-       layoutsCalculated   []string
-       linkTitle           string
-       frontmatter         []byte
-       rawContent          []byte
+       Params            map[string]interface{}
+       Content           template.HTML
+       Summary           template.HTML
+       Aliases           []string
+       Status            string
+       Images            []Image
+       Videos            []Video
+       TableOfContents   template.HTML
+       Truncated         bool
+       Draft             bool
+       PublishDate       time.Time
+       ExpiryDate        time.Time
+       Markup            string
+       translations      Pages
+       extension         string
+       contentType       string
+       renderable        bool
+       Layout            string
+       layoutsCalculated []string
+       linkTitle         string
+       frontmatter       []byte
+
+       // rawContent isn't "raw" as in the same as in the content file.
+       // Hugo cares about memory consumption, so we make changes to it to do
+       // markdown rendering etc., but it is "raw enough" so we can do rebuilds
+       // when shortcode changes etc.
+       rawContent []byte
+
+       // When running Hugo in watch mode, we do partial rebuilds and have to make
+       // a copy of the rawContent to be prepared for rebuilds when shortcodes etc.
+       // have changed.
+       rawContentCopy []byte
+
+       // state telling if this is a "new page" or if we have rendered it previously.
+       rendered bool
+
        contentShortCodes   map[string]func() (string, error)
        shortcodes          map[string]shortcode
        plain               string // TODO should be []byte
@@ -84,7 +98,6 @@ type Page struct {
        Source
        Position `json:"-"`
        Node
-       rendered bool
 }
 
 type Source struct {
@@ -220,7 +233,7 @@ var (
 // Returns the page as summary and main if a user defined split is provided.
 func (p *Page) setUserDefinedSummaryIfProvided() (*summaryContent, error) {
 
-       sc := splitUserDefinedSummaryAndContent(p.Markup, p.rawContent)
+       sc := splitUserDefinedSummaryAndContent(p.Markup, p.rawContentCopy)
 
        if sc == nil {
                // No divider found
@@ -1024,19 +1037,9 @@ func (p *Page) SaveSource() error {
 }
 
 func (p *Page) ProcessShortcodes(t tpl.Template) {
-
-       // these short codes aren't used until after Page render,
-       // but processed here to avoid coupling
-       // TODO(bep) Move this and remove p.contentShortCodes
-       if !p.rendered {
-               tmpContent, tmpContentShortCodes, _ := extractAndRenderShortcodes(string(p.rawContent), p, t)
-               p.rawContent = []byte(tmpContent)
-               p.contentShortCodes = tmpContentShortCodes
-       } else {
-               // shortcode template may have changed, rerender
-               p.contentShortCodes = renderShortcodes(p.shortcodes, p, t)
-       }
-
+       tmpContent, tmpContentShortCodes, _ := extractAndRenderShortcodes(string(p.rawContent), p, t)
+       p.rawContent = []byte(tmpContent)
+       p.contentShortCodes = tmpContentShortCodes
 }
 
 func (p *Page) FullFilePath() string {
index 52cd6893bfb75afdfa28909f35c26a487c3ae6f9..854f8989965467125a28e9186b4af021cd7277cd 100644 (file)
@@ -281,10 +281,6 @@ func renderShortcode(sc shortcode, parent *ShortcodeWithPage, p *Page, t tpl.Tem
 
 func extractAndRenderShortcodes(stringToParse string, p *Page, t tpl.Template) (string, map[string]func() (string, error), error) {
 
-       if p.rendered {
-               panic("Illegal state: Page already marked as rendered, please reuse the shortcodes")
-       }
-
        content, shortcodes, err := extractShortcodes(stringToParse, p, t)
 
        if err != nil {
index 9f435218cc35bd735ed437de3e31c4df185285db..22b22773dcb7435724e7eb6f5cc75dcff79f650d 100644 (file)
@@ -450,9 +450,14 @@ func (s *Site) timerStep(step string) {
        s.timer.Step(step)
 }
 
+type whatChanged struct {
+       source bool
+       other  bool
+}
+
 // reBuild partially rebuilds a site given the filesystem events.
 // It returns whetever the content source was changed.
-func (s *Site) reBuild(events []fsnotify.Event) (bool, error) {
+func (s *Site) reBuild(events []fsnotify.Event) (whatChanged, error) {
 
        jww.DEBUG.Printf("Rebuild for events %q", events)
 
@@ -500,7 +505,6 @@ func (s *Site) reBuild(events []fsnotify.Event) (bool, error) {
        }
 
        if len(i18nChanged) > 0 {
-               // TODO(bep ml
                s.readI18nSources()
        }
 
@@ -564,16 +568,6 @@ func (s *Site) reBuild(events []fsnotify.Event) (bool, error) {
        go incrementalReadCollator(s, readResults, pageChan, fileConvChan, coordinator, errs)
        go converterCollator(s, convertResults, errs)
 
-       if len(tmplChanged) > 0 || len(dataChanged) > 0 {
-               // Do not need to read the files again, but they need conversion
-               // for shortocde re-rendering.
-               for _, p := range s.rawAllPages {
-                       if p.shouldBuild() {
-                               pageChan <- p
-                       }
-               }
-       }
-
        for _, ev := range sourceReallyChanged {
 
                file, err := s.reReadFile(ev.Name)
@@ -610,7 +604,12 @@ func (s *Site) reBuild(events []fsnotify.Event) (bool, error) {
 
        s.timerStep("read & convert pages from source")
 
-       return len(sourceChanged) > 0, nil
+       changed := whatChanged{
+               source: len(sourceChanged) > 0,
+               other:  len(tmplChanged) > 0 || len(i18nChanged) > 0 || len(dataChanged) > 0,
+       }
+
+       return changed, nil
 
 }