package pongo2 import ( "bytes" "fmt" "io" "strings" ) type TemplateWriter interface { io.Writer WriteString(string) (int, error) } type templateWriter struct { w io.Writer } func (tw *templateWriter) WriteString(s string) (int, error) { return tw.w.Write([]byte(s)) } func (tw *templateWriter) Write(b []byte) (int, error) { return tw.w.Write(b) } type Template struct { set *TemplateSet // Input isTplString bool name string tpl string size int // Calculation tokens []*Token parser *Parser // first come, first serve (it's important to not override existing entries in here) level int parent *Template child *Template blocks map[string]*NodeWrapper exportedMacros map[string]*tagMacroNode // Output root *nodeDocument // Options allow you to change the behavior of template-engine. // You can change the options before calling the Execute method. Options *Options } func newTemplateString(set *TemplateSet, tpl []byte) (*Template, error) { return newTemplate(set, "", true, tpl) } func newTemplate(set *TemplateSet, name string, isTplString bool, tpl []byte) (*Template, error) { strTpl := string(tpl) // Create the template t := &Template{ set: set, isTplString: isTplString, name: name, tpl: strTpl, size: len(strTpl), blocks: make(map[string]*NodeWrapper), exportedMacros: make(map[string]*tagMacroNode), Options: newOptions(), } // Copy all settings from another Options. t.Options.Update(set.Options) // Tokenize it tokens, err := lex(name, strTpl) if err != nil { return nil, err } t.tokens = tokens // For debugging purposes, show all tokens: /*for i, t := range tokens { fmt.Printf("%3d. %s\n", i, t) }*/ // Parse it err = t.parse() if err != nil { return nil, err } return t, nil } func (tpl *Template) newContextForExecution(context Context) (*Template, *ExecutionContext, error) { if tpl.Options.TrimBlocks || tpl.Options.LStripBlocks { // Issue #94 https://github.com/flosch/pongo2/issues/94 // If an application configures pongo2 template to trim_blocks, // the first newline after a template tag is removed automatically (like in PHP). prev := &Token{ Typ: TokenHTML, Val: "\n", } for _, t := range tpl.tokens { if tpl.Options.LStripBlocks { if prev.Typ == TokenHTML && t.Typ != TokenHTML && t.Val == "{%" { prev.Val = strings.TrimRight(prev.Val, "\t ") } } if tpl.Options.TrimBlocks { if prev.Typ != TokenHTML && t.Typ == TokenHTML && prev.Val == "%}" { if len(t.Val) > 0 && t.Val[0] == '\n' { t.Val = t.Val[1:len(t.Val)] } } } prev = t } } // Determine the parent to be executed (for template inheritance) parent := tpl for parent.parent != nil { parent = parent.parent } // Create context if none is given newContext := make(Context) newContext.Update(tpl.set.Globals) if context != nil { newContext.Update(context) if len(newContext) > 0 { // Check for context name syntax err := newContext.checkForValidIdentifiers() if err != nil { return parent, nil, err } // Check for clashes with macro names for k := range newContext { _, has := tpl.exportedMacros[k] if has { return parent, nil, &Error{ Filename: tpl.name, Sender: "execution", OrigError: fmt.Errorf("context key name '%s' clashes with macro '%s'", k, k), } } } } } // Create operational context ctx := newExecutionContext(parent, newContext) return parent, ctx, nil } func (tpl *Template) execute(context Context, writer TemplateWriter) error { parent, ctx, err := tpl.newContextForExecution(context) if err != nil { return err } // Run the selected document if err := parent.root.Execute(ctx, writer); err != nil { return err } return nil } func (tpl *Template) newTemplateWriterAndExecute(context Context, writer io.Writer) error { return tpl.execute(context, &templateWriter{w: writer}) } func (tpl *Template) newBufferAndExecute(context Context) (*bytes.Buffer, error) { // Create output buffer // We assume that the rendered template will be 30% larger buffer := bytes.NewBuffer(make([]byte, 0, int(float64(tpl.size)*1.3))) if err := tpl.execute(context, buffer); err != nil { return nil, err } return buffer, nil } // Executes the template with the given context and writes to writer (io.Writer) // on success. Context can be nil. Nothing is written on error; instead the error // is being returned. func (tpl *Template) ExecuteWriter(context Context, writer io.Writer) error { buf, err := tpl.newBufferAndExecute(context) if err != nil { return err } _, err = buf.WriteTo(writer) if err != nil { return err } return nil } // Same as ExecuteWriter. The only difference between both functions is that // this function might already have written parts of the generated template in the // case of an execution error because there's no intermediate buffer involved for // performance reasons. This is handy if you need high performance template // generation or if you want to manage your own pool of buffers. func (tpl *Template) ExecuteWriterUnbuffered(context Context, writer io.Writer) error { return tpl.newTemplateWriterAndExecute(context, writer) } // Executes the template and returns the rendered template as a []byte func (tpl *Template) ExecuteBytes(context Context) ([]byte, error) { // Execute template buffer, err := tpl.newBufferAndExecute(context) if err != nil { return nil, err } return buffer.Bytes(), nil } // Executes the template and returns the rendered template as a string func (tpl *Template) Execute(context Context) (string, error) { // Execute template buffer, err := tpl.newBufferAndExecute(context) if err != nil { return "", err } return buffer.String(), nil } func (tpl *Template) ExecuteBlocks(context Context, blocks []string) (map[string]string, error) { var parents []*Template result := make(map[string]string) parent := tpl for parent != nil { parents = append(parents, parent) parent = parent.parent } for _, t := range parents { buffer := bytes.NewBuffer(make([]byte, 0, int(float64(t.size)*1.3))) _, ctx, err := t.newContextForExecution(context) if err != nil { return nil, err } for _, blockName := range blocks { if _, ok := result[blockName]; ok { continue } if blockWrapper, ok := t.blocks[blockName]; ok { bErr := blockWrapper.Execute(ctx, buffer) if bErr != nil { return nil, bErr } result[blockName] = buffer.String() buffer.Reset() } } // We have found all blocks if len(blocks) == len(result) { break } } return result, nil }