Home | History | Annotate | Download | only in template
      1 // Copyright 2011 The Go Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style
      3 // license that can be found in the LICENSE file.
      4 
      5 package template
      6 
      7 import (
      8 	"fmt"
      9 	"text/template/parse"
     10 )
     11 
     12 // Error describes a problem encountered during template Escaping.
     13 type Error struct {
     14 	// ErrorCode describes the kind of error.
     15 	ErrorCode ErrorCode
     16 	// Node is the node that caused the problem, if known.
     17 	// If not nil, it overrides Name and Line.
     18 	Node parse.Node
     19 	// Name is the name of the template in which the error was encountered.
     20 	Name string
     21 	// Line is the line number of the error in the template source or 0.
     22 	Line int
     23 	// Description is a human-readable description of the problem.
     24 	Description string
     25 }
     26 
     27 // ErrorCode is a code for a kind of error.
     28 type ErrorCode int
     29 
     30 // We define codes for each error that manifests while escaping templates, but
     31 // escaped templates may also fail at runtime.
     32 //
     33 // Output: "ZgotmplZ"
     34 // Example:
     35 //   <img src="{{.X}}">
     36 //   where {{.X}} evaluates to `javascript:...`
     37 // Discussion:
     38 //   "ZgotmplZ" is a special value that indicates that unsafe content reached a
     39 //   CSS or URL context at runtime. The output of the example will be
     40 //     <img src="#ZgotmplZ">
     41 //   If the data comes from a trusted source, use content types to exempt it
     42 //   from filtering: URL(`javascript:...`).
     43 const (
     44 	// OK indicates the lack of an error.
     45 	OK ErrorCode = iota
     46 
     47 	// ErrAmbigContext: "... appears in an ambiguous URL context"
     48 	// Example:
     49 	//   <a href="
     50 	//      {{if .C}}
     51 	//        /path/
     52 	//      {{else}}
     53 	//        /search?q=
     54 	//      {{end}}
     55 	//      {{.X}}
     56 	//   ">
     57 	// Discussion:
     58 	//   {{.X}} is in an ambiguous URL context since, depending on {{.C}},
     59 	//  it may be either a URL suffix or a query parameter.
     60 	//   Moving {{.X}} into the condition removes the ambiguity:
     61 	//   <a href="{{if .C}}/path/{{.X}}{{else}}/search?q={{.X}}">
     62 	ErrAmbigContext
     63 
     64 	// ErrBadHTML: "expected space, attr name, or end of tag, but got ...",
     65 	//   "... in unquoted attr", "... in attribute name"
     66 	// Example:
     67 	//   <a href = /search?q=foo>
     68 	//   <href=foo>
     69 	//   <form na<e=...>
     70 	//   <option selected<
     71 	// Discussion:
     72 	//   This is often due to a typo in an HTML element, but some runes
     73 	//   are banned in tag names, attribute names, and unquoted attribute
     74 	//   values because they can tickle parser ambiguities.
     75 	//   Quoting all attributes is the best policy.
     76 	ErrBadHTML
     77 
     78 	// ErrBranchEnd: "{{if}} branches end in different contexts"
     79 	// Example:
     80 	//   {{if .C}}<a href="{{end}}{{.X}}
     81 	// Discussion:
     82 	//   Package html/template statically examines each path through an
     83 	//   {{if}}, {{range}}, or {{with}} to escape any following pipelines.
     84 	//   The example is ambiguous since {{.X}} might be an HTML text node,
     85 	//   or a URL prefix in an HTML attribute. The context of {{.X}} is
     86 	//   used to figure out how to escape it, but that context depends on
     87 	//   the run-time value of {{.C}} which is not statically known.
     88 	//
     89 	//   The problem is usually something like missing quotes or angle
     90 	//   brackets, or can be avoided by refactoring to put the two contexts
     91 	//   into different branches of an if, range or with. If the problem
     92 	//   is in a {{range}} over a collection that should never be empty,
     93 	//   adding a dummy {{else}} can help.
     94 	ErrBranchEnd
     95 
     96 	// ErrEndContext: "... ends in a non-text context: ..."
     97 	// Examples:
     98 	//   <div
     99 	//   <div title="no close quote>
    100 	//   <script>f()
    101 	// Discussion:
    102 	//   Executed templates should produce a DocumentFragment of HTML.
    103 	//   Templates that end without closing tags will trigger this error.
    104 	//   Templates that should not be used in an HTML context or that
    105 	//   produce incomplete Fragments should not be executed directly.
    106 	//
    107 	//   {{define "main"}} <script>{{template "helper"}}</script> {{end}}
    108 	//   {{define "helper"}} document.write(' <div title=" ') {{end}}
    109 	//
    110 	//   "helper" does not produce a valid document fragment, so should
    111 	//   not be Executed directly.
    112 	ErrEndContext
    113 
    114 	// ErrNoSuchTemplate: "no such template ..."
    115 	// Examples:
    116 	//   {{define "main"}}<div {{template "attrs"}}>{{end}}
    117 	//   {{define "attrs"}}href="{{.URL}}"{{end}}
    118 	// Discussion:
    119 	//   Package html/template looks through template calls to compute the
    120 	//   context.
    121 	//   Here the {{.URL}} in "attrs" must be treated as a URL when called
    122 	//   from "main", but you will get this error if "attrs" is not defined
    123 	//   when "main" is parsed.
    124 	ErrNoSuchTemplate
    125 
    126 	// ErrOutputContext: "cannot compute output context for template ..."
    127 	// Examples:
    128 	//   {{define "t"}}{{if .T}}{{template "t" .T}}{{end}}{{.H}}",{{end}}
    129 	// Discussion:
    130 	//   A recursive template does not end in the same context in which it
    131 	//   starts, and a reliable output context cannot be computed.
    132 	//   Look for typos in the named template.
    133 	//   If the template should not be called in the named start context,
    134 	//   look for calls to that template in unexpected contexts.
    135 	//   Maybe refactor recursive templates to not be recursive.
    136 	ErrOutputContext
    137 
    138 	// ErrPartialCharset: "unfinished JS regexp charset in ..."
    139 	// Example:
    140 	//     <script>var pattern = /foo[{{.Chars}}]/</script>
    141 	// Discussion:
    142 	//   Package html/template does not support interpolation into regular
    143 	//   expression literal character sets.
    144 	ErrPartialCharset
    145 
    146 	// ErrPartialEscape: "unfinished escape sequence in ..."
    147 	// Example:
    148 	//   <script>alert("\{{.X}}")</script>
    149 	// Discussion:
    150 	//   Package html/template does not support actions following a
    151 	//   backslash.
    152 	//   This is usually an error and there are better solutions; for
    153 	//   example
    154 	//     <script>alert("{{.X}}")</script>
    155 	//   should work, and if {{.X}} is a partial escape sequence such as
    156 	//   "xA0", mark the whole sequence as safe content: JSStr(`\xA0`)
    157 	ErrPartialEscape
    158 
    159 	// ErrRangeLoopReentry: "on range loop re-entry: ..."
    160 	// Example:
    161 	//   <script>var x = [{{range .}}'{{.}},{{end}}]</script>
    162 	// Discussion:
    163 	//   If an iteration through a range would cause it to end in a
    164 	//   different context than an earlier pass, there is no single context.
    165 	//   In the example, there is missing a quote, so it is not clear
    166 	//   whether {{.}} is meant to be inside a JS string or in a JS value
    167 	//   context.  The second iteration would produce something like
    168 	//
    169 	//     <script>var x = ['firstValue,'secondValue]</script>
    170 	ErrRangeLoopReentry
    171 
    172 	// ErrSlashAmbig: '/' could start a division or regexp.
    173 	// Example:
    174 	//   <script>
    175 	//     {{if .C}}var x = 1{{end}}
    176 	//     /-{{.N}}/i.test(x) ? doThis : doThat();
    177 	//   </script>
    178 	// Discussion:
    179 	//   The example above could produce `var x = 1/-2/i.test(s)...`
    180 	//   in which the first '/' is a mathematical division operator or it
    181 	//   could produce `/-2/i.test(s)` in which the first '/' starts a
    182 	//   regexp literal.
    183 	//   Look for missing semicolons inside branches, and maybe add
    184 	//   parentheses to make it clear which interpretation you intend.
    185 	ErrSlashAmbig
    186 )
    187 
    188 func (e *Error) Error() string {
    189 	switch {
    190 	case e.Node != nil:
    191 		loc, _ := (*parse.Tree)(nil).ErrorContext(e.Node)
    192 		return fmt.Sprintf("html/template:%s: %s", loc, e.Description)
    193 	case e.Line != 0:
    194 		return fmt.Sprintf("html/template:%s:%d: %s", e.Name, e.Line, e.Description)
    195 	case e.Name != "":
    196 		return fmt.Sprintf("html/template:%s: %s", e.Name, e.Description)
    197 	}
    198 	return "html/template: " + e.Description
    199 }
    200 
    201 // errorf creates an error given a format string f and args.
    202 // The template Name still needs to be supplied.
    203 func errorf(k ErrorCode, node parse.Node, line int, f string, args ...interface{}) *Error {
    204 	return &Error{k, node, "", line, fmt.Sprintf(f, args...)}
    205 }
    206