mirror of
https://github.com/golang/go
synced 2024-11-11 23:10:23 -07:00
html/template: add tests from text/template
Copy and adapt tests from text/template, to exercise more of html/template's copy. Various differences in behavior are flagged with NOTE comments or t.Skip and documented in #40075. Many of them are probably bugs. One clarifying test case added to both text/template and html/template. No changes to the package itself. Change-Id: Ifefad83d647db846040d24c2741a0244b00ade82 Reviewed-on: https://go-review.googlesource.com/c/go/+/241084 Run-TryBot: Russ Cox <rsc@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Rob Pike <r@golang.org>
This commit is contained in:
parent
e88ea87e7b
commit
efbe47b162
@ -15,7 +15,7 @@ import (
|
|||||||
"text/template/parse"
|
"text/template/parse"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestAddParseTree(t *testing.T) {
|
func TestAddParseTreeHTML(t *testing.T) {
|
||||||
root := Must(New("root").Parse(`{{define "a"}} {{.}} {{template "b"}} {{.}} "></a>{{end}}`))
|
root := Must(New("root").Parse(`{{define "a"}} {{.}} {{template "b"}} {{.}} "></a>{{end}}`))
|
||||||
tree, err := parse.Parse("t", `{{define "b"}}<a href="{{end}}`, "", "", nil, nil)
|
tree, err := parse.Parse("t", `{{define "b"}}<a href="{{end}}`, "", "", nil, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -401,11 +401,11 @@ func TestTypedContent(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Test that we print using the String method. Was issue 3073.
|
// Test that we print using the String method. Was issue 3073.
|
||||||
type stringer struct {
|
type myStringer struct {
|
||||||
v int
|
v int
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *stringer) String() string {
|
func (s *myStringer) String() string {
|
||||||
return fmt.Sprintf("string=%d", s.v)
|
return fmt.Sprintf("string=%d", s.v)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -418,7 +418,7 @@ func (s *errorer) Error() string {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestStringer(t *testing.T) {
|
func TestStringer(t *testing.T) {
|
||||||
s := &stringer{3}
|
s := &myStringer{3}
|
||||||
b := new(bytes.Buffer)
|
b := new(bytes.Buffer)
|
||||||
tmpl := Must(New("x").Parse("{{.}}"))
|
tmpl := Must(New("x").Parse("{{.}}"))
|
||||||
if err := tmpl.Execute(b, s); err != nil {
|
if err := tmpl.Execute(b, s); err != nil {
|
||||||
|
@ -1821,7 +1821,7 @@ func TestIndirectPrint(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// This is a test for issue 3272.
|
// This is a test for issue 3272.
|
||||||
func TestEmptyTemplate(t *testing.T) {
|
func TestEmptyTemplateHTML(t *testing.T) {
|
||||||
page := Must(New("page").ParseFiles(os.DevNull))
|
page := Must(New("page").ParseFiles(os.DevNull))
|
||||||
if err := page.ExecuteTemplate(os.Stdout, "page", "nothing"); err == nil {
|
if err := page.ExecuteTemplate(os.Stdout, "page", "nothing"); err == nil {
|
||||||
t.Fatal("expected error")
|
t.Fatal("expected error")
|
||||||
|
1708
src/html/template/exec_test.go
Normal file
1708
src/html/template/exec_test.go
Normal file
File diff suppressed because it is too large
Load Diff
246
src/html/template/multi_test.go
Normal file
246
src/html/template/multi_test.go
Normal file
@ -0,0 +1,246 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
// Tests for multiple-template execution, copied from text/template.
|
||||||
|
|
||||||
|
package template
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"testing"
|
||||||
|
"text/template/parse"
|
||||||
|
)
|
||||||
|
|
||||||
|
var multiExecTests = []execTest{
|
||||||
|
{"empty", "", "", nil, true},
|
||||||
|
{"text", "some text", "some text", nil, true},
|
||||||
|
{"invoke x", `{{template "x" .SI}}`, "TEXT", tVal, true},
|
||||||
|
{"invoke x no args", `{{template "x"}}`, "TEXT", tVal, true},
|
||||||
|
{"invoke dot int", `{{template "dot" .I}}`, "17", tVal, true},
|
||||||
|
{"invoke dot []int", `{{template "dot" .SI}}`, "[3 4 5]", tVal, true},
|
||||||
|
{"invoke dotV", `{{template "dotV" .U}}`, "v", tVal, true},
|
||||||
|
{"invoke nested int", `{{template "nested" .I}}`, "17", tVal, true},
|
||||||
|
{"variable declared by template", `{{template "nested" $x:=.SI}},{{index $x 1}}`, "[3 4 5],4", tVal, true},
|
||||||
|
|
||||||
|
// User-defined function: test argument evaluator.
|
||||||
|
{"testFunc literal", `{{oneArg "joe"}}`, "oneArg=joe", tVal, true},
|
||||||
|
{"testFunc .", `{{oneArg .}}`, "oneArg=joe", "joe", true},
|
||||||
|
}
|
||||||
|
|
||||||
|
// These strings are also in testdata/*.
|
||||||
|
const multiText1 = `
|
||||||
|
{{define "x"}}TEXT{{end}}
|
||||||
|
{{define "dotV"}}{{.V}}{{end}}
|
||||||
|
`
|
||||||
|
|
||||||
|
const multiText2 = `
|
||||||
|
{{define "dot"}}{{.}}{{end}}
|
||||||
|
{{define "nested"}}{{template "dot" .}}{{end}}
|
||||||
|
`
|
||||||
|
|
||||||
|
func TestMultiExecute(t *testing.T) {
|
||||||
|
// Declare a couple of templates first.
|
||||||
|
template, err := New("root").Parse(multiText1)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("parse error for 1: %s", err)
|
||||||
|
}
|
||||||
|
_, err = template.Parse(multiText2)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("parse error for 2: %s", err)
|
||||||
|
}
|
||||||
|
testExecute(multiExecTests, template, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseFiles(t *testing.T) {
|
||||||
|
_, err := ParseFiles("DOES NOT EXIST")
|
||||||
|
if err == nil {
|
||||||
|
t.Error("expected error for non-existent file; got none")
|
||||||
|
}
|
||||||
|
template := New("root")
|
||||||
|
_, err = template.ParseFiles("testdata/file1.tmpl", "testdata/file2.tmpl")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error parsing files: %v", err)
|
||||||
|
}
|
||||||
|
testExecute(multiExecTests, template, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseGlob(t *testing.T) {
|
||||||
|
_, err := ParseGlob("DOES NOT EXIST")
|
||||||
|
if err == nil {
|
||||||
|
t.Error("expected error for non-existent file; got none")
|
||||||
|
}
|
||||||
|
_, err = New("error").ParseGlob("[x")
|
||||||
|
if err == nil {
|
||||||
|
t.Error("expected error for bad pattern; got none")
|
||||||
|
}
|
||||||
|
template := New("root")
|
||||||
|
_, err = template.ParseGlob("testdata/file*.tmpl")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error parsing files: %v", err)
|
||||||
|
}
|
||||||
|
testExecute(multiExecTests, template, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
// In these tests, actual content (not just template definitions) comes from the parsed files.
|
||||||
|
|
||||||
|
var templateFileExecTests = []execTest{
|
||||||
|
{"test", `{{template "tmpl1.tmpl"}}{{template "tmpl2.tmpl"}}`, "template1\n\ny\ntemplate2\n\nx\n", 0, true},
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseFilesWithData(t *testing.T) {
|
||||||
|
template, err := New("root").ParseFiles("testdata/tmpl1.tmpl", "testdata/tmpl2.tmpl")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error parsing files: %v", err)
|
||||||
|
}
|
||||||
|
testExecute(templateFileExecTests, template, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParseGlobWithData(t *testing.T) {
|
||||||
|
template, err := New("root").ParseGlob("testdata/tmpl*.tmpl")
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("error parsing files: %v", err)
|
||||||
|
}
|
||||||
|
testExecute(templateFileExecTests, template, t)
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
cloneText1 = `{{define "a"}}{{template "b"}}{{template "c"}}{{end}}`
|
||||||
|
cloneText2 = `{{define "b"}}b{{end}}`
|
||||||
|
cloneText3 = `{{define "c"}}root{{end}}`
|
||||||
|
cloneText4 = `{{define "c"}}clone{{end}}`
|
||||||
|
)
|
||||||
|
|
||||||
|
// Issue 7032
|
||||||
|
func TestAddParseTreeToUnparsedTemplate(t *testing.T) {
|
||||||
|
master := "{{define \"master\"}}{{end}}"
|
||||||
|
tmpl := New("master")
|
||||||
|
tree, err := parse.Parse("master", master, "", "", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("unexpected parse err: %v", err)
|
||||||
|
}
|
||||||
|
masterTree := tree["master"]
|
||||||
|
tmpl.AddParseTree("master", masterTree) // used to panic
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRedefinition(t *testing.T) {
|
||||||
|
var tmpl *Template
|
||||||
|
var err error
|
||||||
|
if tmpl, err = New("tmpl1").Parse(`{{define "test"}}foo{{end}}`); err != nil {
|
||||||
|
t.Fatalf("parse 1: %v", err)
|
||||||
|
}
|
||||||
|
if _, err = tmpl.Parse(`{{define "test"}}bar{{end}}`); err != nil {
|
||||||
|
t.Fatalf("got error %v, expected nil", err)
|
||||||
|
}
|
||||||
|
if _, err = tmpl.New("tmpl2").Parse(`{{define "test"}}bar{{end}}`); err != nil {
|
||||||
|
t.Fatalf("got error %v, expected nil", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Issue 10879
|
||||||
|
func TestEmptyTemplateCloneCrash(t *testing.T) {
|
||||||
|
t1 := New("base")
|
||||||
|
t1.Clone() // used to panic
|
||||||
|
}
|
||||||
|
|
||||||
|
// Issue 10910, 10926
|
||||||
|
func TestTemplateLookUp(t *testing.T) {
|
||||||
|
t.Skip("broken on html/template") // TODO
|
||||||
|
t1 := New("foo")
|
||||||
|
if t1.Lookup("foo") != nil {
|
||||||
|
t.Error("Lookup returned non-nil value for undefined template foo")
|
||||||
|
}
|
||||||
|
t1.New("bar")
|
||||||
|
if t1.Lookup("bar") != nil {
|
||||||
|
t.Error("Lookup returned non-nil value for undefined template bar")
|
||||||
|
}
|
||||||
|
t1.Parse(`{{define "foo"}}test{{end}}`)
|
||||||
|
if t1.Lookup("foo") == nil {
|
||||||
|
t.Error("Lookup returned nil value for defined template")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestParse(t *testing.T) {
|
||||||
|
// In multiple calls to Parse with the same receiver template, only one call
|
||||||
|
// can contain text other than space, comments, and template definitions
|
||||||
|
t1 := New("test")
|
||||||
|
if _, err := t1.Parse(`{{define "test"}}{{end}}`); err != nil {
|
||||||
|
t.Fatalf("parsing test: %s", err)
|
||||||
|
}
|
||||||
|
if _, err := t1.Parse(`{{define "test"}}{{/* this is a comment */}}{{end}}`); err != nil {
|
||||||
|
t.Fatalf("parsing test: %s", err)
|
||||||
|
}
|
||||||
|
if _, err := t1.Parse(`{{define "test"}}foo{{end}}`); err != nil {
|
||||||
|
t.Fatalf("parsing test: %s", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestEmptyTemplate(t *testing.T) {
|
||||||
|
cases := []struct {
|
||||||
|
defn []string
|
||||||
|
in string
|
||||||
|
want string
|
||||||
|
}{
|
||||||
|
{[]string{"x", "y"}, "", "y"},
|
||||||
|
{[]string{""}, "once", ""},
|
||||||
|
{[]string{"", ""}, "twice", ""},
|
||||||
|
{[]string{"{{.}}", "{{.}}"}, "twice", "twice"},
|
||||||
|
{[]string{"{{/* a comment */}}", "{{/* a comment */}}"}, "comment", ""},
|
||||||
|
{[]string{"{{.}}", ""}, "twice", "twice"}, // TODO: should want "" not "twice"
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, c := range cases {
|
||||||
|
root := New("root")
|
||||||
|
|
||||||
|
var (
|
||||||
|
m *Template
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
for _, d := range c.defn {
|
||||||
|
m, err = root.New(c.in).Parse(d)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
if err := m.Execute(buf, c.in); err != nil {
|
||||||
|
t.Error(i, err)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if buf.String() != c.want {
|
||||||
|
t.Errorf("expected string %q: got %q", c.want, buf.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Issue 19249 was a regression in 1.8 caused by the handling of empty
|
||||||
|
// templates added in that release, which got different answers depending
|
||||||
|
// on the order templates appeared in the internal map.
|
||||||
|
func TestIssue19294(t *testing.T) {
|
||||||
|
// The empty block in "xhtml" should be replaced during execution
|
||||||
|
// by the contents of "stylesheet", but if the internal map associating
|
||||||
|
// names with templates is built in the wrong order, the empty block
|
||||||
|
// looks non-empty and this doesn't happen.
|
||||||
|
var inlined = map[string]string{
|
||||||
|
"stylesheet": `{{define "stylesheet"}}stylesheet{{end}}`,
|
||||||
|
"xhtml": `{{block "stylesheet" .}}{{end}}`,
|
||||||
|
}
|
||||||
|
all := []string{"stylesheet", "xhtml"}
|
||||||
|
for i := 0; i < 100; i++ {
|
||||||
|
res, err := New("title.xhtml").Parse(`{{template "xhtml" .}}`)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
for _, name := range all {
|
||||||
|
_, err := res.New(name).Parse(inlined[name])
|
||||||
|
if err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var buf bytes.Buffer
|
||||||
|
res.Execute(&buf, 0)
|
||||||
|
if buf.String() != "stylesheet" {
|
||||||
|
t.Fatalf("iteration %d: got %q; expected %q", i, buf.String(), "stylesheet")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
2
src/html/template/testdata/file1.tmpl
vendored
Normal file
2
src/html/template/testdata/file1.tmpl
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
{{define "x"}}TEXT{{end}}
|
||||||
|
{{define "dotV"}}{{.V}}{{end}}
|
2
src/html/template/testdata/file2.tmpl
vendored
Normal file
2
src/html/template/testdata/file2.tmpl
vendored
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
{{define "dot"}}{{.}}{{end}}
|
||||||
|
{{define "nested"}}{{template "dot" .}}{{end}}
|
3
src/html/template/testdata/tmpl1.tmpl
vendored
Normal file
3
src/html/template/testdata/tmpl1.tmpl
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
template1
|
||||||
|
{{define "x"}}x{{end}}
|
||||||
|
{{template "y"}}
|
3
src/html/template/testdata/tmpl2.tmpl
vendored
Normal file
3
src/html/template/testdata/tmpl2.tmpl
vendored
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
template2
|
||||||
|
{{define "y"}}y{{end}}
|
||||||
|
{{template "x"}}
|
@ -359,6 +359,7 @@ func TestEmptyTemplate(t *testing.T) {
|
|||||||
in string
|
in string
|
||||||
want string
|
want string
|
||||||
}{
|
}{
|
||||||
|
{[]string{"x", "y"}, "", "y"},
|
||||||
{[]string{""}, "once", ""},
|
{[]string{""}, "once", ""},
|
||||||
{[]string{"", ""}, "twice", ""},
|
{[]string{"", ""}, "twice", ""},
|
||||||
{[]string{"{{.}}", "{{.}}"}, "twice", "twice"},
|
{[]string{"{{.}}", "{{.}}"}, "twice", "twice"},
|
||||||
|
Loading…
Reference in New Issue
Block a user