1
0
mirror of https://github.com/danog/liquid.git synced 2025-01-22 16:31:13 +01:00

Render tree is distinct type from parse AST

This commit is contained in:
Oliver Steele 2017-07-06 09:21:26 -04:00
parent 8f63cb781e
commit 803471c15f
10 changed files with 124 additions and 81 deletions

View File

@ -8,13 +8,11 @@ import (
// ASTNode is a node of an AST.
type ASTNode interface {
// Render evaluates an AST node and writes the result to an io.Writer.
}
// ASTBlock represents a {% tag %}…{% endtag %}.
type ASTBlock struct {
Chunk
renderer func(io.Writer, Context) error
syntax BlockSyntax
Body []ASTNode
Branches []*ASTBlock

View File

@ -6,9 +6,9 @@ import (
)
// BlockParser builds a renderer for the tag instance.
type BlockParser func(ASTBlock) (func(io.Writer, Context) error, error)
type BlockParser func(BlockNode) (func(io.Writer, Context) error, error)
// blockDef tells the parser how to parse control tags.
// blockDef tells the parser how to parse control tagc.
type blockDef struct {
name string
isBranchTag, isEndTag bool
@ -42,38 +42,38 @@ func (c *blockDef) ParentTags() []string {
}
func (c *blockDef) TagName() string { return c.name }
func (s Config) addBlockDef(ct *blockDef) {
s.blockDefs[ct.name] = ct
func (c Config) addBlockDef(ct *blockDef) {
c.blockDefs[ct.name] = ct
}
func (s Config) findBlockDef(name string) (*blockDef, bool) {
ct, found := s.blockDefs[name]
func (c Config) findBlockDef(name string) (*blockDef, bool) {
ct, found := c.blockDefs[name]
return ct, found
}
// BlockSyntax is part of the Grammar interface.
func (s Config) BlockSyntax(name string) (BlockSyntax, bool) {
ct, found := s.blockDefs[name]
func (c Config) BlockSyntax(name string) (BlockSyntax, bool) {
ct, found := c.blockDefs[name]
return ct, found
}
type blockDefBuilder struct {
s Config
cfg Config
tag *blockDef
}
// AddBlock defines a control tag and its matching end tag.
func (s Config) AddBlock(name string) blockDefBuilder { // nolint: golint
func (c Config) AddBlock(name string) blockDefBuilder { // nolint: golint
ct := &blockDef{name: name}
s.addBlockDef(ct)
s.addBlockDef(&blockDef{name: "end" + name, isEndTag: true, parent: ct})
return blockDefBuilder{s, ct}
c.addBlockDef(ct)
c.addBlockDef(&blockDef{name: "end" + name, isEndTag: true, parent: ct})
return blockDefBuilder{c, ct}
}
// Branch tells the parser that the named tag can appear immediately between this tag and its end tag,
// so long as it is not nested within any other control tags.
// so long as it is not nested within any other control tagc.
func (b blockDefBuilder) Branch(name string) blockDefBuilder {
b.s.addBlockDef(&blockDef{name: name, isBranchTag: true, parent: b.tag})
b.cfg.addBlockDef(&blockDef{name: name, isBranchTag: true, parent: b.tag})
return b
}
@ -84,7 +84,7 @@ func (b blockDefBuilder) Governs(_ []string) blockDefBuilder {
// SameSyntaxAs tells the parser that this tag has the same syntax as the named tag.
func (b blockDefBuilder) SameSyntaxAs(name string) blockDefBuilder {
rt := b.s.blockDefs[name]
rt := b.cfg.blockDefs[name]
if rt == nil {
panic(fmt.Errorf("undefined: %s", name))
}
@ -99,7 +99,7 @@ func (b blockDefBuilder) Parser(fn BlockParser) {
// Renderer sets the render action for a control tag definition.
func (b blockDefBuilder) Renderer(fn func(io.Writer, Context) error) {
b.tag.parser = func(node ASTBlock) (func(io.Writer, Context) error, error) {
b.tag.parser = func(node BlockNode) (func(io.Writer, Context) error, error) {
// TODO parse error if there are arguments?
return fn, nil
}

View File

@ -2,7 +2,6 @@ package render
import (
"fmt"
"io"
)
// A CompilationError is a parse error during template compilation.
@ -24,7 +23,7 @@ func (c Config) Compile(source string) (ASTNode, error) {
}
// nolint: gocyclo
func (c Config) compileNode(n ASTNode) (ASTNode, error) {
func (c Config) compileNode(n ASTNode) (Node, error) {
switch n := n.(type) {
case *ASTBlock:
body, err := c.compileNodes(n.Body)
@ -40,54 +39,53 @@ func (c Config) compileNode(n ASTNode) (ASTNode, error) {
if !ok {
return nil, compilationErrorf("undefined tag %q", n.Name)
}
var renderer func(io.Writer, Context) error
if cd.parser != nil {
r, err := cd.parser(*n)
if err != nil {
return nil, err
}
renderer = r
}
return &ASTBlock{
node := BlockNode{
Chunk: n.Chunk,
renderer: renderer,
syntax: n.syntax,
Body: body,
Branches: branches,
}, nil
}
if cd.parser != nil {
r, err := cd.parser(node)
if err != nil {
return nil, err
}
node.renderer = r
}
return &node, nil
case *ASTFunctional:
return &ASTFunctional{n.Chunk, n.render}, nil
return &FunctionalNode{n.Chunk, n.render}, nil
case *ASTRaw:
return &ASTRaw{n.slices}, nil
return &RawNode{n.slices}, nil
case *ASTSeq:
children, err := c.compileNodes(n.Children)
if err != nil {
return nil, err
}
return &ASTSeq{children}, nil
return &SeqNode{children}, nil
case *ASTText:
return &ASTText{n.Chunk}, nil
return &TextNode{n.Chunk}, nil
case *ASTObject:
return &ASTObject{n.Chunk, n.expr}, nil
return &ObjectNode{n.Chunk, n.expr}, nil
default:
panic(fmt.Errorf("un-compilable node type %T", n))
}
}
func (c Config) compileBlocks(blocks []*ASTBlock) ([]*ASTBlock, error) {
out := make([]*ASTBlock, 0, len(blocks))
func (c Config) compileBlocks(blocks []*ASTBlock) ([]*BlockNode, error) {
out := make([]*BlockNode, 0, len(blocks))
for _, child := range blocks {
compiled, err := c.compileNode(child)
if err != nil {
return nil, err
}
out = append(out, compiled.(*ASTBlock))
out = append(out, compiled.(*BlockNode))
}
return out, nil
}
func (c Config) compileNodes(nodes []ASTNode) ([]ASTNode, error) {
out := make([]ASTNode, 0, len(nodes))
func (c Config) compileNodes(nodes []ASTNode) ([]Node, error) {
out := make([]Node, 0, len(nodes))
for _, child := range nodes {
compiled, err := c.compileNode(child)
if err != nil {

View File

@ -9,7 +9,7 @@ import (
)
func addCompilerTestTags(s Config) {
s.AddBlock("block").Parser(func(c ASTBlock) (func(io.Writer, Context) error, error) {
s.AddBlock("block").Parser(func(c BlockNode) (func(io.Writer, Context) error, error) {
return nil, fmt.Errorf("block compiler error")
})
}

View File

@ -18,7 +18,7 @@ type Context interface {
EvaluateStatement(tag, source string) (interface{}, error)
ExpandTagArg() (string, error)
InnerString() (string, error)
RenderChild(io.Writer, *ASTBlock) error
RenderChild(io.Writer, *BlockNode) error
RenderChildren(io.Writer) error
RenderFile(string, map[string]interface{}) (string, error)
Set(name string, value interface{})
@ -29,8 +29,8 @@ type Context interface {
type renderContext struct {
ctx nodeContext
node *ASTFunctional
cn *ASTBlock
node *FunctionalNode
cn *BlockNode
}
// Evaluate evaluates an expression within the template context.
@ -81,8 +81,8 @@ func (c renderContext) ExpandTagArg() (string, error) {
}
// RenderChild renders a node.
func (c renderContext) RenderChild(w io.Writer, b *ASTBlock) error {
return c.ctx.RenderASTSequence(w, b.Body)
func (c renderContext) RenderChild(w io.Writer, b *BlockNode) error {
return c.ctx.RenderSequence(w, b.Body)
}
// RenderChildren renders the current node's children.
@ -90,7 +90,7 @@ func (c renderContext) RenderChildren(w io.Writer) error {
if c.cn == nil {
return nil
}
return c.ctx.RenderASTSequence(w, c.cn.Body)
return c.ctx.RenderSequence(w, c.cn.Body)
}
func (c renderContext) RenderFile(filename string, b map[string]interface{}) (string, error) {

View File

@ -19,32 +19,16 @@ func Errorf(format string, a ...interface{}) Error {
return Error(fmt.Sprintf(format, a...))
}
// Render renders the AST rooted at node to the writer.
func Render(node ASTNode, w io.Writer, b map[string]interface{}, c Config) error {
// Render renders the render tree.
func Render(node Node, w io.Writer, b map[string]interface{}, c Config) error {
return renderNode(node, w, newNodeContext(b, c))
}
func renderNode(node ASTNode, w io.Writer, ctx nodeContext) error { // nolint: gocyclo
func renderNode(node Node, w io.Writer, ctx nodeContext) error { // nolint: gocyclo
switch n := node.(type) {
case *ASTSeq:
for _, c := range n.Children {
if err := renderNode(c, w, ctx); err != nil {
return err
}
}
case *ASTFunctional:
case *FunctionalNode:
return n.render(w, renderContext{ctx, n, nil})
case *ASTText:
_, err := w.Write([]byte(n.Source))
return err
case *ASTRaw:
for _, s := range n.slices {
_, err := w.Write([]byte(s))
if err != nil {
return err
}
}
case *ASTBlock:
case *BlockNode:
cd, ok := ctx.config.findBlockDef(n.Name)
if !ok || cd.parser == nil {
return parseErrorf("unknown tag: %s", n.Name)
@ -54,12 +38,28 @@ func renderNode(node ASTNode, w io.Writer, ctx nodeContext) error { // nolint: g
panic(parseErrorf("unset renderer for %v", n))
}
return renderer(w, renderContext{ctx, nil, n})
case *ASTObject:
case *RawNode:
for _, s := range n.slices {
_, err := w.Write([]byte(s))
if err != nil {
return err
}
}
case *ObjectNode:
value, err := ctx.Evaluate(n.expr)
if err != nil {
return parseErrorf("%s in %s", err, n.Source)
}
return writeObject(value, w)
case *SeqNode:
for _, c := range n.Children {
if err := renderNode(c, w, ctx); err != nil {
return err
}
}
case *TextNode:
_, err := w.Write([]byte(n.Source))
return err
default:
panic(parseErrorf("unknown node type %T", node))
}
@ -91,7 +91,7 @@ func writeObject(value interface{}, w io.Writer) error {
}
// RenderASTSequence renders a sequence of nodes.
func (c nodeContext) RenderASTSequence(w io.Writer, seq []ASTNode) error {
func (c nodeContext) RenderSequence(w io.Writer, seq []Node) error {
for _, n := range seq {
if err := renderNode(n, w, c); err != nil {
return err

47
render/render_node.go Normal file
View File

@ -0,0 +1,47 @@
package render
import (
"io"
"github.com/osteele/liquid/expression"
)
// Node is a node of the render tree.
type Node interface {
}
// BlockNode represents a {% tag %}…{% endtag %}.
type BlockNode struct {
Chunk
renderer func(io.Writer, Context) error
syntax BlockSyntax
Body []Node
Branches []*BlockNode
}
// RawNode holds the text between the start and end of a raw tag.
type RawNode struct {
slices []string
}
// FunctionalNode renders itself via a render function that is created during parsing.
type FunctionalNode struct {
Chunk
render func(io.Writer, Context) error
}
// TextNode is a text chunk, that is rendered verbatim.
type TextNode struct {
Chunk
}
// ObjectNode is an {{ object }} object.
type ObjectNode struct {
Chunk
expr expression.Expression
}
// SeqNode is a sequence of nodes.
type SeqNode struct {
Children []Node
}

View File

@ -11,7 +11,7 @@ import (
)
func addRenderTestTags(s Config) {
s.AddBlock("parse").Parser(func(c ASTBlock) (func(io.Writer, Context) error, error) {
s.AddBlock("parse").Parser(func(c BlockNode) (func(io.Writer, Context) error, error) {
a := c.Args
return func(w io.Writer, c Context) error {
_, err := w.Write([]byte(a))
@ -36,7 +36,7 @@ func addRenderTestTags(s Config) {
return err
}, nil
})
s.AddBlock("err2").Parser(func(c ASTBlock) (func(io.Writer, Context) error, error) {
s.AddBlock("err2").Parser(func(c BlockNode) (func(io.Writer, Context) error, error) {
return func(w io.Writer, c Context) error {
return fmt.Errorf("stage 2 error")
}, nil

View File

@ -31,7 +31,7 @@ func parseLoopExpression(source string) (expression.Expression, error) {
return expr, nil
}
func loopTagParser(node render.ASTBlock) (func(io.Writer, render.Context) error, error) { // nolint: gocyclo
func loopTagParser(node render.BlockNode) (func(io.Writer, render.Context) error, error) { // nolint: gocyclo
expr, err := parseLoopExpression(node.Args)
if err != nil {
return nil, err

View File

@ -37,7 +37,7 @@ func assignTag(source string) (func(io.Writer, render.Context) error, error) {
}, nil
}
func captureTagParser(node render.ASTBlock) (func(io.Writer, render.Context) error, error) {
func captureTagParser(node render.BlockNode) (func(io.Writer, render.Context) error, error) {
// TODO verify syntax
varname := node.Args
return func(w io.Writer, ctx render.Context) error {
@ -50,7 +50,7 @@ func captureTagParser(node render.ASTBlock) (func(io.Writer, render.Context) err
}, nil
}
func caseTagParser(node render.ASTBlock) (func(io.Writer, render.Context) error, error) {
func caseTagParser(node render.BlockNode) (func(io.Writer, render.Context) error, error) {
// TODO parse error on non-empty node.Body
// TODO case can include an else
expr, err := expression.Parse(node.Args)
@ -59,7 +59,7 @@ func caseTagParser(node render.ASTBlock) (func(io.Writer, render.Context) error,
}
type caseRec struct {
expr expression.Expression
node *render.ASTBlock
node *render.BlockNode
}
cases := []caseRec{}
for _, branch := range node.Branches {
@ -87,11 +87,11 @@ func caseTagParser(node render.ASTBlock) (func(io.Writer, render.Context) error,
}, nil
}
func ifTagParser(polarity bool) func(render.ASTBlock) (func(io.Writer, render.Context) error, error) { // nolint: gocyclo
return func(node render.ASTBlock) (func(io.Writer, render.Context) error, error) {
func ifTagParser(polarity bool) func(render.BlockNode) (func(io.Writer, render.Context) error, error) { // nolint: gocyclo
return func(node render.BlockNode) (func(io.Writer, render.Context) error, error) {
type branchRec struct {
test expression.Expression
body *render.ASTBlock
body *render.BlockNode
}
expr, err := expression.Parse(node.Args)
if err != nil {