jorge/templates/templates.go
facundoolano e97af58830 Add site struct and layout support
Squashed commit of the following:

commit 0ee8a385f111be807b4485b42316df6698d962f9
Author: facundoolano <facundo.olano@gmail.com>
Date:   Mon Feb 12 14:12:22 2024 -0300

    load layouts

commit 1c2594bb8aa6d6d9fbafcb530fdcdbdec2c146e7
Author: facundoolano <facundo.olano@gmail.com>
Date:   Mon Feb 12 13:17:28 2024 -0300

    add Site struct, explore some refactors

commit 3d8acb3957f5ba38a6e48d2614b9af65f1219298
Author: facundoolano <facundo.olano@gmail.com>
Date:   Mon Feb 12 11:33:19 2024 -0300

    prepare new phases structure for build command

commit fe7dcf9fb08c7b3e5679cf08c80beecc2eeb36e5
Author: facundoolano <facundo.olano@gmail.com>
Date:   Mon Feb 12 10:57:52 2024 -0300

    set template type

commit d9faa70c8d2d23c9b62904e7429a82437517b9c5
Author: facundoolano <facundo.olano@gmail.com>
Date:   Mon Feb 12 10:49:30 2024 -0300

    add Type to template

commit 27e0feede10f6c1340ce722085011a5eebcaee13
Author: facundoolano <facundo.olano@gmail.com>
Date:   Sun Feb 11 19:01:16 2024 -0300

    stub build and render extensions

commit e25518de3440cb3df2aa5674523128eff1d23404
Author: facundoolano <facundo.olano@gmail.com>
Date:   Sun Feb 11 17:37:30 2024 -0300

    pass pre-parsed layouts by arg instead

commit b3c2c9ebeb0d07d425f6db6a1bbbb5ee61c04548
Author: facundoolano <facundo.olano@gmail.com>
Date:   Sun Feb 11 15:13:17 2024 -0300

    first stab at recursively populating layouts

commit 4fe112694a3c44056f7aa5bd2be0794abcf4aa2a
Author: facundoolano <facundo.olano@gmail.com>
Date:   Sun Feb 11 14:33:07 2024 -0300

    initial support for page bindings
2024-02-12 15:16:56 -03:00

137 lines
3.3 KiB
Go

// TODO consider making this another package
package templates
import (
"bufio"
"errors"
"fmt"
"os"
"path/filepath"
"strings"
"github.com/niklasfasching/go-org/org"
"gopkg.in/osteele/liquid.v1"
"gopkg.in/yaml.v3"
)
const FM_SEPARATOR = "---"
type Type string
const (
// a file that doesn't have a front matter header, and thus is not renderable.
STATIC Type = "static"
// Templates in the root /layouts/ can be used to wrap around other template's content
// by setting the `layout` front matter field.
LAYOUT Type = "layout"
// A template that has a date, and thus can be ordered chronologically in a directory.
// They can thus be arranged in archives, feeds, etc.
// Posts are also assumed to have a title and can be excerpted.
POST Type = "post"
// The rest of the templates: no layout and no post
PAGE Type = "page"
)
type Template struct {
Type Type
SrcPath string
Metadata map[string]interface{}
}
// TODO think about knowledge boundaries
// should this know to tell if its a layout based on srcPath conventions?
// should it be able to detect its own type? does it still make sense to track a template type,
// separate from the site?
func Parse(path string) (*Template, error) {
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 {
return &Template{Type: STATIC}, nil
}
// read and parse the yaml from the front matter
var yamlContent []byte
closed := false
for scanner.Scan() {
line := scanner.Text()
if strings.TrimSpace(line) == FM_SEPARATOR {
closed = true
break
}
yamlContent = append(yamlContent, []byte(line+"\n")...)
}
if !closed {
return nil, errors.New("front matter not closed")
}
var metadata map[string]interface{}
if len(yamlContent) != 0 {
err := yaml.Unmarshal([]byte(yamlContent), &metadata)
if err != nil {
return nil, fmt.Errorf("invalid yaml: %s", err)
}
}
templ := Template{SrcPath: path, Metadata: metadata}
// FIXME this also should check that it's in the root folder
if strings.HasSuffix(filepath.Dir(templ.SrcPath), "layouts") {
templ.Type = LAYOUT
} else if _, ok := metadata["date"]; ok {
templ.Type = POST
} else {
templ.Type = PAGE
}
return &templ, nil
}
func (templ Template) Ext() string {
ext := filepath.Ext(templ.SrcPath)
if ext == ".org" {
ext = ".html"
}
return ext
}
func (templ Template) Render(context map[string]interface{}) (string, error) {
file, _ := os.Open(templ.SrcPath)
defer file.Close()
scanner := bufio.NewScanner(file)
// first line is the front matter delimiter, Scan to skip
// and keep skipping until the closing delimiter
scanner.Scan()
scanner.Scan()
for scanner.Text() != FM_SEPARATOR {
scanner.Scan()
}
// now read the proper template contents to memory
contents := ""
for scanner.Scan() {
contents += scanner.Text() + "\n"
}
if strings.HasSuffix(templ.SrcPath, ".org") {
// if it's an org file, convert to html
doc := org.New().Parse(strings.NewReader(contents), templ.SrcPath)
return doc.Write(org.NewHTMLWriter())
}
// for other file types, assume a liquid template
engine := liquid.NewEngine()
return engine.ParseAndRenderString(contents, context)
}