| // Copyright 2011 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package template |
| |
| import ( |
| "fmt" |
| "reflect" |
| "text/template/parse" |
| ) |
| |
| // common holds the information shared by related templates. |
| type common struct { |
| tmpl map[string]*Template |
| // We use two maps, one for parsing and one for execution. |
| // This separation makes the API cleaner since it doesn't |
| // expose reflection to the client. |
| parseFuncs FuncMap |
| execFuncs map[string]reflect.Value |
| } |
| |
| // Template is the representation of a parsed template. The *parse.Tree |
| // field is exported only for use by html/template and should be treated |
| // as unexported by all other clients. |
| type Template struct { |
| name string |
| *parse.Tree |
| *common |
| leftDelim string |
| rightDelim string |
| } |
| |
| // New allocates a new template with the given name. |
| func New(name string) *Template { |
| return &Template{ |
| name: name, |
| } |
| } |
| |
| // Name returns the name of the template. |
| func (t *Template) Name() string { |
| return t.name |
| } |
| |
| // New allocates a new template associated with the given one and with the same |
| // delimiters. The association, which is transitive, allows one template to |
| // invoke another with a {{template}} action. |
| func (t *Template) New(name string) *Template { |
| t.init() |
| return &Template{ |
| name: name, |
| common: t.common, |
| leftDelim: t.leftDelim, |
| rightDelim: t.rightDelim, |
| } |
| } |
| |
| func (t *Template) init() { |
| if t.common == nil { |
| t.common = new(common) |
| t.tmpl = make(map[string]*Template) |
| t.parseFuncs = make(FuncMap) |
| t.execFuncs = make(map[string]reflect.Value) |
| } |
| } |
| |
| // Clone returns a duplicate of the template, including all associated |
| // templates. The actual representation is not copied, but the name space of |
| // associated templates is, so further calls to Parse in the copy will add |
| // templates to the copy but not to the original. Clone can be used to prepare |
| // common templates and use them with variant definitions for other templates |
| // by adding the variants after the clone is made. |
| func (t *Template) Clone() (*Template, error) { |
| nt := t.copy(nil) |
| nt.init() |
| nt.tmpl[t.name] = nt |
| for k, v := range t.tmpl { |
| if k == t.name { // Already installed. |
| continue |
| } |
| // The associated templates share nt's common structure. |
| tmpl := v.copy(nt.common) |
| nt.tmpl[k] = tmpl |
| } |
| for k, v := range t.parseFuncs { |
| nt.parseFuncs[k] = v |
| } |
| for k, v := range t.execFuncs { |
| nt.execFuncs[k] = v |
| } |
| return nt, nil |
| } |
| |
| // copy returns a shallow copy of t, with common set to the argument. |
| func (t *Template) copy(c *common) *Template { |
| nt := New(t.name) |
| nt.Tree = t.Tree |
| nt.common = c |
| nt.leftDelim = t.leftDelim |
| nt.rightDelim = t.rightDelim |
| return nt |
| } |
| |
| // AddParseTree creates a new template with the name and parse tree |
| // and associates it with t. |
| func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error) { |
| if t.common != nil && t.tmpl[name] != nil { |
| return nil, fmt.Errorf("template: redefinition of template %q", name) |
| } |
| nt := t.New(name) |
| nt.Tree = tree |
| t.tmpl[name] = nt |
| return nt, nil |
| } |
| |
| // Templates returns a slice of the templates associated with t, including t |
| // itself. |
| func (t *Template) Templates() []*Template { |
| if t.common == nil { |
| return nil |
| } |
| // Return a slice so we don't expose the map. |
| m := make([]*Template, 0, len(t.tmpl)) |
| for _, v := range t.tmpl { |
| m = append(m, v) |
| } |
| return m |
| } |
| |
| // Delims sets the action delimiters to the specified strings, to be used in |
| // subsequent calls to Parse, ParseFiles, or ParseGlob. Nested template |
| // definitions will inherit the settings. An empty delimiter stands for the |
| // corresponding default: {{ or }}. |
| // The return value is the template, so calls can be chained. |
| func (t *Template) Delims(left, right string) *Template { |
| t.leftDelim = left |
| t.rightDelim = right |
| return t |
| } |
| |
| // Funcs adds the elements of the argument map to the template's function map. |
| // It panics if a value in the map is not a function with appropriate return |
| // type. However, it is legal to overwrite elements of the map. The return |
| // value is the template, so calls can be chained. |
| func (t *Template) Funcs(funcMap FuncMap) *Template { |
| t.init() |
| addValueFuncs(t.execFuncs, funcMap) |
| addFuncs(t.parseFuncs, funcMap) |
| return t |
| } |
| |
| // Lookup returns the template with the given name that is associated with t, |
| // or nil if there is no such template. |
| func (t *Template) Lookup(name string) *Template { |
| if t.common == nil { |
| return nil |
| } |
| return t.tmpl[name] |
| } |
| |
| // Parse parses a string into a template. Nested template definitions will be |
| // associated with the top-level template t. Parse may be called multiple times |
| // to parse definitions of templates to associate with t. It is an error if a |
| // resulting template is non-empty (contains content other than template |
| // definitions) and would replace a non-empty template with the same name. |
| // (In multiple calls to Parse with the same receiver template, only one call |
| // can contain text other than space, comments, and template definitions.) |
| func (t *Template) Parse(text string) (*Template, error) { |
| t.init() |
| trees, err := parse.Parse(t.name, text, t.leftDelim, t.rightDelim, t.parseFuncs, builtins) |
| if err != nil { |
| return nil, err |
| } |
| // Add the newly parsed trees, including the one for t, into our common structure. |
| for name, tree := range trees { |
| // If the name we parsed is the name of this template, overwrite this template. |
| // The associate method checks it's not a redefinition. |
| tmpl := t |
| if name != t.name { |
| tmpl = t.New(name) |
| } |
| // Even if t == tmpl, we need to install it in the common.tmpl map. |
| if replace, err := t.associate(tmpl, tree); err != nil { |
| return nil, err |
| } else if replace { |
| tmpl.Tree = tree |
| } |
| tmpl.leftDelim = t.leftDelim |
| tmpl.rightDelim = t.rightDelim |
| } |
| return t, nil |
| } |
| |
| // associate installs the new template into the group of templates associated |
| // with t. It is an error to reuse a name except to overwrite an empty |
| // template. The two are already known to share the common structure. |
| // The boolean return value reports wither to store this tree as t.Tree. |
| func (t *Template) associate(new *Template, tree *parse.Tree) (bool, error) { |
| if new.common != t.common { |
| panic("internal error: associate not common") |
| } |
| name := new.name |
| if old := t.tmpl[name]; old != nil { |
| oldIsEmpty := parse.IsEmptyTree(old.Root) |
| newIsEmpty := parse.IsEmptyTree(tree.Root) |
| if newIsEmpty { |
| // Whether old is empty or not, new is empty; no reason to replace old. |
| return false, nil |
| } |
| if !oldIsEmpty { |
| return false, fmt.Errorf("template: redefinition of template %q", name) |
| } |
| } |
| t.tmpl[name] = new |
| return true, nil |
| } |