|  | // 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 ( | 
|  | "reflect" | 
|  | "sync" | 
|  |  | 
|  | "golang.org/x/website/internal/backport/text/template/parse" | 
|  | ) | 
|  |  | 
|  | // common holds the information shared by related templates. | 
|  | type common struct { | 
|  | tmpl   map[string]*Template // Map from name to defined templates. | 
|  | muTmpl sync.RWMutex         // protects tmpl | 
|  | option option | 
|  | // 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. | 
|  | muFuncs    sync.RWMutex // protects parseFuncs and execFuncs | 
|  | 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, undefined template with the given name. | 
|  | func New(name string) *Template { | 
|  | t := &Template{ | 
|  | name: name, | 
|  | } | 
|  | t.init() | 
|  | return t | 
|  | } | 
|  |  | 
|  | // Name returns the name of the template. | 
|  | func (t *Template) Name() string { | 
|  | return t.name | 
|  | } | 
|  |  | 
|  | // New allocates a new, undefined 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. | 
|  | // | 
|  | // Because associated templates share underlying data, template construction | 
|  | // cannot be done safely in parallel. Once the templates are constructed, they | 
|  | // can be executed in parallel. | 
|  | func (t *Template) New(name string) *Template { | 
|  | t.init() | 
|  | nt := &Template{ | 
|  | name:       name, | 
|  | common:     t.common, | 
|  | leftDelim:  t.leftDelim, | 
|  | rightDelim: t.rightDelim, | 
|  | } | 
|  | return nt | 
|  | } | 
|  |  | 
|  | // init guarantees that t has a valid common structure. | 
|  | func (t *Template) init() { | 
|  | if t.common == nil { | 
|  | c := new(common) | 
|  | c.tmpl = make(map[string]*Template) | 
|  | c.parseFuncs = make(FuncMap) | 
|  | c.execFuncs = make(map[string]reflect.Value) | 
|  | t.common = c | 
|  | } | 
|  | } | 
|  |  | 
|  | // 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() | 
|  | if t.common == nil { | 
|  | return nt, nil | 
|  | } | 
|  | t.muTmpl.RLock() | 
|  | defer t.muTmpl.RUnlock() | 
|  | for k, v := range t.tmpl { | 
|  | if k == t.name { | 
|  | nt.tmpl[t.name] = nt | 
|  | continue | 
|  | } | 
|  | // The associated templates share nt's common structure. | 
|  | tmpl := v.copy(nt.common) | 
|  | nt.tmpl[k] = tmpl | 
|  | } | 
|  | t.muFuncs.RLock() | 
|  | defer t.muFuncs.RUnlock() | 
|  | 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 { | 
|  | return &Template{ | 
|  | name:       t.name, | 
|  | Tree:       t.Tree, | 
|  | common:     c, | 
|  | leftDelim:  t.leftDelim, | 
|  | rightDelim: t.rightDelim, | 
|  | } | 
|  | } | 
|  |  | 
|  | // AddParseTree associates the argument parse tree with the template t, giving | 
|  | // it the specified name. If the template has not been defined, this tree becomes | 
|  | // its definition. If it has been defined and already has that name, the existing | 
|  | // definition is replaced; otherwise a new template is created, defined, and returned. | 
|  | func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error) { | 
|  | t.muTmpl.Lock() | 
|  | defer t.muTmpl.Unlock() | 
|  | t.init() | 
|  | nt := t | 
|  | if name != t.name { | 
|  | nt = t.New(name) | 
|  | } | 
|  | // Even if nt == t, we need to install it in the common.tmpl map. | 
|  | if t.associate(nt, tree) || nt.Tree == nil { | 
|  | nt.Tree = tree | 
|  | } | 
|  | return nt, nil | 
|  | } | 
|  |  | 
|  | // Templates returns a slice of defined templates associated with t. | 
|  | func (t *Template) Templates() []*Template { | 
|  | if t.common == nil { | 
|  | return nil | 
|  | } | 
|  | // Return a slice so we don't expose the map. | 
|  | t.muTmpl.RLock() | 
|  | defer t.muTmpl.RUnlock() | 
|  | 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.init() | 
|  | t.leftDelim = left | 
|  | t.rightDelim = right | 
|  | return t | 
|  | } | 
|  |  | 
|  | // Funcs adds the elements of the argument map to the template's function map. | 
|  | // It must be called before the template is parsed. | 
|  | // It panics if a value in the map is not a function with appropriate return | 
|  | // type or if the name cannot be used syntactically as a function in a template. | 
|  | // 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() | 
|  | t.muFuncs.Lock() | 
|  | defer t.muFuncs.Unlock() | 
|  | addValueFuncs(t.execFuncs, funcMap) | 
|  | addFuncs(t.parseFuncs, funcMap) | 
|  | return t | 
|  | } | 
|  |  | 
|  | // Lookup returns the template with the given name that is associated with t. | 
|  | // It returns nil if there is no such template or the template has no definition. | 
|  | func (t *Template) Lookup(name string) *Template { | 
|  | if t.common == nil { | 
|  | return nil | 
|  | } | 
|  | t.muTmpl.RLock() | 
|  | defer t.muTmpl.RUnlock() | 
|  | return t.tmpl[name] | 
|  | } | 
|  |  | 
|  | // Parse parses text as a template body for t. | 
|  | // Named template definitions ({{define ...}} or {{block ...}} statements) in text | 
|  | // define additional templates associated with t and are removed from the | 
|  | // definition of t itself. | 
|  | // | 
|  | // Templates can be redefined in successive calls to Parse. | 
|  | // A template definition with a body containing only white space and comments | 
|  | // is considered empty and will not replace an existing template's body. | 
|  | // This allows using Parse to add new named template definitions without | 
|  | // overwriting the main template body. | 
|  | func (t *Template) Parse(text string) (*Template, error) { | 
|  | t.init() | 
|  | t.muFuncs.RLock() | 
|  | trees, err := parse.Parse(t.name, text, t.leftDelim, t.rightDelim, t.parseFuncs, builtins()) | 
|  | t.muFuncs.RUnlock() | 
|  | 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 _, err := t.AddParseTree(name, tree); err != nil { | 
|  | return nil, err | 
|  | } | 
|  | } | 
|  | return t, nil | 
|  | } | 
|  |  | 
|  | // associate installs the new template into the group of templates associated | 
|  | // with t. The two are already known to share the common structure. | 
|  | // The boolean return value reports whether to store this tree as t.Tree. | 
|  | func (t *Template) associate(new *Template, tree *parse.Tree) bool { | 
|  | if new.common != t.common { | 
|  | panic("internal error: associate not common") | 
|  | } | 
|  | if old := t.tmpl[new.name]; old != nil && parse.IsEmptyTree(tree.Root) && old.Tree != nil { | 
|  | // If a template by that name exists, | 
|  | // don't replace it with an empty template. | 
|  | return false | 
|  | } | 
|  | t.tmpl[new.name] = new | 
|  | return true | 
|  | } |