salix/tags.go

83 lines
2.2 KiB
Go

package salix
import (
"bytes"
"io"
"go.elara.ws/salix/ast"
)
// Tag represents a tag in a Salix template
type Tag interface {
Run(tc *TagContext, block, args []ast.Node) error
}
var globalTags = map[string]Tag{
"if": ifTag{},
"for": forTag{},
"include": includeTag{},
"macro": macroTag{},
}
// TagContext is passed to Tag implementations to allow them to control the interpreter
type TagContext struct {
Tag ast.Tag
w io.Writer
t *Template
local map[string]any
}
// Execute runs the interpreter on the given AST nodes, with the given local variables.
func (tc *TagContext) Execute(nodes []ast.Node, local map[string]any) error {
return tc.t.execute(tc.w, nodes, mergeMap(tc.local, local))
}
// ExecuteToMemory runs the interpreter on the given AST nodes, with the given local variables, and
// returns the resulting bytes rather than writing them out.
func (tc *TagContext) ExecuteToMemory(nodes []ast.Node, local map[string]any) ([]byte, error) {
buf := &bytes.Buffer{}
err := tc.t.execute(buf, nodes, mergeMap(tc.local, local))
if err != nil {
return nil, err
}
return buf.Bytes(), nil
}
// GetValue evaluates the given AST node using the given local variables.
func (tc *TagContext) GetValue(node ast.Node, local map[string]any) (any, error) {
return tc.t.getValue(node, mergeMap(tc.local, local))
}
// PosError returns an error with the file position prepended. This should be used
// for errors wherever possible, to make it easier for users to find errors.
func (tc *TagContext) PosError(node ast.Node, format string, v ...any) error {
return ast.PosError(node, format, v...)
}
// NodeToString returns a textual representation of the given AST node for users to see,
// such as in error messages. This does not directly correlate to Salix source code.
func (tc *TagContext) NodeToString(node ast.Node) string {
return valueToString(node)
}
// Write writes b to the underlying writer. It implements
// the io.Writer interface.
func (tc *TagContext) Write(b []byte) (int, error) {
return tc.w.Write(b)
}
func mergeMap(a, b map[string]any) map[string]any {
out := map[string]any{}
if a != nil {
for k, v := range a {
out[k] = v
}
}
if b != nil {
for k, v := range b {
out[k] = v
}
}
return out
}