mirror of
https://github.com/danog/gojekyll.git
synced 2025-01-23 02:11:17 +01:00
244 lines
5.4 KiB
Go
244 lines
5.4 KiB
Go
package pages
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"path"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/osteele/gojekyll/frontmatter"
|
|
"github.com/osteele/gojekyll/templates"
|
|
"github.com/osteele/liquid/evaluator"
|
|
)
|
|
|
|
// Page is a document with frontmatter.
|
|
type Page interface {
|
|
Document
|
|
// Render asks a page to compute its content.
|
|
// This has the side effect of causing the content to subsequently appear in the drop.
|
|
Render() error
|
|
SetContent(string)
|
|
FrontMatter() map[string]interface{}
|
|
// PostDate returns the date computed from the filename or frontmatter.
|
|
// It is an uncaught error to call this on a page that is not a Post.
|
|
// TODO Should posts have their own interface?
|
|
PostDate() time.Time
|
|
|
|
Categories() []string
|
|
Tags() []string
|
|
}
|
|
|
|
// PageEmbed can be embedded to give defaults for the Page interface.
|
|
type PageEmbed struct {
|
|
Path string
|
|
}
|
|
|
|
// Permalink is in the pages.Page interface.
|
|
func (p *PageEmbed) Permalink() string { return p.Path }
|
|
|
|
// OutputExt is in the pages.Page interface.
|
|
func (p *PageEmbed) OutputExt() string { return path.Ext(p.Path) }
|
|
|
|
// SourcePath is in the pages.Page interface.
|
|
func (p *PageEmbed) SourcePath() string { return "" }
|
|
|
|
// Published is in the pages.Page interface.
|
|
func (p *PageEmbed) Published() bool { return true }
|
|
|
|
// Static is in the pages.Page interface.
|
|
func (p *PageEmbed) Static() bool { return false } // FIXME means different things to different callers
|
|
|
|
// Reload is in the pages.Page interface.
|
|
func (p *PageEmbed) Reload() error { return nil }
|
|
|
|
// A page is a concrete implementation of the Page interface.
|
|
type page struct {
|
|
file
|
|
firstLine int
|
|
raw []byte
|
|
|
|
sync.RWMutex
|
|
content string
|
|
contentError error
|
|
contentOnce sync.Once
|
|
excerpt interface{} // []byte or string, depending on rendering stage
|
|
rendered bool
|
|
}
|
|
|
|
// Static is in the File interface.
|
|
func (p *page) Static() bool { return false }
|
|
|
|
func makePage(filename string, f file) (*page, error) {
|
|
raw, lineNo, err := readFrontMatter(&f)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
p := page{
|
|
file: f,
|
|
firstLine: lineNo,
|
|
raw: raw,
|
|
}
|
|
if err := p.setPermalink(); err != nil {
|
|
return nil, err
|
|
}
|
|
return &p, nil
|
|
}
|
|
|
|
func (p *page) Reload() error {
|
|
if err := p.file.Reload(); err != nil {
|
|
return err
|
|
}
|
|
// FIXME use original defaults
|
|
raw, lineNo, err := readFrontMatter(&p.file)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
p.firstLine = lineNo
|
|
p.raw = raw
|
|
p.reset()
|
|
return nil
|
|
}
|
|
|
|
func (p *page) reset() {
|
|
p.contentOnce = sync.Once{}
|
|
p.rendered = false
|
|
}
|
|
|
|
func readFrontMatter(f *file) (b []byte, lineNo int, err error) {
|
|
b, err = ioutil.ReadFile(f.filename)
|
|
if err != nil {
|
|
return
|
|
}
|
|
lineNo = 1
|
|
frontMatter, err := frontmatter.Read(&b, &lineNo)
|
|
if err != nil {
|
|
return
|
|
}
|
|
f.frontMatter = templates.MergeVariableMaps(f.frontMatter, frontMatter)
|
|
return
|
|
}
|
|
|
|
func (p *page) FrontMatter() map[string]interface{} {
|
|
return p.frontMatter
|
|
}
|
|
|
|
// Categories is in the Page interface
|
|
func (p *page) Categories() []string {
|
|
return frontmatter.FrontMatter(p.frontMatter).SortedStringArray("categories")
|
|
}
|
|
|
|
// Tags is in the Page interface
|
|
func (p *page) Tags() []string {
|
|
return frontmatter.FrontMatter(p.frontMatter).SortedStringArray("tags")
|
|
}
|
|
|
|
// TemplateContext returns the local variables for template evaluation
|
|
func (p *page) TemplateContext() map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"page": p,
|
|
"site": p.site,
|
|
}
|
|
}
|
|
|
|
// PostDate is part of the Page interface.
|
|
// FIXME move this back to Page interface, or re-work this entirely.
|
|
func (f *file) PostDate() time.Time {
|
|
switch value := f.frontMatter["date"].(type) {
|
|
case time.Time:
|
|
return value
|
|
case string:
|
|
t, err := evaluator.ParseDate(value)
|
|
if err == nil {
|
|
return t
|
|
}
|
|
default:
|
|
panic(fmt.Sprintf("expected a date %v", value))
|
|
}
|
|
panic("read posts should have set this")
|
|
}
|
|
|
|
// Write applies Liquid and Markdown, as appropriate.
|
|
func (p *page) Write(w io.Writer) error {
|
|
err := p.Render()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
content := p.content
|
|
layout, ok := p.frontMatter["layout"].(string)
|
|
if ok && layout != "" {
|
|
rp := p.site.RenderingPipeline()
|
|
b, e := rp.ApplyLayout(layout, []byte(content), p.TemplateContext())
|
|
if e != nil {
|
|
return e
|
|
}
|
|
_, err = w.Write(b)
|
|
} else {
|
|
_, err = io.WriteString(w, content)
|
|
}
|
|
return err
|
|
}
|
|
|
|
// Content computes the page content.
|
|
func (p *page) Render() error {
|
|
p.contentOnce.Do(func() {
|
|
cn, ex, err := p.computeContent()
|
|
p.Lock()
|
|
defer p.Unlock()
|
|
p.content = cn
|
|
p.contentError = err
|
|
p.excerpt = ex
|
|
p.rendered = true
|
|
})
|
|
return p.contentError
|
|
}
|
|
|
|
func (p *page) Excerpt() interface{} {
|
|
p.RLock()
|
|
defer p.RUnlock()
|
|
if exc, ok := p.frontMatter["excerpt"]; ok {
|
|
return exc
|
|
}
|
|
if p.rendered {
|
|
return p.excerpt
|
|
}
|
|
return p.extractExcerpt()
|
|
}
|
|
|
|
func (p *page) computeContent() (cn string, ex interface{}, err error) {
|
|
pl := p.site.RenderingPipeline()
|
|
buf := new(bytes.Buffer)
|
|
err = pl.Render(buf, p.raw, p.TemplateContext(), p.filename, p.firstLine)
|
|
if err != nil {
|
|
return
|
|
}
|
|
cn = buf.String()
|
|
ex = cn
|
|
exb := p.extractExcerpt()
|
|
if !bytes.Equal(exb, p.raw) {
|
|
buf.Reset()
|
|
ex, err = pl.RenderTemplate(exb, p.TemplateContext(), p.filename, p.firstLine)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (p *page) extractExcerpt() []byte {
|
|
raw := p.raw
|
|
pos := bytes.Index(raw, []byte(p.site.Config().ExcerptSeparator))
|
|
if pos >= 0 {
|
|
return raw[:pos]
|
|
}
|
|
return raw
|
|
}
|
|
|
|
func (p *page) SetContent(content string) {
|
|
p.Lock()
|
|
defer p.Unlock()
|
|
p.content = content
|
|
p.contentError = nil
|
|
}
|