Fix 0.62.1 server rebuild slowdown regression
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 22 Jan 2020 10:57:23 +0000 (11:57 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Thu, 23 Jan 2020 10:50:02 +0000 (11:50 +0100)
Fixes #6784

hugofs/fileinfo.go
hugofs/rootmapping_fs.go
hugofs/rootmapping_fs_test.go
hugolib/content_render_hooks_test.go
hugolib/filesystems/basefs.go
hugolib/hugo_modules_test.go
hugolib/hugo_sites.go
hugolib/hugo_sites_build.go
hugolib/page__per_output.go
hugolib/testhelpers_test.go

index 893436df17578ec9efd27032aeb701102415dbac..c8a71bf21a5296b3df9306b8a8e68c50e8132e1f 100644 (file)
@@ -34,9 +34,9 @@ import (
 )
 
 const (
-       metaKeyFilename                   = "filename"
-       metaKeyPathFile                   = "pathFile"    // Path of filename relative to a root.
-       metaKeyIsFileMount                = "isFileMount" // Whether the source mount was a file.
+       metaKeyFilename = "filename"
+
+       metaKeyBaseDir                    = "baseDir" // Abs base directory of source file.
        metaKeyMountRoot                  = "mountRoot"
        metaKeyOriginalFilename           = "originalFilename"
        metaKeyName                       = "name"
@@ -116,29 +116,19 @@ func (f FileMeta) Path() string {
        return f.stringV(metaKeyPath)
 }
 
-// PathFile returns the relative file path for the file source. This
-// will in most cases be the same as Path.
+// PathFile returns the relative file path for the file source.
 func (f FileMeta) PathFile() string {
-       pf := f.stringV(metaKeyPathFile)
-       if f.isFileMount() {
-               return pf
-       }
-       mountRoot := f.mountRoot()
-       if mountRoot == pf {
-               return f.Path()
+       base := f.stringV(metaKeyBaseDir)
+       if base == "" {
+               return ""
        }
-
-       return pf + (strings.TrimPrefix(f.Path(), mountRoot))
+       return strings.TrimPrefix(strings.TrimPrefix(f.Filename(), base), filepathSeparator)
 }
 
-func (f FileMeta) mountRoot() string {
+func (f FileMeta) MountRoot() string {
        return f.stringV(metaKeyMountRoot)
 }
 
-func (f FileMeta) isFileMount() bool {
-       return f.GetBool(metaKeyIsFileMount)
-}
-
 func (f FileMeta) Weight() int {
        return f.GetInt(metaKeyWeight)
 }
index dd60452fc5997f85315333c8ef03904a98b8e0df..2196be8e0c90f31a91134d1878400e78cdc8793c 100644 (file)
@@ -57,12 +57,8 @@ func NewRootMappingFs(fs afero.Fs, rms ...RootMapping) (*RootMappingFs, error) {
                // Extract "blog" from "content/blog"
                rm.path = strings.TrimPrefix(strings.TrimPrefix(rm.From, fromBase), filepathSeparator)
                if rm.Meta != nil {
-                       rm.Meta[metaKeyIsFileMount] = !fi.IsDir()
+                       rm.Meta[metaKeyBaseDir] = rm.ToBasedir
                        rm.Meta[metaKeyMountRoot] = rm.path
-                       if rm.ToBasedir != "" {
-                               pathFile := strings.TrimPrefix(strings.TrimPrefix(rm.To, rm.ToBasedir), filepathSeparator)
-                               rm.Meta[metaKeyPathFile] = pathFile
-                       }
                }
 
                meta := copyFileMeta(rm.Meta)
index 7d685c77eb073af9c0fd62cdef81220a421f1976..f7637a61f45367c310bd8153474a5d4826669f0d 100644 (file)
@@ -271,7 +271,7 @@ func TestRootMappingFsMount(t *testing.T) {
        c.Assert(singles, qt.HasLen, 2)
        for i, lang := range []string{"no", "sv"} {
                fi := singles[i].(FileMetaInfo)
-               c.Assert(fi.Meta().PathFile(), qt.Equals, lang+".txt")
+               c.Assert(fi.Meta().PathFile(), qt.Equals, filepath.FromSlash("themes/a/singlefiles/"+lang+".txt"))
                c.Assert(fi.Meta().Lang(), qt.Equals, lang)
                c.Assert(fi.Name(), qt.Equals, "p1.md")
        }
index 8aba1dd8cc82aff29cb227449a692c193797b679..5290ebcbda6d6732cc022b1e46370ccc40c3c4da 100644 (file)
 package hugolib
 
 import (
+       "fmt"
        "testing"
+
+       qt "github.com/frankban/quicktest"
 )
 
 func TestRenderHooks(t *testing.T) {
@@ -118,7 +121,20 @@ title: With RenderString
 {{< myshortcode5 >}}Inner Link: [Inner Link](https://www.gohugo.io "Hugo's Homepage"){{< /myshortcode5 >}}
 
 `)
-       b.Build(BuildCfg{})
+
+       for i := 1; i <= 30; i++ {
+               // Add some content with no shortcodes or links, i.e no templates needed.
+               b.WithContent(fmt.Sprintf("blog/notempl%d.md", i), `---
+title: No Template
+---
+
+## Content
+`)
+       }
+       counters := &testCounters{}
+       b.Build(BuildCfg{testCounters: counters})
+       b.Assert(int(counters.contentRenderCounter), qt.Equals, 50)
+
        b.AssertFileContent("public/blog/p1/index.html", `
 <p>Cool Page|https://www.google.com|Title: Google's Homepage|Text: First Link|END</p>
 Text: Second
@@ -149,7 +165,11 @@ SHORT3|
                "layouts/shortcodes/myshortcode3.html", `SHORT3_EDITED|`,
        )
 
-       b.Build(BuildCfg{})
+       counters = &testCounters{}
+       b.Build(BuildCfg{testCounters: counters})
+       // Make sure that only content using the changed templates are re-rendered.
+       b.Assert(int(counters.contentRenderCounter), qt.Equals, 7)
+
        b.AssertFileContent("public/customview/p1/index.html", `.Render: myrender: Custom View|P4: PARTIAL4_EDITED`)
        b.AssertFileContent("public/blog/p1/index.html", `<p>EDITED: https://www.google.com|</p>`, "SHORT3_EDITED|")
        b.AssertFileContent("public/blog/p2/index.html", `PARTIAL1_EDITED`)
index cf9ff3c38b2e1ec750591b01cf98a8d05e2dd51e..5cede88d0f8e16997e770684970029a6f965aefd 100644 (file)
@@ -295,21 +295,16 @@ func (d *SourceFilesystem) Contains(filename string) bool {
        return false
 }
 
-// Path returns the relative path to the given filename if it is a member of
+// Path returns the mount relative path to the given filename if it is a member of
 // of the current filesystem, an empty string if not.
 func (d *SourceFilesystem) Path(filename string) string {
        for _, dir := range d.Dirs {
                meta := dir.Meta()
-               if !dir.IsDir() {
-                       if filename == meta.Filename() {
-                               return meta.PathFile()
-                       }
-                       continue
-               }
-
                if strings.HasPrefix(filename, meta.Filename()) {
                        p := strings.TrimPrefix(strings.TrimPrefix(filename, meta.Filename()), filePathSeparator)
-                       p = path.Join(meta.PathFile(), p)
+                       if mountRoot := meta.MountRoot(); mountRoot != "" {
+                               return filepath.Join(mountRoot, p)
+                       }
                        return p
                }
        }
index 5c2b46b30c753316e08b619e31e9bf3212674405..4b71a54c87801ffc245a04d101857fd34469bf05 100644 (file)
@@ -668,6 +668,126 @@ Readme Edit
 
 }
 
+func TestMountsPaths(t *testing.T) {
+       c := qt.New(t)
+
+       type test struct {
+               b          *sitesBuilder
+               clean      func()
+               workingDir string
+       }
+
+       prepare := func(c *qt.C, mounts string) test {
+               workingDir, clean, err := htesting.CreateTempDir(hugofs.Os, "hugo-mounts-paths")
+               c.Assert(err, qt.IsNil)
+
+               configTemplate := `
+baseURL = "https://example.com"
+title = "My Modular Site"
+workingDir = %q
+
+%s
+
+`
+               config := fmt.Sprintf(configTemplate, workingDir, mounts)
+               config = strings.Replace(config, "WORKING_DIR", workingDir, -1)
+
+               b := newTestSitesBuilder(c).Running()
+
+               b.Fs = hugofs.NewDefault(viper.New())
+
+               os.MkdirAll(filepath.Join(workingDir, "content", "blog"), 0777)
+
+               b.WithWorkingDir(workingDir).WithConfigFile("toml", config)
+
+               return test{
+                       b:          b,
+                       clean:      clean,
+                       workingDir: workingDir,
+               }
+
+       }
+
+       c.Run("Default", func(c *qt.C) {
+               mounts := ``
+
+               test := prepare(c, mounts)
+               b := test.b
+               defer test.clean()
+
+               b.WithContent("blog/p1.md", `---
+title: P1
+---`)
+
+               b.Build(BuildCfg{})
+
+               p := b.GetPage("blog/p1.md")
+               f := p.File().FileInfo().Meta()
+               b.Assert(filepath.ToSlash(f.Path()), qt.Equals, "blog/p1.md")
+               b.Assert(filepath.ToSlash(f.PathFile()), qt.Equals, "content/blog/p1.md")
+
+               b.Assert(b.H.BaseFs.Layouts.Path(filepath.Join(test.workingDir, "layouts", "_default", "single.html")), qt.Equals, filepath.FromSlash("_default/single.html"))
+
+       })
+
+       c.Run("Mounts", func(c *qt.C) {
+               absDir, clean, err := htesting.CreateTempDir(hugofs.Os, "hugo-mounts-paths-abs")
+               c.Assert(err, qt.IsNil)
+               defer clean()
+
+               mounts := `[module]
+  [[module.mounts]]
+    source = "README.md"
+    target = "content/_index.md"
+  [[module.mounts]]
+    source = "mycontent"
+    target = "content/blog"
+   [[module.mounts]]
+    source = "subdir/mypartials"
+    target = "layouts/partials"
+   [[module.mounts]]
+    source = %q
+    target = "layouts/shortcodes"
+`
+               mounts = fmt.Sprintf(mounts, filepath.Join(absDir, "/abs/myshortcodes"))
+
+               test := prepare(c, mounts)
+               b := test.b
+               defer test.clean()
+
+               subContentDir := filepath.Join(test.workingDir, "mycontent", "sub")
+               os.MkdirAll(subContentDir, 0777)
+               myPartialsDir := filepath.Join(test.workingDir, "subdir", "mypartials")
+               os.MkdirAll(myPartialsDir, 0777)
+
+               absShortcodesDir := filepath.Join(absDir, "abs", "myshortcodes")
+               os.MkdirAll(absShortcodesDir, 0777)
+
+               b.WithSourceFile("README.md", "---\ntitle: Readme\n---")
+               b.WithSourceFile("mycontent/sub/p1.md", "---\ntitle: P1\n---")
+
+               b.WithSourceFile(filepath.Join(absShortcodesDir, "myshort.html"), "MYSHORT")
+               b.WithSourceFile(filepath.Join(myPartialsDir, "mypartial.html"), "MYPARTIAL")
+
+               b.Build(BuildCfg{})
+
+               p1_1 := b.GetPage("/blog/sub/p1.md")
+               p1_2 := b.GetPage("/mycontent/sub/p1.md")
+               b.Assert(p1_1, qt.Not(qt.IsNil))
+               b.Assert(p1_2, qt.Equals, p1_1)
+
+               f := p1_1.File().FileInfo().Meta()
+               b.Assert(filepath.ToSlash(f.Path()), qt.Equals, "blog/sub/p1.md")
+               b.Assert(filepath.ToSlash(f.PathFile()), qt.Equals, "mycontent/sub/p1.md")
+               b.Assert(b.H.BaseFs.Layouts.Path(filepath.Join(myPartialsDir, "mypartial.html")), qt.Equals, filepath.FromSlash("partials/mypartial.html"))
+               b.Assert(b.H.BaseFs.Layouts.Path(filepath.Join(absShortcodesDir, "myshort.html")), qt.Equals, filepath.FromSlash("shortcodes/myshort.html"))
+               b.Assert(b.H.BaseFs.Content.Path(filepath.Join(subContentDir, "p1.md")), qt.Equals, filepath.FromSlash("blog/sub/p1.md"))
+               b.Assert(b.H.BaseFs.Content.Path(filepath.Join(test.workingDir, "README.md")), qt.Equals, filepath.FromSlash("_index.md"))
+
+       })
+
+}
+
 // https://github.com/gohugoio/hugo/issues/6299
 func TestSiteWithGoModButNoModules(t *testing.T) {
        t.Parallel()
index a0c62f01e238f7f7e3153acb49416e0fd7fe4abd..50694fbba18e7843287ca7b3f82fe994dbaa8834 100644 (file)
@@ -19,6 +19,7 @@ import (
        "sort"
        "strings"
        "sync"
+       "sync/atomic"
 
        "github.com/gohugoio/hugo/identity"
 
@@ -82,6 +83,19 @@ type HugoSites struct {
        init *hugoSitesInit
 
        *fatalErrorHandler
+       *testCounters
+}
+
+// Only used in tests.
+type testCounters struct {
+       contentRenderCounter uint64
+}
+
+func (h *testCounters) IncrContentRender() {
+       if h == nil {
+               return
+       }
+       atomic.AddUint64(&h.contentRenderCounter, 1)
 }
 
 type fatalErrorHandler struct {
@@ -579,6 +593,8 @@ type BuildCfg struct {
 
        // Recently visited URLs. This is used for partial re-rendering.
        RecentlyVisited map[string]bool
+
+       testCounters *testCounters
 }
 
 // shouldRender is used in the Fast Render Mode to determine if we need to re-render
index 901941bda61cc502d8cf76ec2fe69797b098d1b2..cf7b1431145906f9621b5b69751b386c2d4a7a7a 100644 (file)
@@ -66,6 +66,8 @@ func (h *HugoSites) Build(config BuildCfg, events ...fsnotify.Event) error {
                h.Metrics.Reset()
        }
 
+       h.testCounters = config.testCounters
+
        // Need a pointer as this may be modified.
        conf := &config
 
index 59440c7cb7a3be40bab18304c930dde5a54a374f..330b0d75d0935a7e0882bd97200984fae3812bad 100644 (file)
@@ -80,6 +80,8 @@ func newPageContentOutput(p *pageState, po *pageOutput) (*pageContentOutput, err
        }
 
        initContent := func() (err error) {
+               p.s.h.IncrContentRender()
+
                if p.cmap == nil {
                        // Nothing to do.
                        return nil
index 376c0899c9542f6295065ed94c059bbd60d6ba82..7a87245a6bf760ecccb43c930861d6e49e48b8d3 100644 (file)
@@ -224,6 +224,9 @@ func (s *sitesBuilder) WithSourceFile(filenameContent ...string) *sitesBuilder {
 
 func (s *sitesBuilder) absFilename(filename string) string {
        filename = filepath.FromSlash(filename)
+       if filepath.IsAbs(filename) {
+               return filename
+       }
        if s.workingDir != "" && !strings.HasPrefix(filename, s.workingDir) {
                filename = filepath.Join(s.workingDir, filename)
        }
@@ -736,6 +739,12 @@ func (s *sitesBuilder) CheckExists(filename string) bool {
        return destinationExists(s.Fs, filepath.Clean(filename))
 }
 
+func (s *sitesBuilder) GetPage(ref string) page.Page {
+       p, err := s.H.Sites[0].getPageNew(nil, ref)
+       s.Assert(err, qt.IsNil)
+       return p
+}
+
 func newTestHelper(cfg config.Provider, fs *hugofs.Fs, t testing.TB) testHelper {
        return testHelper{
                Cfg: cfg,