Revert "Revert "Allow rendering static files to disk and dynamic to memory in server...
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Mon, 14 Mar 2022 15:02:04 +0000 (16:02 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 8 Apr 2022 11:26:16 +0000 (13:26 +0200)
This reverts commit 64b7b7a89753a39661219b2fcb92d7f185a03f63.

commands/commandeer.go
commands/hugo.go
commands/server.go
commands/static_syncer.go
hugofs/fs.go
hugolib/filesystems/basefs.go
hugolib/pages_process.go
hugolib/site.go

index 8a19258931ed3db818628aa6120ae393b04363fa..6f74bf8640db4c602f2ca594150492aa6bebb26b 100644 (file)
@@ -94,6 +94,7 @@ type commandeer struct {
        languagesConfigured bool
        languages           langs.Languages
        doLiveReload        bool
+       renderStaticToDisk  bool
        fastRenderMode      bool
        showErrorInBrowser  bool
        wasError            bool
@@ -375,8 +376,9 @@ func (c *commandeer) loadConfig() error {
        }
 
        createMemFs := config.GetBool("renderToMemory")
+       c.renderStaticToDisk = config.GetBool("renderStaticToDisk")
 
-       if createMemFs {
+       if createMemFs && !c.renderStaticToDisk {
                // Rendering to memoryFS, publish to Root regardless of publishDir.
                config.Set("publishDir", "/")
        }
@@ -387,6 +389,14 @@ func (c *commandeer) loadConfig() error {
                if c.destinationFs != nil {
                        // Need to reuse the destination on server rebuilds.
                        fs.Destination = c.destinationFs
+               } else if createMemFs && c.renderStaticToDisk {
+                       // Writes the dynamic output on memory,
+                       // while serve others directly from publishDir
+                       publishDir := config.GetString("publishDir")
+                       writableFs := afero.NewBasePathFs(afero.NewMemMapFs(), publishDir)
+                       publicFs := afero.NewOsFs()
+                       fs.Destination = afero.NewCopyOnWriteFs(afero.NewReadOnlyFs(publicFs), writableFs)
+                       fs.DestinationStatic = publicFs
                } else if createMemFs {
                        // Hugo writes the output to memory instead of the disk.
                        fs.Destination = new(afero.MemMapFs)
@@ -404,11 +414,13 @@ func (c *commandeer) loadConfig() error {
 
                        changeDetector.PrepareNew()
                        fs.Destination = hugofs.NewHashingFs(fs.Destination, changeDetector)
+                       fs.DestinationStatic = hugofs.NewHashingFs(fs.DestinationStatic, changeDetector)
                        c.changeDetector = changeDetector
                }
 
                if c.Cfg.GetBool("logPathWarnings") {
                        fs.Destination = hugofs.NewCreateCountingFs(fs.Destination)
+                       fs.DestinationStatic = hugofs.NewCreateCountingFs(fs.DestinationStatic)
                }
 
                // To debug hard-to-find path issues.
index 8c5294f006daffce37887977f87afaec5f716a4d..21140fa4307b89f9a95ad68f55918220c9cf5b55 100644 (file)
@@ -652,6 +652,9 @@ func (c *commandeer) copyStaticTo(sourceFs *filesystems.SourceFilesystem) (uint6
        syncer.ChmodFilter = chmodFilter
        syncer.SrcFs = fs
        syncer.DestFs = c.Fs.Destination
+       if c.renderStaticToDisk {
+               syncer.DestFs = c.Fs.DestinationStatic
+       }
        // Now that we are using a unionFs for the static directories
        // We can effectively clean the publishDir on initial sync
        syncer.Delete = c.Cfg.GetBool("cleanDestinationDir")
index bb6a4e15d4581177489316f39146603e9e3438ae..23301d004f65f30a690c0c7c8134e4275d737262 100644 (file)
@@ -50,15 +50,16 @@ type serverCmd struct {
        // Can be used to stop the server. Useful in tests
        stop chan bool
 
-       disableLiveReload bool
-       navigateToChanged bool
-       renderToDisk      bool
-       serverAppend      bool
-       serverInterface   string
-       serverPort        int
-       liveReloadPort    int
-       serverWatch       bool
-       noHTTPCache       bool
+       disableLiveReload  bool
+       navigateToChanged  bool
+       renderToDisk       bool
+       renderStaticToDisk bool
+       serverAppend       bool
+       serverInterface    string
+       serverPort         int
+       liveReloadPort     int
+       serverWatch        bool
+       noHTTPCache        bool
 
        disableFastRender   bool
        disableBrowserError bool
@@ -109,6 +110,7 @@ of a second, you will be able to save and see your changes nearly instantly.`,
        cc.cmd.Flags().BoolVar(&cc.renderToDisk, "renderToDisk", false, "render to Destination path (default is render to memory & serve from there)")
        cc.cmd.Flags().BoolVar(&cc.disableFastRender, "disableFastRender", false, "enables full re-renders on changes")
        cc.cmd.Flags().BoolVar(&cc.disableBrowserError, "disableBrowserError", false, "do not show build errors in the browser")
+       cc.cmd.Flags().BoolVar(&cc.renderStaticToDisk, "renderStaticToDisk", false, "render static files to disk but dynamic files render to memory.")
 
        cc.cmd.Flags().String("memstats", "", "log memory usage to this file")
        cc.cmd.Flags().String("meminterval", "100ms", "interval to poll memory usage (requires --memstats), valid time units are \"ns\", \"us\" (or \"µs\"), \"ms\", \"s\", \"m\", \"h\".")
@@ -147,6 +149,7 @@ func (sc *serverCmd) server(cmd *cobra.Command, args []string) error {
 
        cfgInit := func(c *commandeer) (rerr error) {
                c.Set("renderToMemory", !sc.renderToDisk)
+               c.Set("renderStaticToDisk", sc.renderStaticToDisk)
                if cmd.Flags().Changed("navigateToChanged") {
                        c.Set("navigateToChanged", sc.navigateToChanged)
                }
@@ -340,6 +343,8 @@ func (f *fileServer) createEndpoint(i int) (*http.ServeMux, net.Listener, string
        if i == 0 {
                if f.s.renderToDisk {
                        jww.FEEDBACK.Println("Serving pages from " + absPublishDir)
+               } else if f.s.renderStaticToDisk {
+                       jww.FEEDBACK.Println("Serving pages from memory and static files from " + absPublishDir)
                } else {
                        jww.FEEDBACK.Println("Serving pages from memory")
                }
index 5569d4de6882c435b1c22a78b811db60094746c6..2eb2b666233883d883a2c33639c436fa6d2f60bd 100644 (file)
@@ -56,6 +56,9 @@ func (s *staticSyncer) syncsStaticEvents(staticEvents []fsnotify.Event) error {
                syncer.ChmodFilter = chmodFilter
                syncer.SrcFs = sourceFs.Fs
                syncer.DestFs = c.Fs.Destination
+               if c.renderStaticToDisk {
+                       syncer.DestFs = c.Fs.DestinationStatic
+               }
 
                // prevent spamming the log on changes
                logger := helpers.NewDistinctErrorLogger()
@@ -101,7 +104,11 @@ func (s *staticSyncer) syncsStaticEvents(staticEvents []fsnotify.Event) error {
                                        toRemove := filepath.Join(publishDir, relPath)
 
                                        logger.Println("File no longer exists in static dir, removing", toRemove)
-                                       _ = c.Fs.Destination.RemoveAll(toRemove)
+                                       if c.renderStaticToDisk {
+                                               _ = c.Fs.DestinationStatic.RemoveAll(toRemove)
+                                       } else {
+                                               _ = c.Fs.Destination.RemoveAll(toRemove)
+                                       }
                                } else if err == nil {
                                        // If file still exists, sync it
                                        logger.Println("Syncing", relPath, "to", publishDir)
index 54d962553e70ae6ba2f8a5e7babc40fb72fca01c..95645204e9f6d0e434ec4ad214ffd4a77fc2315c 100644 (file)
@@ -35,6 +35,9 @@ type Fs struct {
        // Destination is Hugo's destination file system.
        Destination afero.Fs
 
+       // Destination used for `renderStaticToDisk`
+       DestinationStatic afero.Fs
+
        // Os is an OS file system.
        // NOTE: Field is currently unused.
        Os afero.Fs
@@ -69,10 +72,11 @@ func NewFrom(fs afero.Fs, cfg config.Provider) *Fs {
 
 func newFs(base afero.Fs, cfg config.Provider) *Fs {
        return &Fs{
-               Source:      base,
-               Destination: base,
-               Os:          &afero.OsFs{},
-               WorkingDir:  getWorkingDirFs(base, cfg),
+               Source:            base,
+               Destination:       base,
+               DestinationStatic: base,
+               Os:                &afero.OsFs{},
+               WorkingDir:        getWorkingDirFs(base, cfg),
        }
 }
 
index 6e3f88a4be1cdcd02f94ffc15d2c9a29a5e8dd75..0290d2e1cf32e71e234c2ae1029cda3183ce2adc 100644 (file)
@@ -71,6 +71,9 @@ type BaseFs struct {
        // A read-only filesystem starting from the project workDir.
        WorkDir afero.Fs
 
+       // The filesystem used for renderStaticToDisk.
+       PublishFsStatic afero.Fs
+
        theBigFs *filesystemsCollector
 
        // Locks.
@@ -438,15 +441,17 @@ func NewBase(p *paths.Paths, logger loggers.Logger, options ...func(*BaseFs) err
 
        publishFs := hugofs.NewBaseFileDecorator(afero.NewBasePathFs(fs.Destination, p.AbsPublishDir))
        sourceFs := hugofs.NewBaseFileDecorator(afero.NewBasePathFs(fs.Source, p.WorkingDir))
+       publishFsStatic := afero.NewBasePathFs(fs.Source, p.AbsPublishDir)
 
        // Same as sourceFs, but no decoration. This is what's used by os.ReadDir etc.
        workDir := afero.NewBasePathFs(afero.NewReadOnlyFs(fs.Source), p.WorkingDir)
 
        b := &BaseFs{
-               SourceFs:  sourceFs,
-               WorkDir:   workDir,
-               PublishFs: publishFs,
-               buildMu:   lockedfile.MutexAt(filepath.Join(p.WorkingDir, lockFileBuild)),
+               SourceFs:        sourceFs,
+               WorkDir:         workDir,
+               PublishFs:       publishFs,
+               PublishFsStatic: publishFsStatic,
+               buildMu:         lockedfile.MutexAt(filepath.Join(p.WorkingDir, lockFileBuild)),
        }
 
        for _, opt := range options {
index d33f70d8e670c6b5976a00f135403bd4d554845a..47687eaad8ac7077bec2e2767bca0e469eb4e054 100644 (file)
@@ -33,9 +33,10 @@ func newPagesProcessor(h *HugoSites, sp *source.SourceSpec) *pagesProcessor {
        procs := make(map[string]pagesCollectorProcessorProvider)
        for _, s := range h.Sites {
                procs[s.Lang()] = &sitePagesProcessor{
-                       m:           s.pageMap,
-                       errorSender: s.h,
-                       itemChan:    make(chan any, config.GetNumWorkerMultiplier()*2),
+                       m:                  s.pageMap,
+                       errorSender:        s.h,
+                       itemChan:           make(chan interface{}, config.GetNumWorkerMultiplier()*2),
+                       renderStaticToDisk: h.Cfg.GetBool("renderStaticToDisk"),
                }
        }
        return &pagesProcessor{
@@ -118,6 +119,8 @@ type sitePagesProcessor struct {
        ctx       context.Context
        itemChan  chan any
        itemGroup *errgroup.Group
+
+       renderStaticToDisk bool
 }
 
 func (p *sitePagesProcessor) Process(item any) error {
@@ -162,7 +165,12 @@ func (p *sitePagesProcessor) copyFile(fim hugofs.FileMetaInfo) error {
 
        defer f.Close()
 
-       return s.publish(&s.PathSpec.ProcessingStats.Files, target, f)
+       fs := s.PublishFs
+       if p.renderStaticToDisk {
+               fs = s.PublishFsStatic
+       }
+
+       return s.publish(&s.PathSpec.ProcessingStats.Files, target, f, fs)
 }
 
 func (p *sitePagesProcessor) doProcess(item any) error {
index efa9368306a0fea6de64e3e027a3912b9534965d..d7b5cb64e553a46e26f7595437a830457e863adc 100644 (file)
@@ -1829,10 +1829,10 @@ func (s *Site) lookupTemplate(layouts ...string) (tpl.Template, bool) {
        return nil, false
 }
 
-func (s *Site) publish(statCounter *uint64, path string, r io.Reader) (err error) {
+func (s *Site) publish(statCounter *uint64, path string, r io.Reader, fs afero.Fs) (err error) {
        s.PathSpec.ProcessingStats.Incr(statCounter)
 
-       return helpers.WriteToDisk(filepath.Clean(path), r, s.BaseFs.PublishFs)
+       return helpers.WriteToDisk(filepath.Clean(path), r, fs)
 }
 
 func (s *Site) kindFromFileInfoOrSections(fi *fileInfo, sections []string) string {