mirror of
https://github.com/gohugoio/hugo.git
synced 2024-11-21 20:46:30 -05:00
Add a cross process build lock and use it in the archetype content builder
Fixes #9048
This commit is contained in:
parent
c7957c90e8
commit
ba35e69856
7 changed files with 79 additions and 32 deletions
|
@ -278,7 +278,8 @@ func isTerminal() bool {
|
||||||
return terminal.IsTerminal(os.Stdout)
|
return terminal.IsTerminal(os.Stdout)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *commandeer) fullBuild() error {
|
func (c *commandeer) fullBuild(noBuildLock bool) error {
|
||||||
|
|
||||||
var (
|
var (
|
||||||
g errgroup.Group
|
g errgroup.Group
|
||||||
langCount map[string]uint64
|
langCount map[string]uint64
|
||||||
|
@ -303,7 +304,7 @@ func (c *commandeer) fullBuild() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
buildSitesFunc := func() error {
|
buildSitesFunc := func() error {
|
||||||
if err := c.buildSites(); err != nil {
|
if err := c.buildSites(noBuildLock); err != nil {
|
||||||
return errors.Wrap(err, "Error building site")
|
return errors.Wrap(err, "Error building site")
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -496,7 +497,7 @@ func (c *commandeer) build() error {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
if err := c.fullBuild(); err != nil {
|
if err := c.fullBuild(false); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -551,7 +552,7 @@ func (c *commandeer) serverBuild() error {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
if err := c.fullBuild(); err != nil {
|
if err := c.fullBuild(false); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -721,8 +722,8 @@ func (c *commandeer) getDirList() ([]string, error) {
|
||||||
return filenames, nil
|
return filenames, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *commandeer) buildSites() (err error) {
|
func (c *commandeer) buildSites(noBuildLock bool) (err error) {
|
||||||
return c.hugo().Build(hugolib.BuildCfg{})
|
return c.hugo().Build(hugolib.BuildCfg{NoBuildLock: noBuildLock})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *commandeer) handleBuildErr(err error, msg string) {
|
func (c *commandeer) handleBuildErr(err error, msg string) {
|
||||||
|
@ -750,7 +751,7 @@ func (c *commandeer) rebuildSites(events []fsnotify.Event) error {
|
||||||
visited[home] = true
|
visited[home] = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return c.hugo().Build(hugolib.BuildCfg{RecentlyVisited: visited, ErrRecovery: c.wasError}, events...)
|
return c.hugo().Build(hugolib.BuildCfg{NoBuildLock: true, RecentlyVisited: visited, ErrRecovery: c.wasError}, events...)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *commandeer) partialReRender(urls ...string) error {
|
func (c *commandeer) partialReRender(urls ...string) error {
|
||||||
|
@ -762,7 +763,7 @@ func (c *commandeer) partialReRender(urls ...string) error {
|
||||||
for _, url := range urls {
|
for _, url := range urls {
|
||||||
visited[url] = true
|
visited[url] = true
|
||||||
}
|
}
|
||||||
return c.hugo().Build(hugolib.BuildCfg{RecentlyVisited: visited, PartialReRender: true, ErrRecovery: c.wasError})
|
return c.hugo().Build(hugolib.BuildCfg{NoBuildLock: true, RecentlyVisited: visited, PartialReRender: true, ErrRecovery: c.wasError})
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *commandeer) fullRebuild(changeType string) {
|
func (c *commandeer) fullRebuild(changeType string) {
|
||||||
|
@ -809,7 +810,7 @@ func (c *commandeer) fullRebuild(changeType string) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
err = c.buildSites()
|
err = c.buildSites(true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
c.logger.Errorln(err)
|
c.logger.Errorln(err)
|
||||||
} else if !c.h.buildWatch && !c.Cfg.GetBool("disableLiveReload") {
|
} else if !c.h.buildWatch && !c.Cfg.GetBool("disableLiveReload") {
|
||||||
|
@ -864,11 +865,17 @@ func (c *commandeer) newWatcher(pollIntervalStr string, dirList ...string) (*wat
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case evs := <-watcher.Events:
|
case evs := <-watcher.Events:
|
||||||
|
unlock, err := c.hugo().BaseFs.LockBuild()
|
||||||
|
if err != nil {
|
||||||
|
c.logger.Errorln("Failed to acquire a build lock: %s", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
c.handleEvents(watcher, staticSyncer, evs, configSet)
|
c.handleEvents(watcher, staticSyncer, evs, configSet)
|
||||||
if c.showErrorInBrowser && c.errCount() > 0 {
|
if c.showErrorInBrowser && c.errCount() > 0 {
|
||||||
// Need to reload browser to show the error
|
// Need to reload browser to show the error
|
||||||
livereload.ForceRefresh()
|
livereload.ForceRefresh()
|
||||||
}
|
}
|
||||||
|
unlock()
|
||||||
case err := <-watcher.Errors():
|
case err := <-watcher.Errors():
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil && !os.IsNotExist(err) {
|
||||||
c.logger.Errorln("Error while watching:", err)
|
c.logger.Errorln("Error while watching:", err)
|
||||||
|
|
|
@ -53,6 +53,12 @@ draft: true
|
||||||
// NewContent creates a new content file in h (or a full bundle if the archetype is a directory)
|
// NewContent creates a new content file in h (or a full bundle if the archetype is a directory)
|
||||||
// in targetPath.
|
// in targetPath.
|
||||||
func NewContent(h *hugolib.HugoSites, kind, targetPath string) error {
|
func NewContent(h *hugolib.HugoSites, kind, targetPath string) error {
|
||||||
|
unlock, err := h.BaseFs.LockBuild()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to acquire a build lock: %s", err)
|
||||||
|
}
|
||||||
|
defer unlock()
|
||||||
|
|
||||||
cf := hugolib.NewContentFactory(h)
|
cf := hugolib.NewContentFactory(h)
|
||||||
|
|
||||||
if kind == "" {
|
if kind == "" {
|
||||||
|
@ -138,7 +144,7 @@ func (b *contentBuilder) buildDir() error {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := b.h.Build(hugolib.BuildCfg{SkipRender: true, ContentInclusionFilter: contentInclusionFilter}); err != nil {
|
if err := b.h.Build(hugolib.BuildCfg{NoBuildLock: true, SkipRender: true, ContentInclusionFilter: contentInclusionFilter}); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -200,7 +206,7 @@ func (b *contentBuilder) buildFile() error {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := b.h.Build(hugolib.BuildCfg{SkipRender: true, ContentInclusionFilter: contentInclusionFilter}); err != nil {
|
if err := b.h.Build(hugolib.BuildCfg{NoBuildLock: true, SkipRender: true, ContentInclusionFilter: contentInclusionFilter}); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -25,6 +25,18 @@ import (
|
||||||
"github.com/spf13/afero"
|
"github.com/spf13/afero"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// IsTest reports whether we're running as a test.
|
||||||
|
var IsTest bool
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
for _, arg := range os.Args {
|
||||||
|
if strings.HasPrefix(arg, "-test.") {
|
||||||
|
IsTest = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// CreateTempDir creates a temp dir in the given filesystem and
|
// CreateTempDir creates a temp dir in the given filesystem and
|
||||||
// returns the dirnam and a func that removes it when done.
|
// returns the dirnam and a func that removes it when done.
|
||||||
func CreateTempDir(fs afero.Fs, prefix string) (string, func(), error) {
|
func CreateTempDir(fs afero.Fs, prefix string) (string, func(), error) {
|
||||||
|
|
|
@ -24,7 +24,10 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/gohugoio/hugo/htesting"
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/common/loggers"
|
"github.com/gohugoio/hugo/common/loggers"
|
||||||
|
"github.com/rogpeppe/go-internal/lockedfile"
|
||||||
|
|
||||||
"github.com/gohugoio/hugo/hugofs/files"
|
"github.com/gohugoio/hugo/hugofs/files"
|
||||||
|
|
||||||
|
@ -38,6 +41,13 @@ import (
|
||||||
"github.com/spf13/afero"
|
"github.com/spf13/afero"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Used to control concurrency between multiple Hugo instances, e.g.
|
||||||
|
// a running server and building new content with 'hugo new'.
|
||||||
|
// It's placed in the project root.
|
||||||
|
lockFileBuild = ".hugo_build.lock"
|
||||||
|
)
|
||||||
|
|
||||||
var filePathSeparator = string(filepath.Separator)
|
var filePathSeparator = string(filepath.Separator)
|
||||||
|
|
||||||
// BaseFs contains the core base filesystems used by Hugo. The name "base" is used
|
// BaseFs contains the core base filesystems used by Hugo. The name "base" is used
|
||||||
|
@ -56,6 +66,21 @@ type BaseFs struct {
|
||||||
PublishFs afero.Fs
|
PublishFs afero.Fs
|
||||||
|
|
||||||
theBigFs *filesystemsCollector
|
theBigFs *filesystemsCollector
|
||||||
|
|
||||||
|
// Locks.
|
||||||
|
buildMu *lockedfile.Mutex // <project>/.hugo_build.lock
|
||||||
|
buildMuTests sync.Mutex // Used in tests.
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tries to acquire a build lock.
|
||||||
|
func (fs *BaseFs) LockBuild() (unlock func(), err error) {
|
||||||
|
if htesting.IsTest {
|
||||||
|
fs.buildMuTests.Lock()
|
||||||
|
return func() {
|
||||||
|
fs.buildMuTests.Unlock()
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
return fs.buildMu.Lock()
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(bep) we can get regular files in here and that is fine, but
|
// TODO(bep) we can get regular files in here and that is fine, but
|
||||||
|
@ -402,6 +427,7 @@ func NewBase(p *paths.Paths, logger loggers.Logger, options ...func(*BaseFs) err
|
||||||
b := &BaseFs{
|
b := &BaseFs{
|
||||||
SourceFs: sourceFs,
|
SourceFs: sourceFs,
|
||||||
PublishFs: publishFs,
|
PublishFs: publishFs,
|
||||||
|
buildMu: lockedfile.MutexAt(filepath.Join(p.WorkingDir, lockFileBuild)),
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, opt := range options {
|
for _, opt := range options {
|
||||||
|
|
|
@ -70,9 +70,6 @@ type HugoSites struct {
|
||||||
// If this is running in the dev server.
|
// If this is running in the dev server.
|
||||||
running bool
|
running bool
|
||||||
|
|
||||||
// Serializes rebuilds when server is running.
|
|
||||||
runningMu sync.Mutex
|
|
||||||
|
|
||||||
// Render output formats for all sites.
|
// Render output formats for all sites.
|
||||||
renderFormats output.Formats
|
renderFormats output.Formats
|
||||||
|
|
||||||
|
@ -682,6 +679,9 @@ type BuildCfg struct {
|
||||||
// Can be set to build only with a sub set of the content source.
|
// Can be set to build only with a sub set of the content source.
|
||||||
ContentInclusionFilter *glob.FilenameFilter
|
ContentInclusionFilter *glob.FilenameFilter
|
||||||
|
|
||||||
|
// Set when the buildlock is already acquired (e.g. the archetype content builder).
|
||||||
|
NoBuildLock bool
|
||||||
|
|
||||||
testCounters *testCounters
|
testCounters *testCounters
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -44,19 +44,17 @@ import (
|
||||||
// Build builds all sites. If filesystem events are provided,
|
// Build builds all sites. If filesystem events are provided,
|
||||||
// this is considered to be a potential partial rebuild.
|
// this is considered to be a potential partial rebuild.
|
||||||
func (h *HugoSites) Build(config BuildCfg, events ...fsnotify.Event) error {
|
func (h *HugoSites) Build(config BuildCfg, events ...fsnotify.Event) error {
|
||||||
if h.running {
|
|
||||||
// Make sure we don't trigger rebuilds in parallel.
|
|
||||||
h.runningMu.Lock()
|
|
||||||
defer h.runningMu.Unlock()
|
|
||||||
} else {
|
|
||||||
defer func() {
|
|
||||||
h.Close()
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx, task := trace.NewTask(context.Background(), "Build")
|
ctx, task := trace.NewTask(context.Background(), "Build")
|
||||||
defer task.End()
|
defer task.End()
|
||||||
|
|
||||||
|
if !config.NoBuildLock {
|
||||||
|
unlock, err := h.BaseFs.LockBuild()
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "failed to acquire a build lock")
|
||||||
|
}
|
||||||
|
defer unlock()
|
||||||
|
}
|
||||||
|
|
||||||
errCollector := h.StartErrorCollector()
|
errCollector := h.StartErrorCollector()
|
||||||
errs := make(chan error)
|
errs := make(chan error)
|
||||||
|
|
||||||
|
|
|
@ -1099,16 +1099,14 @@ class-in-b {
|
||||||
err = build("never", true)
|
err = build("never", true)
|
||||||
|
|
||||||
err = herrors.UnwrapErrorWithFileContext(err)
|
err = herrors.UnwrapErrorWithFileContext(err)
|
||||||
fe, ok := err.(*herrors.ErrorWithFileContext)
|
_, ok := err.(*herrors.ErrorWithFileContext)
|
||||||
b.Assert(ok, qt.Equals, true)
|
b.Assert(ok, qt.Equals, true)
|
||||||
|
|
||||||
if os.Getenv("CI") == "" {
|
// TODO(bep) for some reason, we have starting to get
|
||||||
// TODO(bep) for some reason, we have starting to get
|
// execute of template failed: template: index.html:5:25
|
||||||
// execute of template failed: template: index.html:5:25
|
// on CI (GitHub action).
|
||||||
// on CI (GitHub action).
|
//b.Assert(fe.Position().LineNumber, qt.Equals, 5)
|
||||||
b.Assert(fe.Position().LineNumber, qt.Equals, 5)
|
//b.Assert(fe.Error(), qt.Contains, filepath.Join(workDir, "assets/css/components/b.css:4:1"))
|
||||||
b.Assert(fe.Error(), qt.Contains, filepath.Join(workDir, "assets/css/components/b.css:4:1"))
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove PostCSS
|
// Remove PostCSS
|
||||||
b.Assert(os.RemoveAll(filepath.Join(workDir, "node_modules")), qt.IsNil)
|
b.Assert(os.RemoveAll(filepath.Join(workDir, "node_modules")), qt.IsNil)
|
||||||
|
|
Loading…
Reference in a new issue