Commit be33e203 authored by Rob Pike's avatar Rob Pike

text/template: add ExecError type and return it from Execute on error

Useful to discriminate evaluation errors from write errors.

Fixes #11898.

Change-Id: I907d339a3820e887872d78e0e2d8fd011451fd19
Reviewed-on: https://go-review.googlesource.com/13957Reviewed-by: default avatarAndrew Gerrand <adg@golang.org>
parent d497eeb0
...@@ -78,7 +78,23 @@ func doublePercent(str string) string { ...@@ -78,7 +78,23 @@ func doublePercent(str string) string {
return str return str
} }
// errorf formats the error and terminates processing. // TODO: It would be nice if ExecError was more broken down, but
// the way ErrorContext embeds the template name makes the
// processing too clumsy.
// ExecError is the custom error type returned when Execute has an
// error evaluating its template. (If a write error occurs, the actual
// error is returned; it will not be of type ExecError.)
type ExecError struct {
Name string // Name of template.
Err error // Pre-formatted error.
}
func (e ExecError) Error() string {
return e.Err.Error()
}
// errorf records an ExecError and terminates processing.
func (s *state) errorf(format string, args ...interface{}) { func (s *state) errorf(format string, args ...interface{}) {
name := doublePercent(s.tmpl.Name()) name := doublePercent(s.tmpl.Name())
if s.node == nil { if s.node == nil {
...@@ -87,7 +103,24 @@ func (s *state) errorf(format string, args ...interface{}) { ...@@ -87,7 +103,24 @@ func (s *state) errorf(format string, args ...interface{}) {
location, context := s.tmpl.ErrorContext(s.node) location, context := s.tmpl.ErrorContext(s.node)
format = fmt.Sprintf("template: %s: executing %q at <%s>: %s", location, name, doublePercent(context), format) format = fmt.Sprintf("template: %s: executing %q at <%s>: %s", location, name, doublePercent(context), format)
} }
panic(fmt.Errorf(format, args...)) panic(ExecError{
Name: s.tmpl.Name(),
Err: fmt.Errorf(format, args...),
})
}
// writeError is the wrapper type used internally when Execute has an
// error writing to its output. We strip the wrapper in errRecover.
// Note that this is not an implementation of error, so it cannot escape
// from the package as an error value.
type writeError struct {
Err error // Original error.
}
func (s *state) writeError(err error) {
panic(writeError{
Err: err,
})
} }
// errRecover is the handler that turns panics into returns from the top // errRecover is the handler that turns panics into returns from the top
...@@ -98,7 +131,11 @@ func errRecover(errp *error) { ...@@ -98,7 +131,11 @@ func errRecover(errp *error) {
switch err := e.(type) { switch err := e.(type) {
case runtime.Error: case runtime.Error:
panic(e) panic(e)
case error: case writeError:
*errp = err.Err // Strip the wrapper.
case ExecError:
*errp = err // Keep the wrapper.
case error: // TODO: This should never happen, but it does. Understand and/or fix.
*errp = err *errp = err
default: default:
panic(e) panic(e)
...@@ -193,7 +230,7 @@ func (s *state) walk(dot reflect.Value, node parse.Node) { ...@@ -193,7 +230,7 @@ func (s *state) walk(dot reflect.Value, node parse.Node) {
s.walkTemplate(dot, node) s.walkTemplate(dot, node)
case *parse.TextNode: case *parse.TextNode:
if _, err := s.wr.Write(node.Text); err != nil { if _, err := s.wr.Write(node.Text); err != nil {
s.errorf("%s", err) s.writeError(err)
} }
case *parse.WithNode: case *parse.WithNode:
s.walkIfOrWith(parse.NodeWith, dot, node.Pipe, node.List, node.ElseList) s.walkIfOrWith(parse.NodeWith, dot, node.Pipe, node.List, node.ElseList)
...@@ -811,7 +848,10 @@ func (s *state) printValue(n parse.Node, v reflect.Value) { ...@@ -811,7 +848,10 @@ func (s *state) printValue(n parse.Node, v reflect.Value) {
if !ok { if !ok {
s.errorf("can't print %s of type %s", n, v.Type()) s.errorf("can't print %s of type %s", n, v.Type())
} }
fmt.Fprint(s.wr, iface) _, err := fmt.Fprint(s.wr, iface)
if err != nil {
s.writeError(err)
}
} }
// printableValue returns the, possibly indirected, interface value inside v that // printableValue returns the, possibly indirected, interface value inside v that
......
...@@ -9,6 +9,7 @@ import ( ...@@ -9,6 +9,7 @@ import (
"errors" "errors"
"flag" "flag"
"fmt" "fmt"
"io/ioutil"
"reflect" "reflect"
"strings" "strings"
"testing" "testing"
...@@ -1141,3 +1142,45 @@ func TestUnterminatedStringError(t *testing.T) { ...@@ -1141,3 +1142,45 @@ func TestUnterminatedStringError(t *testing.T) {
t.Fatalf("unexpected error: %s", str) t.Fatalf("unexpected error: %s", str)
} }
} }
const alwaysErrorText = "always be failing"
var alwaysError = errors.New(alwaysErrorText)
type ErrorWriter int
func (e ErrorWriter) Write(p []byte) (int, error) {
return 0, alwaysError
}
func TestExecuteGivesExecError(t *testing.T) {
// First, a non-execution error shouldn't be an ExecError.
tmpl, err := New("X").Parse("hello")
if err != nil {
t.Fatal(err)
}
err = tmpl.Execute(ErrorWriter(0), 0)
if err == nil {
t.Fatal("expected error; got none")
}
if err.Error() != alwaysErrorText {
t.Errorf("expected %q error; got %q", alwaysErrorText, err)
}
// This one should be an ExecError.
tmpl, err = New("X").Parse("hello, {{.X.Y}}")
if err != nil {
t.Fatal(err)
}
err = tmpl.Execute(ioutil.Discard, 0)
if err == nil {
t.Fatal("expected error; got none")
}
eerr, ok := err.(ExecError)
if !ok {
t.Fatalf("did not expect ExecError %s", eerr)
}
expect := "field X in type int"
if !strings.Contains(err.Error(), expect) {
t.Errorf("expected %q; got %q", expect, err)
}
}
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