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  }