hubolib: Refactor site rendering with an "output format context"
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Fri, 28 Apr 2017 07:40:50 +0000 (09:40 +0200)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Sat, 13 May 2017 19:44:15 +0000 (22:44 +0300)
Fixes #3397

hugolib/site.go
hugolib/site_render.go

index 4f7c2c5fb595249b9b940edd6fff7b14a4a6fe01..9bd9b49236d6e8d6f50034c876edc007f6088c84 100644 (file)
@@ -22,6 +22,7 @@ import (
        "net/url"
        "os"
        "path/filepath"
+       "sort"
        "strconv"
        "strings"
        "sync"
@@ -118,12 +119,42 @@ type Site struct {
        outputFormatsConfig output.Formats
        mediaTypesConfig    media.Types
 
+       // We render each site for all the relevant output formats in serial with
+       // this rendering context pointing to the current one.
+       rc *siteRenderingContext
+
+       // The output formats that we need to render this site in. This slice
+       // will be fixed once set.
+       // This will be the union of Site.Pages' outputFormats.
+       // This slice will be sorted.
+       renderFormats output.Formats
+
        // Logger etc.
        *deps.Deps `json:"-"`
 
        siteStats *siteStats
 }
 
+type siteRenderingContext struct {
+       output.Format
+}
+
+func (s *Site) initRenderFormats() {
+       formatSet := make(map[string]bool)
+       formats := output.Formats{}
+       for _, p := range s.Pages {
+               for _, f := range p.outputFormats {
+                       if !formatSet[f.Name] {
+                               formats = append(formats, f)
+                               formatSet[f.Name] = true
+                       }
+               }
+       }
+
+       sort.Sort(formats)
+       s.renderFormats = formats
+}
+
 type siteStats struct {
        pageCount        int
        pageCountRegular int
@@ -971,8 +1002,13 @@ func (s *Site) render() (err error) {
        }
        s.timerStep("render and write aliases")
 
-       if err = s.renderPages(); err != nil {
-               return
+       // TODO(bep) render consider this, ref. render404 etc.
+       s.initRenderFormats()
+       for _, rf := range s.renderFormats {
+               s.rc = &siteRenderingContext{Format: rf}
+               if err = s.renderPages(); err != nil {
+                       return
+               }
        }
        s.timerStep("render and write pages")
 
index 6aeaf677a6bcf2e59cddae67eae796af6141d75f..ac7b25da7da2b2882a257d7113fca2a732479a02 100644 (file)
@@ -81,6 +81,11 @@ func pageRenderer(s *Site, pages <-chan *Page, results chan<- error, wg *sync.Wa
                                pageOutput, err = page.mainPageOutput.copyWithFormat(outFormat)
                        }
 
+                       if outFormat != page.s.rc.Format {
+                               // Will be rendered  ... later.
+                               continue
+                       }
+
                        if err != nil {
                                s.Log.ERROR.Printf("Failed to create output page for type %q for page %q: %s", outFormat.Name, page, err)
                                continue