github.com/kovansky/hugo@v0.92.3-0.20220224232819-63076e4ff19f/tpl/internal/go_templates/htmltemplate/multi_test.go (about) 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 // Tests for multiple-template execution, copied from text/template. 6 7 // +build go1.13,!windows 8 9 package template 10 11 import ( 12 "archive/zip" 13 "bytes" 14 "os" 15 "testing" 16 17 "github.com/gohugoio/hugo/tpl/internal/go_templates/texttemplate/parse" 18 ) 19 20 var multiExecTests = []execTest{ 21 {"empty", "", "", nil, true}, 22 {"text", "some text", "some text", nil, true}, 23 {"invoke x", `{{template "x" .SI}}`, "TEXT", tVal, true}, 24 {"invoke x no args", `{{template "x"}}`, "TEXT", tVal, true}, 25 {"invoke dot int", `{{template "dot" .I}}`, "17", tVal, true}, 26 {"invoke dot []int", `{{template "dot" .SI}}`, "[3 4 5]", tVal, true}, 27 {"invoke dotV", `{{template "dotV" .U}}`, "v", tVal, true}, 28 {"invoke nested int", `{{template "nested" .I}}`, "17", tVal, true}, 29 {"variable declared by template", `{{template "nested" $x:=.SI}},{{index $x 1}}`, "[3 4 5],4", tVal, true}, 30 31 // User-defined function: test argument evaluator. 32 {"testFunc literal", `{{oneArg "joe"}}`, "oneArg=joe", tVal, true}, 33 {"testFunc .", `{{oneArg .}}`, "oneArg=joe", "joe", true}, 34 } 35 36 // These strings are also in testdata/*. 37 const multiText1 = ` 38 {{define "x"}}TEXT{{end}} 39 {{define "dotV"}}{{.V}}{{end}} 40 ` 41 42 const multiText2 = ` 43 {{define "dot"}}{{.}}{{end}} 44 {{define "nested"}}{{template "dot" .}}{{end}} 45 ` 46 47 func TestMultiExecute(t *testing.T) { 48 // Declare a couple of templates first. 49 template, err := New("root").Parse(multiText1) 50 if err != nil { 51 t.Fatalf("parse error for 1: %s", err) 52 } 53 _, err = template.Parse(multiText2) 54 if err != nil { 55 t.Fatalf("parse error for 2: %s", err) 56 } 57 testExecute(multiExecTests, template, t) 58 } 59 60 func TestParseFiles(t *testing.T) { 61 _, err := ParseFiles("DOES NOT EXIST") 62 if err == nil { 63 t.Error("expected error for non-existent file; got none") 64 } 65 template := New("root") 66 _, err = template.ParseFiles("testdata/file1.tmpl", "testdata/file2.tmpl") 67 if err != nil { 68 t.Fatalf("error parsing files: %v", err) 69 } 70 testExecute(multiExecTests, template, t) 71 } 72 73 func TestParseGlob(t *testing.T) { 74 _, err := ParseGlob("DOES NOT EXIST") 75 if err == nil { 76 t.Error("expected error for non-existent file; got none") 77 } 78 _, err = New("error").ParseGlob("[x") 79 if err == nil { 80 t.Error("expected error for bad pattern; got none") 81 } 82 template := New("root") 83 _, err = template.ParseGlob("testdata/file*.tmpl") 84 if err != nil { 85 t.Fatalf("error parsing files: %v", err) 86 } 87 testExecute(multiExecTests, template, t) 88 } 89 90 func TestParseFS(t *testing.T) { 91 fs := os.DirFS("testdata") 92 93 { 94 _, err := ParseFS(fs, "DOES NOT EXIST") 95 if err == nil { 96 t.Error("expected error for non-existent file; got none") 97 } 98 } 99 100 { 101 template := New("root") 102 _, err := template.ParseFS(fs, "file1.tmpl", "file2.tmpl") 103 if err != nil { 104 t.Fatalf("error parsing files: %v", err) 105 } 106 testExecute(multiExecTests, template, t) 107 } 108 109 { 110 template := New("root") 111 _, err := template.ParseFS(fs, "file*.tmpl") 112 if err != nil { 113 t.Fatalf("error parsing files: %v", err) 114 } 115 testExecute(multiExecTests, template, t) 116 } 117 } 118 119 // In these tests, actual content (not just template definitions) comes from the parsed files. 120 121 var templateFileExecTests = []execTest{ 122 {"test", `{{template "tmpl1.tmpl"}}{{template "tmpl2.tmpl"}}`, "template1\n\ny\ntemplate2\n\nx\n", 0, true}, 123 } 124 125 func TestParseFilesWithData(t *testing.T) { 126 template, err := New("root").ParseFiles("testdata/tmpl1.tmpl", "testdata/tmpl2.tmpl") 127 if err != nil { 128 t.Fatalf("error parsing files: %v", err) 129 } 130 testExecute(templateFileExecTests, template, t) 131 } 132 133 func TestParseGlobWithData(t *testing.T) { 134 template, err := New("root").ParseGlob("testdata/tmpl*.tmpl") 135 if err != nil { 136 t.Fatalf("error parsing files: %v", err) 137 } 138 testExecute(templateFileExecTests, template, t) 139 } 140 141 func TestParseZipFS(t *testing.T) { 142 z, err := zip.OpenReader("testdata/fs.zip") 143 if err != nil { 144 t.Fatalf("error parsing zip: %v", err) 145 } 146 template, err := New("root").ParseFS(z, "tmpl*.tmpl") 147 if err != nil { 148 t.Fatalf("error parsing files: %v", err) 149 } 150 testExecute(templateFileExecTests, template, t) 151 } 152 153 const ( 154 cloneText1 = `{{define "a"}}{{template "b"}}{{template "c"}}{{end}}` 155 cloneText2 = `{{define "b"}}b{{end}}` 156 cloneText3 = `{{define "c"}}root{{end}}` 157 cloneText4 = `{{define "c"}}clone{{end}}` 158 ) 159 160 // Issue 7032 161 func TestAddParseTreeToUnparsedTemplate(t *testing.T) { 162 master := "{{define \"master\"}}{{end}}" 163 tmpl := New("master") 164 tree, err := parse.Parse("master", master, "", "", nil) 165 if err != nil { 166 t.Fatalf("unexpected parse err: %v", err) 167 } 168 masterTree := tree["master"] 169 tmpl.AddParseTree("master", masterTree) // used to panic 170 } 171 172 func TestRedefinition(t *testing.T) { 173 var tmpl *Template 174 var err error 175 if tmpl, err = New("tmpl1").Parse(`{{define "test"}}foo{{end}}`); err != nil { 176 t.Fatalf("parse 1: %v", err) 177 } 178 if _, err = tmpl.Parse(`{{define "test"}}bar{{end}}`); err != nil { 179 t.Fatalf("got error %v, expected nil", err) 180 } 181 if _, err = tmpl.New("tmpl2").Parse(`{{define "test"}}bar{{end}}`); err != nil { 182 t.Fatalf("got error %v, expected nil", err) 183 } 184 } 185 186 // Issue 10879 187 func TestEmptyTemplateCloneCrash(t *testing.T) { 188 t1 := New("base") 189 t1.Clone() // used to panic 190 } 191 192 // Issue 10910, 10926 193 func TestTemplateLookUp(t *testing.T) { 194 t.Skip("broken on html/template") // TODO 195 t1 := New("foo") 196 if t1.Lookup("foo") != nil { 197 t.Error("Lookup returned non-nil value for undefined template foo") 198 } 199 t1.New("bar") 200 if t1.Lookup("bar") != nil { 201 t.Error("Lookup returned non-nil value for undefined template bar") 202 } 203 t1.Parse(`{{define "foo"}}test{{end}}`) 204 if t1.Lookup("foo") == nil { 205 t.Error("Lookup returned nil value for defined template") 206 } 207 } 208 209 func TestParse(t *testing.T) { 210 // In multiple calls to Parse with the same receiver template, only one call 211 // can contain text other than space, comments, and template definitions 212 t1 := New("test") 213 if _, err := t1.Parse(`{{define "test"}}{{end}}`); err != nil { 214 t.Fatalf("parsing test: %s", err) 215 } 216 if _, err := t1.Parse(`{{define "test"}}{{/* this is a comment */}}{{end}}`); err != nil { 217 t.Fatalf("parsing test: %s", err) 218 } 219 if _, err := t1.Parse(`{{define "test"}}foo{{end}}`); err != nil { 220 t.Fatalf("parsing test: %s", err) 221 } 222 } 223 224 func TestEmptyTemplate(t *testing.T) { 225 cases := []struct { 226 defn []string 227 in string 228 want string 229 }{ 230 {[]string{"x", "y"}, "", "y"}, 231 {[]string{""}, "once", ""}, 232 {[]string{"", ""}, "twice", ""}, 233 {[]string{"{{.}}", "{{.}}"}, "twice", "twice"}, 234 {[]string{"{{/* a comment */}}", "{{/* a comment */}}"}, "comment", ""}, 235 {[]string{"{{.}}", ""}, "twice", "twice"}, // TODO: should want "" not "twice" 236 } 237 238 for i, c := range cases { 239 root := New("root") 240 241 var ( 242 m *Template 243 err error 244 ) 245 for _, d := range c.defn { 246 m, err = root.New(c.in).Parse(d) 247 if err != nil { 248 t.Fatal(err) 249 } 250 } 251 buf := &bytes.Buffer{} 252 if err := m.Execute(buf, c.in); err != nil { 253 t.Error(i, err) 254 continue 255 } 256 if buf.String() != c.want { 257 t.Errorf("expected string %q: got %q", c.want, buf.String()) 258 } 259 } 260 } 261 262 // Issue 19249 was a regression in 1.8 caused by the handling of empty 263 // templates added in that release, which got different answers depending 264 // on the order templates appeared in the internal map. 265 func TestIssue19294(t *testing.T) { 266 // The empty block in "xhtml" should be replaced during execution 267 // by the contents of "stylesheet", but if the internal map associating 268 // names with templates is built in the wrong order, the empty block 269 // looks non-empty and this doesn't happen. 270 var inlined = map[string]string{ 271 "stylesheet": `{{define "stylesheet"}}stylesheet{{end}}`, 272 "xhtml": `{{block "stylesheet" .}}{{end}}`, 273 } 274 all := []string{"stylesheet", "xhtml"} 275 for i := 0; i < 100; i++ { 276 res, err := New("title.xhtml").Parse(`{{template "xhtml" .}}`) 277 if err != nil { 278 t.Fatal(err) 279 } 280 for _, name := range all { 281 _, err := res.New(name).Parse(inlined[name]) 282 if err != nil { 283 t.Fatal(err) 284 } 285 } 286 var buf bytes.Buffer 287 res.Execute(&buf, 0) 288 if buf.String() != "stylesheet" { 289 t.Fatalf("iteration %d: got %q; expected %q", i, buf.String(), "stylesheet") 290 } 291 } 292 }