hugolib: Read default output formats from site config
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 22 Mar 2017 10:34:17 +0000 (11:34 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Mon, 27 Mar 2017 13:43:56 +0000 (15:43 +0200)
hugolib/hugo_sites_build.go
hugolib/page.go
hugolib/page_output.go
hugolib/site.go
hugolib/site_output.go
hugolib/site_output_test.go
media/mediaType_test.go
output/layout_base.go
output/layout_base_test.go
tpl/tplimpl/template.go

index 32f1d56be4c49b5db6cd02a2b86f269dce0b3433..970948bb32fe4bfbaf4ed7067a8349e7e2c63433 100644 (file)
@@ -177,7 +177,7 @@ func (h *HugoSites) assemble(config *BuildCfg) error {
                for _, p := range s.Pages {
                        // May have been set in front matter
                        if len(p.outputFormats) == 0 {
-                               p.outputFormats = s.defaultOutputDefinitions.ForKind(p.Kind)
+                               p.outputFormats = s.outputFormats[p.Kind]
                        }
                        if err := p.initTargetPathDescriptor(); err != nil {
                                return err
index 390df8070f67dc8488b8c25760376cb0241aaa2a..4e9ea2e7875847562e990e5df72ecb1221ffe93c 100644 (file)
@@ -859,7 +859,7 @@ func (p *Page) RelPermalink() string {
 func (p *Page) initURLs() error {
        // TODO(bep) output
        if len(p.outputFormats) == 0 {
-               p.outputFormats = p.s.defaultOutputDefinitions.ForKind(p.Kind)
+               p.outputFormats = p.s.outputFormats[p.Kind]
        }
        rel := p.createRelativePermalink()
        p.permalink = p.s.permalink(rel)
index 2d441fd4493fea78599590123b1b069f77e51ce5..0038b7fdd88e467ed16f96a774518fad1cd31ee5 100644 (file)
@@ -45,7 +45,6 @@ func (p *PageOutput) targetPath(addends ...string) (string, error) {
                return "", err
        }
        return tp, nil
-
 }
 
 func newPageOutput(p *Page, createCopy bool, f output.Format) (*PageOutput, error) {
index 0959df3d5e0cb44adec39769dc05b0a74cfe8e21..1cdb285baedef9daffbcb7e19de56214380ce557 100644 (file)
@@ -108,7 +108,10 @@ type Site struct {
 
        disabledKinds map[string]bool
 
-       defaultOutputDefinitions siteOutputDefinitions
+       // Output formats defined in site config per Page Kind, or some defaults
+       // if not set.
+       // Output formats defined in Page front matter will override these.
+       outputFormats map[string]output.Formats
 
        // Logger etc.
        *deps.Deps `json:"-"`
@@ -124,12 +127,12 @@ func (s *Site) isEnabled(kind string) bool {
 // reset returns a new Site prepared for rebuild.
 func (s *Site) reset() *Site {
        return &Site{Deps: s.Deps,
-               layoutHandler:            output.NewLayoutHandler(s.PathSpec.ThemeSet()),
-               disabledKinds:            s.disabledKinds,
-               defaultOutputDefinitions: s.defaultOutputDefinitions,
-               Language:                 s.Language,
-               owner:                    s.owner,
-               PageCollections:          newPageCollections()}
+               layoutHandler:   output.NewLayoutHandler(s.PathSpec.ThemeSet()),
+               disabledKinds:   s.disabledKinds,
+               outputFormats:   s.outputFormats,
+               Language:        s.Language,
+               owner:           s.owner,
+               PageCollections: newPageCollections()}
 }
 
 // newSite creates a new site with the given configuration.
@@ -145,14 +148,18 @@ func newSite(cfg deps.DepsCfg) (*Site, error) {
                disabledKinds[disabled] = true
        }
 
-       outputDefs := createSiteOutputDefinitions(cfg.Cfg)
+       outputFormats, err := createSiteOutputFormats(cfg.Language)
+
+       if err != nil {
+               return nil, err
+       }
 
        s := &Site{
-               PageCollections:          c,
-               layoutHandler:            output.NewLayoutHandler(cfg.Cfg.GetString("themesDir") != ""),
-               Language:                 cfg.Language,
-               disabledKinds:            disabledKinds,
-               defaultOutputDefinitions: outputDefs,
+               PageCollections: c,
+               layoutHandler:   output.NewLayoutHandler(cfg.Cfg.GetString("themesDir") != ""),
+               Language:        cfg.Language,
+               disabledKinds:   disabledKinds,
+               outputFormats:   outputFormats,
        }
 
        s.Info = newSiteInfo(siteBuilderCfg{s: s, pageCollections: c, language: s.Language})
@@ -2007,7 +2014,7 @@ func (s *Site) newNodePage(typ string, sections ...string) *Page {
                sections: sections,
                s:        s}
 
-       p.outputFormats = p.s.defaultOutputDefinitions.ForKind(typ)
+       p.outputFormats = p.s.outputFormats[p.Kind]
 
        return p
 
index 6b27bb1a17fd422424227b734e477a7fe90369f3..8b0dd7d6320e2a04dd91256f141085cba1dbbf08 100644 (file)
 package hugolib
 
 import (
+       "fmt"
        "path"
        "strings"
 
+       "github.com/spf13/cast"
        "github.com/spf13/hugo/config"
        "github.com/spf13/hugo/output"
 )
 
-type siteOutputDefinitions []siteOutputDefinition
+func createSiteOutputFormats(cfg config.Provider) (map[string]output.Formats, error) {
+       if !cfg.IsSet("outputs") {
+               return createDefaultOutputFormats(cfg)
+       }
 
-type siteOutputDefinition struct {
-       // What Kinds of pages are excluded in this definition.
-       // A blank strings means NONE.
-       // Comma separated list (for now).
-       ExcludedKinds string
+       outFormats := make(map[string]output.Formats)
 
-       Outputs []output.Format
-}
+       outputs := cfg.GetStringMap("outputs")
 
-func (defs siteOutputDefinitions) ForKind(kind string) []output.Format {
-       var result []output.Format
+       if outputs == nil || len(outputs) == 0 {
+               // TODO(bep) outputs log a warning?
+               return outFormats, nil
+       }
 
-       for _, def := range defs {
-               if def.ExcludedKinds == "" || !strings.Contains(def.ExcludedKinds, kind) {
-                       result = append(result, def.Outputs...)
+       for k, v := range outputs {
+               var formats output.Formats
+               vals := cast.ToStringSlice(v)
+               for _, format := range vals {
+                       f, found := output.GetFormat(format)
+                       if !found {
+                               return nil, fmt.Errorf("Failed to resolve output format %q from site config", format)
+                       }
+                       formats = append(formats, f)
                }
-       }
 
-       return result
-}
+               if len(formats) > 0 {
+                       outFormats[k] = formats
+               }
+       }
 
-func createSiteOutputDefinitions(cfg config.Provider) siteOutputDefinitions {
+       // Make sure every kind has at least one output format
+       for _, kind := range allKinds {
+               if _, found := outFormats[kind]; !found {
+                       outFormats[kind] = output.Formats{output.HTMLType}
+               }
+       }
 
-       var defs siteOutputDefinitions
+       return outFormats, nil
 
-       // All have HTML
-       defs = append(defs, siteOutputDefinition{ExcludedKinds: "", Outputs: []output.Format{output.HTMLType}})
+}
 
-       // TODO(bep) output deprecate rssURI
-       rssBase := cfg.GetString("rssURI")
-       if rssBase == "" {
-               rssBase = "index"
-       }
+func createDefaultOutputFormats(cfg config.Provider) (map[string]output.Formats, error) {
+       outFormats := make(map[string]output.Formats)
+       for _, kind := range allKinds {
+               var formats output.Formats
+               // All have HTML
+               formats = append(formats, output.HTMLType)
+
+               // All but page have RSS
+               if kind != KindPage {
+                       // TODO(bep) output deprecate rssURI
+                       rssBase := cfg.GetString("rssURI")
+                       if rssBase == "" {
+                               rssBase = "index"
+                       }
+
+                       // RSS has now a well defined media type, so strip any suffix provided
+                       rssBase = strings.TrimSuffix(rssBase, path.Ext(rssBase))
+                       rssType := output.RSSType
+                       rssType.BaseName = rssBase
+                       formats = append(formats, rssType)
 
-       // RSS has now a well defined media type, so strip any suffix provided
-       rssBase = strings.TrimSuffix(rssBase, path.Ext(rssBase))
-       rssType := output.RSSType
-       rssType.BaseName = rssBase
+               }
 
-       // Some have RSS
-       defs = append(defs, siteOutputDefinition{ExcludedKinds: "page", Outputs: []output.Format{rssType}})
+               outFormats[kind] = formats
+       }
 
-       return defs
+       return outFormats, nil
 }
index 85b3291feba76dc891e19c330003737c80b246a0..6694a9bfd581d192c0ddebb2bc0798984a39edc1 100644 (file)
@@ -27,23 +27,25 @@ import (
        "github.com/spf13/viper"
 )
 
-func TestDefaultOutputDefinitions(t *testing.T) {
+func TestDefaultOutputFormats(t *testing.T) {
        t.Parallel()
-       defs := createSiteOutputDefinitions(viper.New())
+       defs, err := createDefaultOutputFormats(viper.New())
+
+       require.NoError(t, err)
 
        tests := []struct {
                name string
                kind string
-               want []output.Format
+               want output.Formats
        }{
-               {"RSS not for regular pages", KindPage, []output.Format{output.HTMLType}},
-               {"Home Sweet Home", KindHome, []output.Format{output.HTMLType, output.RSSType}},
+               {"RSS not for regular pages", KindPage, output.Formats{output.HTMLType}},
+               {"Home Sweet Home", KindHome, output.Formats{output.HTMLType, output.RSSType}},
        }
 
        for _, tt := range tests {
                t.Run(tt.name, func(t *testing.T) {
-                       if got := defs.ForKind(tt.kind); !reflect.DeepEqual(got, tt.want) {
-                               t.Errorf("siteOutputDefinitions.ForKind(%v) = %v, want %v", tt.kind, got, tt.want)
+                       if got := defs[tt.kind]; !reflect.DeepEqual(got, tt.want) {
+                               t.Errorf("createDefaultOutputFormats(%v) = %v, want %v", tt.kind, got, tt.want)
                        }
                })
        }
@@ -57,6 +59,7 @@ func TestSiteWithPageOutputs(t *testing.T) {
        }
 }
 
+// TODO(bep) output add test for site outputs config
 func doTestSiteWithPageOutputs(t *testing.T, outputs []string) {
        t.Parallel()
 
index 41cc48369ed571e0a8422d09367a9eb97cf50156..e9de66d0eb9b853791ae50369c91a9f759950636 100644 (file)
@@ -24,14 +24,14 @@ func TestDefaultTypes(t *testing.T) {
        require.Equal(t, "html", HTMLType.SubType)
        require.Equal(t, "html", HTMLType.Suffix)
 
-       require.Equal(t, "text/html", HTMLType.MainType())
+       require.Equal(t, "text/html", HTMLType.Type())
        require.Equal(t, "text/html+html", HTMLType.String())
 
        require.Equal(t, "application", RSSType.MainType)
        require.Equal(t, "rss", RSSType.SubType)
        require.Equal(t, "xml", RSSType.Suffix)
 
-       require.Equal(t, "application/rss", RSSType.MainType())
+       require.Equal(t, "application/rss", RSSType.Type())
        require.Equal(t, "application/rss+xml", RSSType.String())
 
 }
index 929ee07a21c2406134ef54c7d705f50d98623f4c..6b26a4c5bee2645ff9a845f138e81a4e2a02c32e 100644 (file)
@@ -56,11 +56,11 @@ type TemplateLookupDescriptor struct {
        ContainsAny func(filename string, subslices [][]byte) (bool, error)
 }
 
-func CreateTemplateID(d TemplateLookupDescriptor) (TemplateNames, error) {
+func CreateTemplateNames(d TemplateLookupDescriptor) (TemplateNames, error) {
 
        var id TemplateNames
 
-       name := filepath.FromSlash(d.RelPath)
+       name := filepath.ToSlash(d.RelPath)
 
        if d.Prefix != "" {
                name = strings.Trim(d.Prefix, "/") + "/" + name
index 60d9b8c622f29fa17b0a1bd0dbcc285950056858..f20d99befd9b34a00877f0fbcf7846505f3cf4bd 100644 (file)
@@ -124,6 +124,8 @@ func TestLayoutBase(t *testing.T) {
        } {
                t.Run(this.name, func(t *testing.T) {
 
+                       this.basePathMatchStrings = filepath.FromSlash(this.basePathMatchStrings)
+
                        fileExists := func(filename string) (bool, error) {
                                stringsToMatch := strings.Split(this.basePathMatchStrings, "|")
                                for _, s := range stringsToMatch {
@@ -148,7 +150,7 @@ func TestLayoutBase(t *testing.T) {
                        this.expect.MasterFilename = filepath.FromSlash(this.expect.MasterFilename)
                        this.expect.OverlayFilename = filepath.FromSlash(this.expect.OverlayFilename)
 
-                       id, err := CreateTemplateID(this.d)
+                       id, err := CreateTemplateNames(this.d)
 
                        require.NoError(t, err)
                        require.Equal(t, this.expect, id, this.name)
index bf4587e8ef86336660c567e5e96aaa1fe82832bb..04f932dddd4abcf89044c8131b5970ac9e55aea1 100644 (file)
@@ -508,7 +508,7 @@ func (t *GoHTMLTemplate) loadTemplates(absPath string, prefix string) {
                                },
                        }
 
-                       tplID, err := output.CreateTemplateID(descriptor)
+                       tplID, err := output.CreateTemplateNames(descriptor)
                        if err != nil {
                                t.Log.ERROR.Printf("Failed to resolve template in path %q: %s", path, err)
                                return nil