Skip to content

Commit

Permalink
Allow cascade _target to work with non toml fm
Browse files Browse the repository at this point in the history
The TOML lib unmarshals slices of string maps to []map[string]interface{}
whereas YAML and JSON decode to []interface{}

The existing tests only check for TOML working correctly, and _target
with cascade did not work at all for frontmatter defined in other formats.

Add a function to normalize those slices

Fixes gohugoio#7874
  • Loading branch information
gwatts authored and bep committed Oct 22, 2020
1 parent fdfa4a5 commit 3400aff
Show file tree
Hide file tree
Showing 4 changed files with 108 additions and 4 deletions.
18 changes: 18 additions & 0 deletions common/maps/maps.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@
package maps

import (
"fmt"
"strings"

"github.com/gobwas/glob"
Expand Down Expand Up @@ -64,6 +65,23 @@ func ToStringMap(in interface{}) map[string]interface{} {
return m
}

func ToSliceStringMap(in interface{}) ([]map[string]interface{}, error) {
switch v := in.(type) {
case []map[string]interface{}:
return v, nil
case []interface{}:
var s []map[string]interface{}
for _, entry := range v {
if vv, ok := entry.(map[string]interface{}); ok {
s = append(s, vv)
}
}
return s, nil
default:
return nil, fmt.Errorf("unable to cast %#v of type %T to []map[string]interface{}", in, in)
}
}

type keyRename struct {
pattern glob.Glob
newKey string
Expand Down
33 changes: 33 additions & 0 deletions common/maps/maps_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,39 @@ func TestToLower(t *testing.T) {
}
}

func TestToSliceStringMap(t *testing.T) {
c := qt.New(t)

tests := []struct {
input interface{}
expected []map[string]interface{}
}{
{
input: []map[string]interface{}{
{"abc": 123},
},
expected: []map[string]interface{}{
{"abc": 123},
},
}, {
input: []interface{}{
map[string]interface{}{
"def": 456,
},
},
expected: []map[string]interface{}{
{"def": 456},
},
},
}

for _, test := range tests {
v, err := ToSliceStringMap(test.input)
c.Assert(err, qt.IsNil)
c.Assert(v, qt.DeepEquals, test.expected)
}
}

func TestRenameKeys(t *testing.T) {
c := qt.New(t)

Expand Down
54 changes: 54 additions & 0 deletions hugolib/cascade_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -459,4 +459,58 @@ S1|p1:|p2:p2|

})

c.Run("slice with yaml _target", func(c *qt.C) {
b := newBuilder(c)

b.WithContent("_index.md", `---
title: "Home"
cascade:
- p1: p1
_target:
path: "**p1**"
- p2: p2
_target:
kind: "section"
---
`)

b.Build(BuildCfg{})

b.AssertFileContent("public/index.html", `
P1|p1:p1|p2:|
S1|p1:|p2:p2|
`)

})

c.Run("slice with json _target", func(c *qt.C) {
b := newBuilder(c)

b.WithContent("_index.md", `{
"title": "Home",
"cascade": [
{
"p1": "p1",
"_target": {
"path": "**p1**"
}
},{
"p2": "p2",
"_target": {
"kind": "section"
}
}
]
}
`)

b.Build(BuildCfg{})

b.AssertFileContent("public/index.html", `
P1|p1:p1|p2:|
S1|p1:|p2:p2|
`)

})

}
7 changes: 3 additions & 4 deletions hugolib/page__meta.go
Original file line number Diff line number Diff line change
Expand Up @@ -342,8 +342,7 @@ func (pm *pageMeta) setMetadata(parentBucket *pagesMapBucket, p *pageState, fron
if p.bucket != nil {
// Check for any cascade define on itself.
if cv, found := frontmatter["cascade"]; found {
switch v := cv.(type) {
case []map[string]interface{}:
if v, err := maps.ToSliceStringMap(cv); err == nil {
p.bucket.cascade = make(map[page.PageMatcher]maps.Params)

for _, vv := range v {
Expand All @@ -367,12 +366,12 @@ func (pm *pageMeta) setMetadata(parentBucket *pagesMapBucket, p *pageState, fron
}

}
default:
} else {
p.bucket.cascade = map[page.PageMatcher]maps.Params{
page.PageMatcher{}: maps.ToStringMap(cv),
}
}

}
}
}
} else {
Expand Down

0 comments on commit 3400aff

Please sign in to comment.