1
0
mirror of https://github.com/danog/gojekyll.git synced 2025-01-23 03:31:22 +01:00
gojekyll/page.go
2017-06-11 20:51:01 -04:00

195 lines
4.8 KiB
Go

package main
import (
"bytes"
"errors"
"fmt"
"io"
"io/ioutil"
"os"
"path/filepath"
"regexp"
"github.com/acstech/liquid"
"github.com/russross/blackfriday"
yaml "gopkg.in/yaml.v2"
)
const (
printFrontmatter = false
)
var (
frontmatterMatcher = regexp.MustCompile(`(?s)^---\n(.+?\n)---\n`)
templateVariableMatcher = regexp.MustCompile(`:(?:collection|file_ext|name|path|title)\b`)
nonAlphanumericSequenceMatcher = regexp.MustCompile(`[^[:alnum:]]+`)
)
var permalinkStyles = map[string]string{
"date": "/:categories/:year/:month/:day/:title.html",
"pretty": "/:categories/:year/:month/:day/:title/",
"ordinal": "/:categories/:year/:y_day/:title.html",
"none": "/:categories/:title.html",
}
// A Page represents an HTML page.
type Page struct {
Path string
Permalink string
Static bool
Published bool
FrontMatter map[interface{}]interface{}
Content []byte
}
func (p Page) String() string {
return fmt.Sprintf("Page{Path=%v, Permalink=%v, Static=%v}",
p.Path, p.Permalink, p.Static)
}
// CollectionItemData returns metadata for use in the representation of the page as a collection item
func (p Page) CollectionItemData() map[interface{}]interface{} {
// should have title, parts, url, description, due_date
data := map[interface{}]interface{}{
"url": p.Permalink,
}
// TODO additional variables from https://jekyllrb.com/docs/collections/#documents
if p.FrontMatter != nil {
data = mergeMaps(data, p.FrontMatter)
}
return data
}
func readPage(path string, defaults map[interface{}]interface{}) (*Page, error) {
var (
frontMatter map[interface{}]interface{}
static = true
body []byte
)
// TODO don't read, parse binary files
source, err := ioutil.ReadFile(filepath.Join(siteConfig.SourceDir, path))
if err != nil {
return nil, err
}
data := defaults
if match := frontmatterMatcher.FindSubmatchIndex(source); match != nil {
static = false
// TODO only prepend newlines if it's markdown
body = append(
regexp.MustCompile(`[^\n\r]+`).ReplaceAllLiteral(source[:match[1]], []byte{}),
source[match[1]:]...)
frontMatter = map[interface{}]interface{}{}
err = yaml.Unmarshal(source[match[2]:match[3]], &frontMatter)
if err != nil {
err := &os.PathError{Op: "read frontmatter", Path: path, Err: err}
return nil, err
}
data = mergeMaps(data, frontMatter)
} else {
body = []byte{}
}
permalink := path
if val, ok := data["permalink"]; ok {
pattern, ok := val.(string)
if !ok {
err := errors.New("permalink value must be a string")
err = &os.PathError{Op: "render", Path: path, Err: err}
return nil, err
}
permalink = expandPermalinkPattern(pattern, data, path)
}
return &Page{
Path: path,
Permalink: permalink,
Static: static,
Published: getBool(data, "published", true),
FrontMatter: frontMatter,
Content: body,
}, nil
}
// Render applies Liquid and Markdown, as appropriate.
func (p Page) Render(w io.Writer) error {
var (
path = p.Path
ext = filepath.Ext(path)
data = p.FrontMatter
)
if printFrontmatter {
b, _ := yaml.Marshal(stringMap(data))
println(string(b))
}
template, err := liquid.Parse(p.Content, nil)
if err != nil {
err := &os.PathError{Op: "Liquid Error", Path: path, Err: err}
return err
}
writer := new(bytes.Buffer)
template.Render(writer, stringMap(data))
body := writer.Bytes()
if ext == ".md" {
body = blackfriday.MarkdownCommon(body)
}
_, err = w.Write(body)
return err
}
func expandPermalinkPattern(pattern string, data map[interface{}]interface{}, path string) string {
if p, found := permalinkStyles[pattern]; found {
pattern = p
}
var (
collectionName string
ext = filepath.Ext(path)
localPath = path
outputExt = ext
name = filepath.Base(localPath)
title = getString(data, "title", name[:len(name)-len(ext)])
)
if ext == ".md" {
outputExt = ""
localPath = localPath[:len(localPath)-len(ext)]
}
if val, found := data["collection"]; found {
collectionName = val.(string)
collectionPath := "_" + collectionName + "/"
localPath = localPath[len(collectionPath):]
}
replaceNonalphumericsByHyphens := func(s string) string {
return nonAlphanumericSequenceMatcher.ReplaceAllString(s, "-")
}
templateVariables := map[string]string{
"collection": collectionName,
"name": replaceNonalphumericsByHyphens(name),
"output_ext": outputExt,
"path": localPath,
"title": replaceNonalphumericsByHyphens(title),
// TODO year month imonth day i_day short_year hour minute second slug categories
}
return templateVariableMatcher.ReplaceAllStringFunc(pattern, func(m string) string {
varname := m[1:]
value := templateVariables[varname]
if value == "" {
fmt.Printf("unknown variable %s in permalink template\n", varname)
}
return value
})
}