1
0
mirror of https://github.com/danog/gojekyll.git synced 2024-11-26 19:24:45 +01:00
gojekyll/config/config.go

235 lines
5.6 KiB
Go

package config
import (
"os"
"path/filepath"
"strings"
"github.com/danog/gojekyll/templates"
"github.com/danog/gojekyll/utils"
yaml "gopkg.in/yaml.v2"
)
// Config is the Jekyll site configuration, typically read from _config.yml.
// See https://jekyllrb.com/docs/configuration/#default-configuration
type Config struct {
// Where things are:
Source string
Destination string
LayoutsDir string `yaml:"layouts_dir"`
DataDir string `yaml:"data_dir"`
IncludesDir string `yaml:"includes_dir"`
Collections map[string]map[string]interface{} `yaml:"-"`
Theme string
// Handling Reading
Include []string
Exclude []string
KeepFiles []string `yaml:"keep_files"`
MarkdownExt string `yaml:"markdown_ext"`
// Filtering Content
Drafts bool `yaml:"show_drafts"`
Future bool
Unpublished bool
// Plugins
Plugins []string
// Conversion
ExcerptSeparator string `yaml:"excerpt_separator"`
Incremental bool
Sass struct {
Dir string `yaml:"sass_dir"`
// TODO Style string // compressed
}
// Serving
Host string
Port int
AbsoluteURL string `yaml:"url"`
BaseURL string
// Outputting
Permalink string
Timezone string
Verbose bool
Defaults []struct {
Scope struct {
Path string
Type string
}
Values map[string]interface{}
}
// CLI-only
DryRun bool `yaml:"-"`
ForcePolling bool `yaml:"-"`
Watch bool `yaml:"-"`
// Meta
ConfigFile string `yaml:"-"`
m map[string]interface{} `yaml:"-"` // config file, as map
ms yaml.MapSlice `yaml:"-"` // config file, as MapSlice
// Plugins
RequireFrontMatter bool `yaml:"-"`
RequireFrontMatterExclude map[string]bool `yaml:"-"`
}
// FromDirectory updates the config from the config file in
// the directory, if such a file exists.
func (c *Config) FromDirectory(dir string) error {
path := filepath.Join(dir, "_config.yml")
bytes, err := os.ReadFile(path)
switch {
case os.IsNotExist(err):
// break
case err != nil:
return err
default:
if err = Unmarshal(bytes, c); err != nil {
return utils.WrapPathError(err, path)
}
c.ConfigFile = path
}
c.Source = dir
return nil
}
type configCompat struct {
Gems []string
}
type collectionsList struct {
Collections []string
}
type collectionsMap struct {
Collections map[string]map[string]interface{}
}
// IsConfigPath returns true if its arguments is a site configuration file.
func (c *Config) IsConfigPath(rel string) bool {
return rel == "_config.yml"
}
// SassDir returns the relative path of the SASS directory.
func (c *Config) SassDir() string {
return "_sass"
}
// SourceDir returns the source directory as an absolute path.
func (c *Config) SourceDir() string {
return utils.MustAbs(c.Source)
}
// GetFrontMatterDefaults implements https://jekyllrb.com/docs/configuration/#front-matter-defaults
func (c *Config) GetFrontMatterDefaults(typename, rel string) (m map[string]interface{}) {
for _, entry := range c.Defaults {
scope := &entry.Scope
hasPrefix := strings.HasPrefix(rel, scope.Path)
hasType := scope.Type == "" || scope.Type == typename
if hasPrefix && hasType {
m = templates.MergeVariableMaps(m, entry.Values)
}
}
return
}
// RequiresFrontMatter returns a bool indicating whether the file requires front matter in order to recognize as a page.
func (c *Config) RequiresFrontMatter(rel string) bool {
switch {
case c.RequireFrontMatter:
return true
case !c.IsMarkdown(rel):
return true
case utils.StringArrayContains(c.Include, rel):
return false
case c.RequireFrontMatterExclude[strings.ToUpper(utils.TrimExt(filepath.Base(rel)))]:
return true
default:
return false
}
}
// Unmarshal updates site from a YAML configuration file.
func Unmarshal(bytes []byte, c *Config) error {
var (
compat configCompat
cList collectionsList
)
if err := yaml.Unmarshal(bytes, &c); err != nil {
return err
}
if err := yaml.Unmarshal(bytes, &c.ms); err != nil {
return err
}
if err := yaml.Unmarshal(bytes, &c.m); err != nil {
return err
}
if err := yaml.Unmarshal(bytes, &cList); err == nil {
if len(c.Collections) == 0 {
c.Collections = make(map[string]map[string]interface{})
}
for _, name := range cList.Collections {
c.Collections[name] = map[string]interface{}{}
}
}
cMap := collectionsMap{c.Collections}
if err := yaml.Unmarshal(bytes, &cMap); err == nil {
c.Collections = cMap.Collections
}
if err := yaml.Unmarshal(bytes, &compat); err != nil {
return err
}
if len(c.Plugins) == 0 {
c.Plugins = compat.Gems
}
return nil
}
// Variables returns the configuration as a Liquid variable map.
func (c *Config) Variables() map[string]interface{} {
m := map[string]interface{}{}
for _, item := range c.ms {
if s, ok := item.Key.(string); ok {
m[s] = item.Value
}
}
return m
}
// Set sets a value in the Liquid variable map.
// This does not update the corresponding value in the Config struct.
func (c *Config) Set(key string, val interface{}) {
c.m[key] = val
for _, item := range c.ms {
if item.Key == key {
item.Value = val
return
}
}
c.ms = append(c.ms, yaml.MapItem{Key: key, Value: val})
}
// Map returns the config indexed by key, if it's a map.
func (c *Config) Map(key string) (map[string]interface{}, bool) {
if m, ok := c.m[key]; ok {
if m, ok := m.(map[string]interface{}); ok {
return m, ok
}
}
return nil, false
}
// String returns the config indexed by key, if it's a string.
func (c *Config) String(key string) (string, bool) {
if m, ok := c.m[key]; ok {
if m, ok := m.(string); ok {
return m, ok
}
}
return "", false
}