2015-09-14 13:42:48 +00:00
|
|
|
package frontmatter
|
|
|
|
|
|
|
|
import (
|
2015-09-17 10:32:27 +00:00
|
|
|
"log"
|
2015-09-17 14:38:04 +00:00
|
|
|
"reflect"
|
2015-09-14 13:42:48 +00:00
|
|
|
"sort"
|
|
|
|
|
|
|
|
"github.com/hacdias/caddy-hugo/utils"
|
|
|
|
"github.com/spf13/hugo/parser"
|
|
|
|
)
|
|
|
|
|
2015-09-17 14:38:04 +00:00
|
|
|
const mainName = "#MAIN#"
|
|
|
|
|
2015-09-14 13:42:48 +00:00
|
|
|
// Pretty creates a new FrontMatter object
|
2015-09-14 21:03:09 +00:00
|
|
|
func Pretty(content []byte) (interface{}, error) {
|
|
|
|
frontType := parser.DetectFrontMatter(rune(content[0]))
|
|
|
|
front, err := frontType.Parse(content)
|
2015-09-14 13:42:48 +00:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return []string{}, err
|
|
|
|
}
|
|
|
|
|
2015-09-17 10:32:27 +00:00
|
|
|
object := new(frontmatter)
|
|
|
|
object.Type = "object"
|
2015-09-17 14:38:04 +00:00
|
|
|
object.Name = mainName
|
2015-09-17 10:32:27 +00:00
|
|
|
|
|
|
|
return rawToPretty(front, object), nil
|
2015-09-14 13:42:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type frontmatter struct {
|
2015-09-15 10:59:48 +00:00
|
|
|
Name string
|
2015-09-17 10:32:27 +00:00
|
|
|
Title string
|
2015-09-15 10:59:48 +00:00
|
|
|
Content interface{}
|
|
|
|
Type string
|
2015-09-17 10:32:27 +00:00
|
|
|
Parent *frontmatter
|
2015-09-14 13:42:48 +00:00
|
|
|
}
|
|
|
|
|
2015-09-17 10:32:27 +00:00
|
|
|
func rawToPretty(config interface{}, parent *frontmatter) interface{} {
|
|
|
|
objects := []*frontmatter{}
|
|
|
|
arrays := []*frontmatter{}
|
|
|
|
fields := []*frontmatter{}
|
2015-09-15 20:28:54 +00:00
|
|
|
|
2015-09-17 10:32:27 +00:00
|
|
|
if parent.Type == "array" {
|
2015-09-15 10:59:48 +00:00
|
|
|
for index, element := range config.([]interface{}) {
|
|
|
|
if utils.IsMap(element) {
|
2015-09-17 14:38:04 +00:00
|
|
|
objects = append(objects, handleObjects(element, parent, string(index)))
|
2015-09-15 10:59:48 +00:00
|
|
|
} else if utils.IsSlice(element) {
|
2015-09-17 14:38:04 +00:00
|
|
|
arrays = append(arrays, handleArrays(element, parent, string(index)))
|
2015-09-15 10:59:48 +00:00
|
|
|
} else {
|
2015-09-17 14:38:04 +00:00
|
|
|
fields = append(fields, handleFlatValues(element, parent, string(index)))
|
2015-09-15 10:59:48 +00:00
|
|
|
}
|
|
|
|
}
|
2015-09-17 10:32:27 +00:00
|
|
|
} else if parent.Type == "object" {
|
|
|
|
for name, element := range config.(map[string]interface{}) {
|
|
|
|
if utils.IsMap(element) {
|
2015-09-17 14:38:04 +00:00
|
|
|
objects = append(objects, handleObjects(element, parent, name))
|
2015-09-17 10:32:27 +00:00
|
|
|
} else if utils.IsSlice(element) {
|
2015-09-17 14:38:04 +00:00
|
|
|
arrays = append(arrays, handleArrays(element, parent, name))
|
2015-09-17 10:32:27 +00:00
|
|
|
} else {
|
2015-09-17 14:38:04 +00:00
|
|
|
fields = append(fields, handleFlatValues(element, parent, name))
|
2015-09-17 10:32:27 +00:00
|
|
|
}
|
2015-09-14 13:42:48 +00:00
|
|
|
}
|
2015-09-17 10:32:27 +00:00
|
|
|
} else {
|
|
|
|
log.Panic("Parent type not allowed.")
|
2015-09-14 13:42:48 +00:00
|
|
|
}
|
|
|
|
|
2015-09-17 10:32:27 +00:00
|
|
|
sortByTitle(objects)
|
|
|
|
sortByTitle(arrays)
|
|
|
|
sortByTitle(fields)
|
|
|
|
|
|
|
|
settings := []*frontmatter{}
|
|
|
|
settings = append(settings, fields...)
|
|
|
|
settings = append(settings, arrays...)
|
|
|
|
settings = append(settings, objects...)
|
|
|
|
return settings
|
2015-09-14 13:42:48 +00:00
|
|
|
}
|
2015-09-17 14:38:04 +00:00
|
|
|
|
2015-09-17 14:45:59 +00:00
|
|
|
func sortByTitle(config []*frontmatter) {
|
|
|
|
keys := make([]string, len(config))
|
|
|
|
positionByTitle := make(map[string]int)
|
|
|
|
|
|
|
|
for index, element := range config {
|
|
|
|
keys[index] = element.Title
|
|
|
|
positionByTitle[element.Title] = index
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Strings(keys)
|
|
|
|
cnf := make([]*frontmatter, len(config))
|
|
|
|
|
|
|
|
for index, title := range keys {
|
|
|
|
cnf[index] = config[positionByTitle[title]]
|
|
|
|
}
|
|
|
|
|
|
|
|
for index := range config {
|
|
|
|
config[index] = cnf[index]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-17 14:38:04 +00:00
|
|
|
func handleObjects(content interface{}, parent *frontmatter, name string) *frontmatter {
|
|
|
|
c := new(frontmatter)
|
|
|
|
c.Parent = parent
|
|
|
|
c.Type = "object"
|
|
|
|
c.Title = name
|
|
|
|
|
|
|
|
if parent.Name == mainName {
|
|
|
|
c.Name = c.Title
|
|
|
|
} else {
|
|
|
|
c.Name = parent.Name + "[" + c.Title + "]"
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Content = rawToPretty(content, c)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleArrays(content interface{}, parent *frontmatter, name string) *frontmatter {
|
|
|
|
c := new(frontmatter)
|
|
|
|
c.Parent = parent
|
|
|
|
c.Type = "array"
|
|
|
|
c.Title = name
|
|
|
|
|
|
|
|
if parent.Type == "object" && parent.Name == mainName {
|
|
|
|
c.Name = name
|
|
|
|
} else {
|
|
|
|
c.Name = parent.Name + "[" + c.Name + "]"
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Content = rawToPretty(content, c)
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleFlatValues(content interface{}, parent *frontmatter, name string) *frontmatter {
|
|
|
|
c := new(frontmatter)
|
|
|
|
c.Parent = parent
|
|
|
|
|
|
|
|
switch reflect.ValueOf(content).Kind() {
|
|
|
|
case reflect.Bool:
|
|
|
|
c.Type = "boolean"
|
|
|
|
case reflect.Int:
|
|
|
|
case reflect.Float32:
|
|
|
|
case reflect.Float64:
|
|
|
|
c.Type = "number"
|
|
|
|
case reflect.String:
|
|
|
|
default:
|
|
|
|
c.Type = "string"
|
|
|
|
}
|
|
|
|
|
|
|
|
if parent.Type == "array" {
|
|
|
|
c.Name = parent.Name + "[]"
|
|
|
|
c.Title = content.(string)
|
|
|
|
} else if parent.Type == "object" {
|
|
|
|
c.Title = name
|
|
|
|
c.Name = parent.Name + "[" + name + "]"
|
|
|
|
|
|
|
|
if parent.Name == mainName {
|
|
|
|
c.Name = name
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Panic("Parent type not allowed in handleFlatValues.")
|
|
|
|
}
|
|
|
|
|
|
|
|
c.Content = content
|
|
|
|
return c
|
|
|
|
}
|