diff options
author | Ian Lance Taylor <iant@google.com> | 2015-01-15 00:27:56 +0000 |
---|---|---|
committer | Ian Lance Taylor <ian@gcc.gnu.org> | 2015-01-15 00:27:56 +0000 |
commit | f8d9fa9e80b57f89e7877ce6cad8a3464879009b (patch) | |
tree | 58a1724fee16d2b03c65678c4dd9b50bb97137a9 /libgo/go/html | |
parent | 6bd3f109d8d8fa58eeccd6b3504721b4f20c00c2 (diff) | |
download | gcc-f8d9fa9e80b57f89e7877ce6cad8a3464879009b.zip gcc-f8d9fa9e80b57f89e7877ce6cad8a3464879009b.tar.gz gcc-f8d9fa9e80b57f89e7877ce6cad8a3464879009b.tar.bz2 |
libgo, compiler: Upgrade libgo to Go 1.4, except for runtime.
This upgrades all of libgo other than the runtime package to
the Go 1.4 release. In Go 1.4 much of the runtime was
rewritten into Go. Merging that code will take more time and
will not change the API, so I'm putting it off for now.
There are a few runtime changes anyhow, to accomodate other
packages that rely on minor modifications to the runtime
support.
The compiler changes slightly to add a one-bit flag to each
type descriptor kind that is stored directly in an interface,
which for gccgo is currently only pointer types. Another
one-bit flag (gcprog) is reserved because it is used by the gc
compiler, but gccgo does not currently use it.
There is another error check in the compiler since I ran
across it during testing.
gotools/:
* Makefile.am (go_cmd_go_files): Sort entries. Add generate.go.
* Makefile.in: Rebuild.
From-SVN: r219627
Diffstat (limited to 'libgo/go/html')
-rw-r--r-- | libgo/go/html/template/error.go | 16 | ||||
-rw-r--r-- | libgo/go/html/template/escape.go | 74 | ||||
-rw-r--r-- | libgo/go/html/template/escape_test.go | 23 | ||||
-rw-r--r-- | libgo/go/html/template/js_test.go | 2 | ||||
-rw-r--r-- | libgo/go/html/template/template.go | 40 | ||||
-rw-r--r-- | libgo/go/html/template/transition.go | 12 |
6 files changed, 90 insertions, 77 deletions
diff --git a/libgo/go/html/template/error.go b/libgo/go/html/template/error.go index 46e49cc..8f99e1b 100644 --- a/libgo/go/html/template/error.go +++ b/libgo/go/html/template/error.go @@ -6,12 +6,16 @@ package template import ( "fmt" + "text/template/parse" ) // Error describes a problem encountered during template Escaping. type Error struct { // ErrorCode describes the kind of error. ErrorCode ErrorCode + // Node is the node that caused the problem, if known. + // If not nil, it overrides Name and Line. + Node parse.Node // Name is the name of the template in which the error was encountered. Name string // Line is the line number of the error in the template source or 0. @@ -182,9 +186,13 @@ const ( ) func (e *Error) Error() string { - if e.Line != 0 { + switch { + case e.Node != nil: + loc, _ := (*parse.Tree)(nil).ErrorContext(e.Node) + return fmt.Sprintf("html/template:%s: %s", loc, e.Description) + case e.Line != 0: return fmt.Sprintf("html/template:%s:%d: %s", e.Name, e.Line, e.Description) - } else if e.Name != "" { + case e.Name != "": return fmt.Sprintf("html/template:%s: %s", e.Name, e.Description) } return "html/template: " + e.Description @@ -192,6 +200,6 @@ func (e *Error) Error() string { // errorf creates an error given a format string f and args. // The template Name still needs to be supplied. -func errorf(k ErrorCode, line int, f string, args ...interface{}) *Error { - return &Error{k, "", line, fmt.Sprintf(f, args...)} +func errorf(k ErrorCode, node parse.Node, line int, f string, args ...interface{}) *Error { + return &Error{k, node, "", line, fmt.Sprintf(f, args...)} } diff --git a/libgo/go/html/template/escape.go b/libgo/go/html/template/escape.go index 4e37982..ee01fb1 100644 --- a/libgo/go/html/template/escape.go +++ b/libgo/go/html/template/escape.go @@ -13,40 +13,33 @@ import ( "text/template/parse" ) -// escapeTemplates rewrites the named templates, which must be +// escapeTemplate rewrites the named template, which must be // associated with t, to guarantee that the output of any of the named -// templates is properly escaped. Names should include the names of -// all templates that might be Executed but need not include helper -// templates. If no error is returned, then the named templates have +// templates is properly escaped. If no error is returned, then the named templates have // been modified. Otherwise the named templates have been rendered // unusable. -func escapeTemplates(tmpl *Template, names ...string) error { +func escapeTemplate(tmpl *Template, node parse.Node, name string) error { e := newEscaper(tmpl) - for _, name := range names { - c, _ := e.escapeTree(context{}, name, 0) - var err error - if c.err != nil { - err, c.err.Name = c.err, name - } else if c.state != stateText { - err = &Error{ErrEndContext, name, 0, fmt.Sprintf("ends in a non-text context: %v", c)} - } - if err != nil { - // Prevent execution of unsafe templates. - for _, name := range names { - if t := tmpl.set[name]; t != nil { - t.text.Tree = nil - t.Tree = nil - } - } - return err + c, _ := e.escapeTree(context{}, node, name, 0) + var err error + if c.err != nil { + err, c.err.Name = c.err, name + } else if c.state != stateText { + err = &Error{ErrEndContext, nil, name, 0, fmt.Sprintf("ends in a non-text context: %v", c)} + } + if err != nil { + // Prevent execution of unsafe templates. + if t := tmpl.set[name]; t != nil { + t.escapeErr = err + t.text.Tree = nil + t.Tree = nil } + return err } e.commit() - for _, name := range names { - if t := tmpl.set[name]; t != nil { - t.escaped = true - t.Tree = t.text.Tree - } + if t := tmpl.set[name]; t != nil { + t.escapeErr = escapeOK + t.Tree = t.text.Tree } return nil } @@ -168,7 +161,7 @@ func (e *escaper) escapeAction(c context, n *parse.ActionNode) context { case urlPartUnknown: return context{ state: stateError, - err: errorf(ErrAmbigContext, n.Line, "%s appears in an ambiguous URL context", n), + err: errorf(ErrAmbigContext, n, n.Line, "%s appears in an ambiguous URL context", n), } default: panic(c.urlPart.String()) @@ -338,7 +331,7 @@ func escFnsEq(a, b string) bool { func newIdentCmd(identifier string, pos parse.Pos) *parse.CommandNode { return &parse.CommandNode{ NodeType: parse.NodeCommand, - Args: []parse.Node{parse.NewIdentifier(identifier).SetPos(pos)}, + Args: []parse.Node{parse.NewIdentifier(identifier).SetTree(nil).SetPos(pos)}, // TODO: SetTree. } } @@ -372,7 +365,7 @@ func nudge(c context) context { // join joins the two contexts of a branch template node. The result is an // error context if either of the input contexts are error contexts, or if the // the input contexts differ. -func join(a, b context, line int, nodeName string) context { +func join(a, b context, node parse.Node, nodeName string) context { if a.state == stateError { return a } @@ -405,14 +398,14 @@ func join(a, b context, line int, nodeName string) context { // ends in an unquoted value state even though the else branch // ends in stateBeforeValue. if c, d := nudge(a), nudge(b); !(c.eq(a) && d.eq(b)) { - if e := join(c, d, line, nodeName); e.state != stateError { + if e := join(c, d, node, nodeName); e.state != stateError { return e } } return context{ state: stateError, - err: errorf(ErrBranchEnd, line, "{{%s}} branches end in different contexts: %v, %v", nodeName, a, b), + err: errorf(ErrBranchEnd, node, 0, "{{%s}} branches end in different contexts: %v, %v", nodeName, a, b), } } @@ -424,7 +417,7 @@ func (e *escaper) escapeBranch(c context, n *parse.BranchNode, nodeName string) // We check that executing n.List once results in the same context // as executing n.List twice. c1, _ := e.escapeListConditionally(c0, n.List, nil) - c0 = join(c0, c1, n.Line, nodeName) + c0 = join(c0, c1, n, nodeName) if c0.state == stateError { // Make clear that this is a problem on loop re-entry // since developers tend to overlook that branch when @@ -435,7 +428,7 @@ func (e *escaper) escapeBranch(c context, n *parse.BranchNode, nodeName string) } } c1 := e.escapeList(c, n.ElseList) - return join(c0, c1, n.Line, nodeName) + return join(c0, c1, n, nodeName) } // escapeList escapes a list template node. @@ -487,7 +480,7 @@ func (e *escaper) escapeListConditionally(c context, n *parse.ListNode, filter f // escapeTemplate escapes a {{template}} call node. func (e *escaper) escapeTemplate(c context, n *parse.TemplateNode) context { - c, name := e.escapeTree(c, n.Name, n.Line) + c, name := e.escapeTree(c, n, n.Name, n.Line) if name != n.Name { e.editTemplateNode(n, name) } @@ -496,7 +489,7 @@ func (e *escaper) escapeTemplate(c context, n *parse.TemplateNode) context { // escapeTree escapes the named template starting in the given context as // necessary and returns its output context. -func (e *escaper) escapeTree(c context, name string, line int) (context, string) { +func (e *escaper) escapeTree(c context, node parse.Node, name string, line int) (context, string) { // Mangle the template name with the input context to produce a reliable // identifier. dname := c.mangle(name) @@ -512,12 +505,12 @@ func (e *escaper) escapeTree(c context, name string, line int) (context, string) if e.tmpl.set[name] != nil { return context{ state: stateError, - err: errorf(ErrNoSuchTemplate, line, "%q is an incomplete or empty template", name), + err: errorf(ErrNoSuchTemplate, node, line, "%q is an incomplete or empty template", name), }, dname } return context{ state: stateError, - err: errorf(ErrNoSuchTemplate, line, "no such template %q", name), + err: errorf(ErrNoSuchTemplate, node, line, "no such template %q", name), }, dname } if dname != name { @@ -549,8 +542,7 @@ func (e *escaper) computeOutCtx(c context, t *template.Template) context { if !ok && c1.state != stateError { return context{ state: stateError, - // TODO: Find the first node with a line in t.text.Tree.Root - err: errorf(ErrOutputContext, 0, "cannot compute output context for template %s", t.Name()), + err: errorf(ErrOutputContext, t.Tree.Root, 0, "cannot compute output context for template %s", t.Name()), } } return c1 @@ -694,7 +686,7 @@ func contextAfterText(c context, s []byte) (context, int) { if j := bytes.IndexAny(s[:i], "\"'<=`"); j >= 0 { return context{ state: stateError, - err: errorf(ErrBadHTML, 0, "%q in unquoted attr: %q", s[j:j+1], s[:i]), + err: errorf(ErrBadHTML, nil, 0, "%q in unquoted attr: %q", s[j:j+1], s[:i]), }, len(s) } } diff --git a/libgo/go/html/template/escape_test.go b/libgo/go/html/template/escape_test.go index 3ccf93e..ef7b877 100644 --- a/libgo/go/html/template/escape_test.go +++ b/libgo/go/html/template/escape_test.go @@ -861,29 +861,29 @@ func TestErrors(t *testing.T) { // Error cases. { "{{if .Cond}}<a{{end}}", - "z:1: {{if}} branches", + "z:1:5: {{if}} branches", }, { "{{if .Cond}}\n{{else}}\n<a{{end}}", - "z:1: {{if}} branches", + "z:1:5: {{if}} branches", }, { // Missing quote in the else branch. `{{if .Cond}}<a href="foo">{{else}}<a href="bar>{{end}}`, - "z:1: {{if}} branches", + "z:1:5: {{if}} branches", }, { // Different kind of attribute: href implies a URL. "<a {{if .Cond}}href='{{else}}title='{{end}}{{.X}}'>", - "z:1: {{if}} branches", + "z:1:8: {{if}} branches", }, { "\n{{with .X}}<a{{end}}", - "z:2: {{with}} branches", + "z:2:7: {{with}} branches", }, { "\n{{with .X}}<a>{{else}}<a{{end}}", - "z:2: {{with}} branches", + "z:2:7: {{with}} branches", }, { "{{range .Items}}<a{{end}}", @@ -891,7 +891,7 @@ func TestErrors(t *testing.T) { }, { "\n{{range .Items}} x='<a{{end}}", - "z:2: on range loop re-entry: {{range}} branches", + "z:2:8: on range loop re-entry: {{range}} branches", }, { "<a b=1 c={{.H}}", @@ -903,7 +903,7 @@ func TestErrors(t *testing.T) { }, { `<a href="{{if .F}}/foo?a={{else}}/bar/{{end}}{{.H}}">`, - "z:1: {{.H}} appears in an ambiguous URL context", + "z:1:47: {{.H}} appears in an ambiguous URL context", }, { `<a onclick="alert('Hello \`, @@ -932,7 +932,7 @@ func TestErrors(t *testing.T) { }, { `{{template "foo"}}`, - "z:1: no such template \"foo\"", + "z:1:11: no such template \"foo\"", }, { `<div{{template "y"}}>` + @@ -994,6 +994,11 @@ func TestErrors(t *testing.T) { t.Errorf("input=%q: error\n\t%q\ndoes not contain expected string\n\t%q", test.input, got, test.err) continue } + // Check that we get the same error if we call Execute again. + if err := tmpl.Execute(buf, nil); err == nil || err.Error() != got { + t.Errorf("input=%q: unexpected error on second call %q", test.input, err) + + } } } diff --git a/libgo/go/html/template/js_test.go b/libgo/go/html/template/js_test.go index 311e1d2..7af7997 100644 --- a/libgo/go/html/template/js_test.go +++ b/libgo/go/html/template/js_test.go @@ -138,7 +138,7 @@ func TestJSValEscaper(t *testing.T) { // Newlines. {"\r\n\u2028\u2029", `"\r\n\u2028\u2029"`}, // "\v" == "v" on IE 6 so use "\x0b" instead. - {"\t\x0b", `"\u0009\u000b"`}, + {"\t\x0b", `"\t\u000b"`}, {struct{ X, Y int }{1, 2}, `{"X":1,"Y":2}`}, {[]interface{}{}, "[]"}, {[]interface{}{42, "foo", nil}, `[42,"foo",null]`}, diff --git a/libgo/go/html/template/template.go b/libgo/go/html/template/template.go index d389658..ce61701 100644 --- a/libgo/go/html/template/template.go +++ b/libgo/go/html/template/template.go @@ -17,7 +17,8 @@ import ( // Template is a specialized Template from "text/template" that produces a safe // HTML document fragment. type Template struct { - escaped bool + // Sticky error if escaping fails. + escapeErr error // We could embed the text/template field, but it's safer not to because // we need to keep our version of the name space and the underlying // template's in sync. @@ -27,6 +28,9 @@ type Template struct { *nameSpace // common to all associated templates } +// escapeOK is a sentinel value used to indicate valid escaping. +var escapeOK = fmt.Errorf("template escaped correctly") + // nameSpace is the data structure shared by all templates in an association. type nameSpace struct { mu sync.Mutex @@ -51,11 +55,12 @@ func (t *Template) Templates() []*Template { func (t *Template) escape() error { t.nameSpace.mu.Lock() defer t.nameSpace.mu.Unlock() - if !t.escaped { - if err := escapeTemplates(t, t.Name()); err != nil { + if t.escapeErr == nil { + if err := escapeTemplate(t, t.text.Root, t.Name()); err != nil { return err } - t.escaped = true + } else if t.escapeErr != escapeOK { + return t.escapeErr } return nil } @@ -97,14 +102,17 @@ func (t *Template) lookupAndEscapeTemplate(name string) (tmpl *Template, err err if tmpl == nil { return nil, fmt.Errorf("html/template: %q is undefined", name) } + if tmpl.escapeErr != nil && tmpl.escapeErr != escapeOK { + return nil, tmpl.escapeErr + } if tmpl.text.Tree == nil || tmpl.text.Root == nil { return nil, fmt.Errorf("html/template: %q is an incomplete template", name) } if t.text.Lookup(name) == nil { panic("html/template internal error: template escaping out of sync") } - if tmpl != nil && !tmpl.escaped { - err = escapeTemplates(tmpl, name) + if tmpl.escapeErr == nil { + err = escapeTemplate(tmpl, tmpl.text.Root, name) } return tmpl, err } @@ -119,7 +127,7 @@ func (t *Template) lookupAndEscapeTemplate(name string) (tmpl *Template, err err // other than space, comments, and template definitions.) func (t *Template) Parse(src string) (*Template, error) { t.nameSpace.mu.Lock() - t.escaped = false + t.escapeErr = nil t.nameSpace.mu.Unlock() ret, err := t.text.Parse(src) if err != nil { @@ -137,7 +145,7 @@ func (t *Template) Parse(src string) (*Template, error) { tmpl = t.new(name) } // Restore our record of this text/template to its unescaped original state. - tmpl.escaped = false + tmpl.escapeErr = nil tmpl.text = v tmpl.Tree = v.Tree } @@ -151,7 +159,7 @@ func (t *Template) Parse(src string) (*Template, error) { func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error) { t.nameSpace.mu.Lock() defer t.nameSpace.mu.Unlock() - if t.escaped { + if t.escapeErr != nil { return nil, fmt.Errorf("html/template: cannot AddParseTree to %q after it has executed", t.Name()) } text, err := t.text.AddParseTree(name, tree) @@ -159,7 +167,7 @@ func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error return nil, err } ret := &Template{ - false, + nil, text, text.Tree, t.nameSpace, @@ -179,7 +187,7 @@ func (t *Template) AddParseTree(name string, tree *parse.Tree) (*Template, error func (t *Template) Clone() (*Template, error) { t.nameSpace.mu.Lock() defer t.nameSpace.mu.Unlock() - if t.escaped { + if t.escapeErr != nil { return nil, fmt.Errorf("html/template: cannot Clone %q after it has executed", t.Name()) } textClone, err := t.text.Clone() @@ -187,7 +195,7 @@ func (t *Template) Clone() (*Template, error) { return nil, err } ret := &Template{ - false, + nil, textClone, textClone.Tree, &nameSpace{ @@ -197,12 +205,12 @@ func (t *Template) Clone() (*Template, error) { for _, x := range textClone.Templates() { name := x.Name() src := t.set[name] - if src == nil || src.escaped { + if src == nil || src.escapeErr != nil { return nil, fmt.Errorf("html/template: cannot Clone %q after it has executed", t.Name()) } x.Tree = x.Tree.Copy() ret.set[name] = &Template{ - false, + nil, x, x.Tree, ret.nameSpace, @@ -214,7 +222,7 @@ func (t *Template) Clone() (*Template, error) { // New allocates a new HTML template with the given name. func New(name string) *Template { tmpl := &Template{ - false, + nil, template.New(name), nil, &nameSpace{ @@ -237,7 +245,7 @@ func (t *Template) New(name string) *Template { // new is the implementation of New, without the lock. func (t *Template) new(name string) *Template { tmpl := &Template{ - false, + nil, t.text.New(name), nil, t.nameSpace, diff --git a/libgo/go/html/template/transition.go b/libgo/go/html/template/transition.go index 7f30a7a..b486fcd 100644 --- a/libgo/go/html/template/transition.go +++ b/libgo/go/html/template/transition.go @@ -102,7 +102,7 @@ func tTag(c context, s []byte) (context, int) { if i == j { return context{ state: stateError, - err: errorf(ErrBadHTML, 0, "expected space, attr name, or end of tag, but got %q", s[i:]), + err: errorf(ErrBadHTML, nil, 0, "expected space, attr name, or end of tag, but got %q", s[i:]), }, len(s) } switch attrType(string(s[i:j])) { @@ -245,7 +245,7 @@ func tJS(c context, s []byte) (context, int) { default: return context{ state: stateError, - err: errorf(ErrSlashAmbig, 0, "'/' could start a division or regexp: %.32q", s[i:]), + err: errorf(ErrSlashAmbig, nil, 0, "'/' could start a division or regexp: %.32q", s[i:]), }, len(s) } default: @@ -277,7 +277,7 @@ func tJSDelimited(c context, s []byte) (context, int) { if i == len(s) { return context{ state: stateError, - err: errorf(ErrPartialEscape, 0, "unfinished escape sequence in JS string: %q", s), + err: errorf(ErrPartialEscape, nil, 0, "unfinished escape sequence in JS string: %q", s), }, len(s) } case '[': @@ -299,7 +299,7 @@ func tJSDelimited(c context, s []byte) (context, int) { // into charsets is desired. return context{ state: stateError, - err: errorf(ErrPartialCharset, 0, "unfinished JS regexp charset: %q", s), + err: errorf(ErrPartialCharset, nil, 0, "unfinished JS regexp charset: %q", s), }, len(s) } @@ -459,7 +459,7 @@ func tCSSStr(c context, s []byte) (context, int) { if i == len(s) { return context{ state: stateError, - err: errorf(ErrPartialEscape, 0, "unfinished escape sequence in CSS string: %q", s), + err: errorf(ErrPartialEscape, nil, 0, "unfinished escape sequence in CSS string: %q", s), }, len(s) } } else { @@ -489,7 +489,7 @@ func eatAttrName(s []byte, i int) (int, *Error) { // These result in a parse warning in HTML5 and are // indicative of serious problems if seen in an attr // name in a template. - return -1, errorf(ErrBadHTML, 0, "%q in attribute name: %.32q", s[j:j+1], s) + return -1, errorf(ErrBadHTML, nil, 0, "%q in attribute name: %.32q", s[j:j+1], s) default: // No-op. } |