2024-02-11 13:16:10 -03:00
|
|
|
package templates
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2024-02-14 23:54:46 -03:00
|
|
|
"bytes"
|
2024-02-11 13:16:10 -03:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/niklasfasching/go-org/org"
|
2024-02-14 23:54:46 -03:00
|
|
|
"github.com/osteele/liquid"
|
|
|
|
"github.com/yuin/goldmark"
|
2024-02-11 13:16:10 -03:00
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
const FM_SEPARATOR = "---"
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
type Engine = liquid.Engine
|
|
|
|
|
2024-02-11 13:16:10 -03:00
|
|
|
type Template struct {
|
2024-02-14 23:54:46 -03:00
|
|
|
SrcPath string
|
|
|
|
Metadata map[string]interface{}
|
|
|
|
liquidTemplate liquid.Template
|
|
|
|
}
|
|
|
|
|
2024-02-16 12:39:19 -03:00
|
|
|
// Create a new template engine, with custom liquid filters.
|
|
|
|
// The `siteUrl` is necessary to provide context for the absolute_url filter.
|
2024-02-16 18:39:21 -03:00
|
|
|
func NewEngine(siteUrl string, includesDir string) *Engine {
|
2024-02-15 13:53:45 -03:00
|
|
|
e := liquid.NewEngine()
|
2024-02-16 18:39:21 -03:00
|
|
|
loadJekyllFilters(e, siteUrl, includesDir)
|
2024-02-15 13:53:45 -03:00
|
|
|
return e
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
func Parse(engine *Engine, path string) (*Template, error) {
|
2024-02-11 13:16:10 -03:00
|
|
|
file, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
scanner := bufio.NewScanner(file)
|
|
|
|
|
|
|
|
scanner.Scan()
|
|
|
|
line := scanner.Text()
|
|
|
|
|
|
|
|
// if the file doesn't start with a front matter delimiter, it's not a template
|
|
|
|
if strings.TrimSpace(line) != FM_SEPARATOR {
|
2024-02-12 15:33:30 -03:00
|
|
|
return nil, nil
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
// extract the yaml front matter and save the rest of the template content separately
|
2024-02-11 13:16:10 -03:00
|
|
|
var yamlContent []byte
|
2024-02-14 23:54:46 -03:00
|
|
|
var liquidContent []byte
|
|
|
|
yamlClosed := false
|
2024-02-11 13:16:10 -03:00
|
|
|
for scanner.Scan() {
|
2024-02-14 23:54:46 -03:00
|
|
|
line := append(scanner.Bytes(), '\n')
|
|
|
|
if yamlClosed {
|
|
|
|
liquidContent = append(liquidContent, line...)
|
|
|
|
} else {
|
|
|
|
if strings.TrimSpace(scanner.Text()) == FM_SEPARATOR {
|
|
|
|
yamlClosed = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
yamlContent = append(yamlContent, line...)
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
}
|
2024-02-14 23:54:46 -03:00
|
|
|
liquidContent = bytes.TrimSuffix(liquidContent, []byte("\n"))
|
|
|
|
|
|
|
|
if !yamlClosed {
|
2024-02-11 13:16:10 -03:00
|
|
|
return nil, errors.New("front matter not closed")
|
|
|
|
}
|
|
|
|
|
2024-02-13 21:32:04 -03:00
|
|
|
metadata := make(map[string]interface{})
|
2024-02-11 13:16:10 -03:00
|
|
|
if len(yamlContent) != 0 {
|
|
|
|
err := yaml.Unmarshal([]byte(yamlContent), &metadata)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("invalid yaml: %s", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
liquid, err := engine.ParseTemplateAndCache(liquidContent, path, 0)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
templ := Template{SrcPath: path, Metadata: metadata, liquidTemplate: *liquid}
|
2024-02-12 15:16:56 -03:00
|
|
|
return &templ, nil
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
// Return the extension for the output format of this template
|
2024-02-11 13:16:10 -03:00
|
|
|
func (templ Template) Ext() string {
|
2024-02-12 15:16:56 -03:00
|
|
|
ext := filepath.Ext(templ.SrcPath)
|
2024-02-14 23:54:46 -03:00
|
|
|
if ext == ".org" || ext == ".md" {
|
|
|
|
return ".html"
|
2024-02-11 14:03:28 -03:00
|
|
|
}
|
|
|
|
return ext
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
func (templ Template) Render(context map[string]interface{}) ([]byte, error) {
|
2024-02-15 18:11:40 -03:00
|
|
|
// liquid rendering
|
2024-02-14 23:54:46 -03:00
|
|
|
content, err := templ.liquidTemplate.Render(context)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
ext := filepath.Ext(templ.SrcPath)
|
2024-02-15 18:11:40 -03:00
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
if ext == ".org" {
|
2024-02-15 18:11:40 -03:00
|
|
|
// org-mode rendering
|
2024-02-14 23:54:46 -03:00
|
|
|
doc := org.New().Parse(bytes.NewReader(content), templ.SrcPath)
|
2024-02-15 18:38:09 -03:00
|
|
|
htmlWriter := org.NewHTMLWriter()
|
|
|
|
|
|
|
|
// make * -> h1, ** -> h2, etc
|
|
|
|
htmlWriter.TopLevelHLevel = 1
|
|
|
|
|
|
|
|
contentStr, err := doc.Write(htmlWriter)
|
2024-02-14 23:54:46 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2024-02-13 13:43:20 -03:00
|
|
|
}
|
2024-02-14 23:54:46 -03:00
|
|
|
content = []byte(contentStr)
|
|
|
|
} else if ext == ".md" {
|
2024-02-15 18:11:40 -03:00
|
|
|
// markdown rendering
|
2024-02-14 23:54:46 -03:00
|
|
|
var buf bytes.Buffer
|
|
|
|
if err := goldmark.Convert(content, &buf); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
content = buf.Bytes()
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
return content, nil
|
2024-02-11 13:16:10 -03:00
|
|
|
}
|