mirror of
https://github.com/danog/gojekyll.git
synced 2025-01-23 10:21:12 +01:00
204 lines
5.0 KiB
Go
204 lines
5.0 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 {
|
|
if p.Static {
|
|
source, err := ioutil.ReadFile(filepath.Join(siteConfig.SourceDir, p.Path))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = w.Write(source)
|
|
return err
|
|
}
|
|
|
|
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
|
|
})
|
|
}
|