2024-02-26 12:16:06 -03:00
|
|
|
package markup
|
2024-02-11 13:16:10 -03:00
|
|
|
|
|
|
|
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"
|
|
|
|
|
2024-02-27 08:45:29 -03:00
|
|
|
"github.com/alecthomas/chroma/v2"
|
|
|
|
"github.com/alecthomas/chroma/v2/formatters/html"
|
|
|
|
"github.com/alecthomas/chroma/v2/lexers"
|
|
|
|
"github.com/alecthomas/chroma/v2/styles"
|
|
|
|
|
2024-02-11 13:16:10 -03:00
|
|
|
"github.com/niklasfasching/go-org/org"
|
2024-02-14 23:54:46 -03:00
|
|
|
"github.com/osteele/liquid"
|
|
|
|
"github.com/yuin/goldmark"
|
2024-02-27 08:45:29 -03:00
|
|
|
gm_highlight "github.com/yuin/goldmark-highlighting/v2"
|
2024-02-11 13:16:10 -03:00
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
const FM_SEPARATOR = "---"
|
2024-02-27 12:40:40 -03:00
|
|
|
const NO_SYNTAX_HIGHLIGHTING = ""
|
2024-02-11 13:16:10 -03:00
|
|
|
|
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-26 12:16:06 -03:00
|
|
|
// Try to parse a liquid template at the given location.
|
|
|
|
// Files starting with front matter (--- sorrrounded yaml)
|
|
|
|
// are considered templates. If the given file is not headed by front matter
|
|
|
|
// return (nil, nil).
|
|
|
|
// The front matter contents are stored in the returned template's Metadata.
|
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-27 12:24:45 -03:00
|
|
|
// Return the extension of this template's source file.
|
|
|
|
func (templ Template) SrcExt() string {
|
|
|
|
return filepath.Ext(templ.SrcPath)
|
|
|
|
}
|
|
|
|
|
2024-02-14 23:54:46 -03:00
|
|
|
// Return the extension for the output format of this template
|
2024-02-27 12:24:45 -03:00
|
|
|
func (templ Template) TargetExt() 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-27 12:24:45 -03:00
|
|
|
func (templ Template) IsDraft() bool {
|
|
|
|
if draft, ok := templ.Metadata["draft"]; ok {
|
|
|
|
return draft.(bool)
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (templ Template) IsPost() bool {
|
|
|
|
_, ok := templ.Metadata["date"]
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
2024-02-27 12:40:40 -03:00
|
|
|
// Renders the liquid template with default bindings.
|
|
|
|
func (templ Template) Render() ([]byte, error) {
|
|
|
|
ctx := map[string]interface{}{
|
|
|
|
"page": templ.Metadata,
|
|
|
|
}
|
|
|
|
return templ.RenderWith(ctx, NO_SYNTAX_HIGHLIGHTING)
|
|
|
|
}
|
|
|
|
|
2024-02-26 12:16:06 -03:00
|
|
|
// Renders the liquid template with the given context as bindings.
|
|
|
|
// If the template source is org or md, convert them to html after the
|
|
|
|
// liquid rendering.
|
2024-02-27 12:40:40 -03:00
|
|
|
func (templ Template) RenderWith(context map[string]interface{}, hlTheme string) ([]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-27 12:24:45 -03:00
|
|
|
if templ.SrcExt() == ".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
|
2024-02-27 12:40:40 -03:00
|
|
|
if hlTheme != NO_SYNTAX_HIGHLIGHTING {
|
|
|
|
htmlWriter.HighlightCodeBlock = highlightCodeBlock(hlTheme)
|
|
|
|
}
|
2024-02-15 18:38:09 -03:00
|
|
|
|
|
|
|
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)
|
2024-02-27 12:24:45 -03:00
|
|
|
} else if templ.SrcExt() == ".md" {
|
2024-02-15 18:11:40 -03:00
|
|
|
// markdown rendering
|
2024-02-14 23:54:46 -03:00
|
|
|
var buf bytes.Buffer
|
2024-02-27 12:40:40 -03:00
|
|
|
|
|
|
|
options := make([]goldmark.Option, 0)
|
|
|
|
if hlTheme != NO_SYNTAX_HIGHLIGHTING {
|
|
|
|
options = append(options, goldmark.WithExtensions(gm_highlight.NewHighlighting(
|
2024-02-27 08:45:29 -03:00
|
|
|
gm_highlight.WithStyle(hlTheme),
|
2024-02-27 12:40:40 -03:00
|
|
|
)))
|
|
|
|
}
|
|
|
|
md := goldmark.New(options...)
|
2024-02-27 08:45:29 -03:00
|
|
|
if err := md.Convert(content, &buf); err != nil {
|
2024-02-14 23:54:46 -03:00
|
|
|
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
|
|
|
}
|
2024-02-27 08:45:29 -03:00
|
|
|
|
|
|
|
func highlightCodeBlock(hlTheme string) func(source string, lang string, inline bool, params map[string]string) string {
|
|
|
|
// from https://github.com/niklasfasching/go-org/blob/a32df1461eb34a451b1e0dab71bd9b2558ea5dc4/blorg/util.go#L58
|
|
|
|
return func(source, lang string, inline bool, params map[string]string) string {
|
|
|
|
var w strings.Builder
|
|
|
|
l := lexers.Get(lang)
|
|
|
|
if l == nil {
|
|
|
|
l = lexers.Fallback
|
|
|
|
}
|
|
|
|
l = chroma.Coalesce(l)
|
|
|
|
it, _ := l.Tokenise(nil, source)
|
|
|
|
options := []html.Option{}
|
|
|
|
if params[":hl_lines"] != "" {
|
|
|
|
ranges := org.ParseRanges(params[":hl_lines"])
|
|
|
|
if ranges != nil {
|
|
|
|
options = append(options, html.HighlightLines(ranges))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ = html.New(options...).Format(&w, styles.Get(hlTheme), it)
|
|
|
|
if inline {
|
|
|
|
return `<div class="highlight-inline">` + "\n" + w.String() + "\n" + `</div>`
|
|
|
|
}
|
|
|
|
return `<div class="highlight">` + "\n" + w.String() + "\n" + `</div>`
|
|
|
|
}
|
|
|
|
}
|