Commit 882a6404 authored by Samuel Tan's avatar Samuel Tan Committed by Russ Cox

html/template: only search identifier nodes for predefined escapers

Predefined escapers (i.e. "html" and "urlquery") should only occur in
Identifier nodes, and never in Field or Chain nodes, since these are
global functions that return string values (see inline comments for more
details). Therefore, skip Chain and Field nodes when searching for
predefined escapers in template pipelines.

Also, make a non-functional change two existing test cases to avoid
giving the impression that it is valid to reference a field of a
predefined escaper.

Fixes #20323

Change-Id: I34f722f443c778699fcdd575dc3e0fd1fd6f2eb3
Reviewed-on: https://go-review.googlesource.com/43296Reviewed-by: default avatarSamuel Tan <samueltan@google.com>
Reviewed-by: default avatarMike Samuel <mikesamuel@gmail.com>
Reviewed-by: default avatarRuss Cox <rsc@golang.org>
Run-TryBot: Russ Cox <rsc@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent 2c3c8c42
...@@ -139,20 +139,6 @@ func (e *escaper) escape(c context, n parse.Node) context { ...@@ -139,20 +139,6 @@ func (e *escaper) escape(c context, n parse.Node) context {
panic("escaping " + n.String() + " is unimplemented") panic("escaping " + n.String() + " is unimplemented")
} }
// allIdents returns the names of the identifiers under the Ident field of the node,
// which might be a singleton (Identifier) or a slice (Field or Chain).
func allIdents(node parse.Node) []string {
switch node := node.(type) {
case *parse.IdentifierNode:
return []string{node.Ident}
case *parse.FieldNode:
return node.Ident
case *parse.ChainNode:
return node.Field
}
return nil
}
// escapeAction escapes an action template node. // escapeAction escapes an action template node.
func (e *escaper) escapeAction(c context, n *parse.ActionNode) context { func (e *escaper) escapeAction(c context, n *parse.ActionNode) context {
if len(n.Pipe.Decl) != 0 { if len(n.Pipe.Decl) != 0 {
...@@ -162,7 +148,18 @@ func (e *escaper) escapeAction(c context, n *parse.ActionNode) context { ...@@ -162,7 +148,18 @@ func (e *escaper) escapeAction(c context, n *parse.ActionNode) context {
c = nudge(c) c = nudge(c)
// Check for disallowed use of predefined escapers in the pipeline. // Check for disallowed use of predefined escapers in the pipeline.
for pos, idNode := range n.Pipe.Cmds { for pos, idNode := range n.Pipe.Cmds {
for _, ident := range allIdents(idNode.Args[0]) { node, ok := idNode.Args[0].(*parse.IdentifierNode)
if !ok {
// A predefined escaper "esc" will never be found as an identifier in a
// Chain or Field node, since:
// - "esc.x ..." is invalid, since predefined escapers return strings, and
// strings do not have methods, keys or fields.
// - "... .esc" is invalid, since predefined escapers are global functions,
// not methods or fields of any types.
// Therefore, it is safe to ignore these two node types.
continue
}
ident := node.Ident
if _, ok := predefinedEscapers[ident]; ok { if _, ok := predefinedEscapers[ident]; ok {
if pos < len(n.Pipe.Cmds)-1 || if pos < len(n.Pipe.Cmds)-1 ||
c.state == stateAttr && c.delim == delimSpaceOrTagEnd && ident == "html" { c.state == stateAttr && c.delim == delimSpaceOrTagEnd && ident == "html" {
...@@ -173,7 +170,6 @@ func (e *escaper) escapeAction(c context, n *parse.ActionNode) context { ...@@ -173,7 +170,6 @@ func (e *escaper) escapeAction(c context, n *parse.ActionNode) context {
} }
} }
} }
}
s := make([]string, 0, 3) s := make([]string, 0, 3)
switch c.state { switch c.state {
case stateError: case stateError:
......
...@@ -685,6 +685,40 @@ func TestEscape(t *testing.T) { ...@@ -685,6 +685,40 @@ func TestEscape(t *testing.T) {
} }
} }
func TestEscapeMap(t *testing.T) {
data := map[string]string{
"html": `<h1>Hi!</h1>`,
"urlquery": `http://www.foo.com/index.html?title=main`,
}
for _, test := range [...]struct {
desc, input, output string
}{
// covering issue 20323
{
"field with predefined escaper name 1",
`{{.html | print}}`,
`&lt;h1&gt;Hi!&lt;/h1&gt;`,
},
// covering issue 20323
{
"field with predefined escaper name 2",
`{{.urlquery | print}}`,
`http://www.foo.com/index.html?title=main`,
},
} {
tmpl := Must(New("").Parse(test.input))
b := new(bytes.Buffer)
if err := tmpl.Execute(b, data); err != nil {
t.Errorf("%s: template execution failed: %s", test.desc, err)
continue
}
if w, g := test.output, b.String(); w != g {
t.Errorf("%s: escaped output: want\n\t%q\ngot\n\t%q", test.desc, w, g)
continue
}
}
}
func TestEscapeSet(t *testing.T) { func TestEscapeSet(t *testing.T) {
type dataItem struct { type dataItem struct {
Children []*dataItem Children []*dataItem
...@@ -1595,14 +1629,14 @@ func TestEnsurePipelineContains(t *testing.T) { ...@@ -1595,14 +1629,14 @@ func TestEnsurePipelineContains(t *testing.T) {
}, },
{ {
// covering issue 10801 // covering issue 10801
"{{.X | js.x }}", "{{.X | println.x }}",
".X | js.x | urlquery | html", ".X | println.x | urlquery | html",
[]string{"urlquery", "html"}, []string{"urlquery", "html"},
}, },
{ {
// covering issue 10801 // covering issue 10801
"{{.X | (print 12 | js).x }}", "{{.X | (print 12 | println).x }}",
".X | (print 12 | js).x | urlquery | html", ".X | (print 12 | println).x | urlquery | html",
[]string{"urlquery", "html"}, []string{"urlquery", "html"},
}, },
// The following test cases ensure that the merging of internal escapers // The following test cases ensure that the merging of internal escapers
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment