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 }