2010-12-03 05:34:57 +01:00
|
|
|
// Copyright 2009 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.
|
|
|
|
|
|
|
|
package template
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"container/vector"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"json"
|
|
|
|
"os"
|
2011-01-21 19:19:03 +01:00
|
|
|
"strings"
|
2010-12-03 05:34:57 +01:00
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Test struct {
|
|
|
|
in, out, err string
|
|
|
|
}
|
|
|
|
|
|
|
|
type T struct {
|
2011-01-21 19:19:03 +01:00
|
|
|
Item string
|
|
|
|
Value string
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type U struct {
|
2011-01-21 19:19:03 +01:00
|
|
|
Mp map[string]int
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
type S struct {
|
2011-01-21 19:19:03 +01:00
|
|
|
Header string
|
2011-03-17 00:05:44 +01:00
|
|
|
HeaderPtr *string
|
2011-01-21 19:19:03 +01:00
|
|
|
Integer int
|
2011-03-17 00:05:44 +01:00
|
|
|
IntegerPtr *int
|
|
|
|
NilPtr *int
|
2011-01-21 19:19:03 +01:00
|
|
|
InnerT T
|
|
|
|
InnerPointerT *T
|
|
|
|
Data []T
|
|
|
|
Pdata []*T
|
|
|
|
Empty []*T
|
|
|
|
Emptystring string
|
|
|
|
Null []*T
|
|
|
|
Vec *vector.Vector
|
|
|
|
True bool
|
|
|
|
False bool
|
|
|
|
Mp map[string]string
|
|
|
|
JSON interface{}
|
|
|
|
Innermap U
|
|
|
|
Stringmap map[string]string
|
2011-03-17 00:05:44 +01:00
|
|
|
Ptrmap map[string]*string
|
2011-01-21 19:19:03 +01:00
|
|
|
Iface interface{}
|
|
|
|
Ifaceptr interface{}
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
func (s *S) PointerMethod() string { return "ptrmethod!" }
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
func (s S) ValueMethod() string { return "valmethod!" }
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
var t1 = T{"ItemNumber1", "ValueNumber1"}
|
|
|
|
var t2 = T{"ItemNumber2", "ValueNumber2"}
|
|
|
|
|
|
|
|
func uppercase(v interface{}) string {
|
|
|
|
s := v.(string)
|
|
|
|
t := ""
|
|
|
|
for i := 0; i < len(s); i++ {
|
|
|
|
c := s[i]
|
|
|
|
if 'a' <= c && c <= 'z' {
|
|
|
|
c = c + 'A' - 'a'
|
|
|
|
}
|
|
|
|
t += string(c)
|
|
|
|
}
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
|
|
|
func plus1(v interface{}) string {
|
|
|
|
i := v.(int)
|
|
|
|
return fmt.Sprint(i + 1)
|
|
|
|
}
|
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
func writer(f func(interface{}) string) func(io.Writer, string, ...interface{}) {
|
|
|
|
return func(w io.Writer, format string, v ...interface{}) {
|
|
|
|
if len(v) != 1 {
|
|
|
|
panic("test writer expected one arg")
|
|
|
|
}
|
|
|
|
io.WriteString(w, f(v[0]))
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
func multiword(w io.Writer, format string, value ...interface{}) {
|
|
|
|
for _, v := range value {
|
|
|
|
fmt.Fprintf(w, "<%v>", v)
|
|
|
|
}
|
|
|
|
}
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
var formatters = FormatterMap{
|
|
|
|
"uppercase": writer(uppercase),
|
|
|
|
"+1": writer(plus1),
|
2011-01-21 19:19:03 +01:00
|
|
|
"multiword": multiword,
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
var tests = []*Test{
|
|
|
|
// Simple
|
|
|
|
&Test{"", "", ""},
|
|
|
|
&Test{"abc", "abc", ""},
|
|
|
|
&Test{"abc\ndef\n", "abc\ndef\n", ""},
|
|
|
|
&Test{" {.meta-left} \n", "{", ""},
|
|
|
|
&Test{" {.meta-right} \n", "}", ""},
|
|
|
|
&Test{" {.space} \n", " ", ""},
|
|
|
|
&Test{" {.tab} \n", "\t", ""},
|
|
|
|
&Test{" {#comment} \n", "", ""},
|
|
|
|
&Test{"\tSome Text\t\n", "\tSome Text\t\n", ""},
|
|
|
|
&Test{" {.meta-right} {.meta-right} {.meta-right} \n", " } } } \n", ""},
|
|
|
|
|
|
|
|
// Variables at top level
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{Header}={Integer}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "Header=77\n",
|
|
|
|
},
|
|
|
|
|
2011-03-17 00:05:44 +01:00
|
|
|
&Test{
|
|
|
|
in: "Pointers: {*HeaderPtr}={*IntegerPtr}\n",
|
|
|
|
|
|
|
|
out: "Pointers: Header=77\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
|
|
|
in: "Stars but not pointers: {*Header}={*Integer}\n",
|
|
|
|
|
|
|
|
out: "Stars but not pointers: Header=77\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
|
|
|
in: "nil pointer: {*NilPtr}={*Integer}\n",
|
|
|
|
|
|
|
|
out: "nil pointer: <nil>=77\n",
|
|
|
|
},
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
// Method at top level
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "ptrmethod={PointerMethod}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "ptrmethod=ptrmethod!\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "valmethod={ValueMethod}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "valmethod=valmethod!\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
// Section
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Data }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"some text for the section\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "some text for the section\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Data }\n" +
|
|
|
|
"{Header}={Integer}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "Header=77\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
|
|
|
"{Header}={Integer}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "Header=77\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"data present\n" +
|
|
|
|
"{.or}\n" +
|
|
|
|
"data not present\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "data present\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Empty }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"data present\n" +
|
|
|
|
"{.or}\n" +
|
|
|
|
"data not present\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "data not present\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Null }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"data present\n" +
|
|
|
|
"{.or}\n" +
|
|
|
|
"data not present\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "data not present\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
|
|
|
"{Header}={Integer}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{Header}={Integer}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "Header=77\n" +
|
|
|
|
"Header=77\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Data}{.end} {Header}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: " Header\n",
|
|
|
|
},
|
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
&Test{
|
|
|
|
in: "{.section Integer}{@}{.end}",
|
|
|
|
|
|
|
|
out: "77",
|
|
|
|
},
|
|
|
|
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
// Repeated
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.repeated section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{Item}={Value}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "ItemNumber1=ValueNumber1\n" +
|
|
|
|
"ItemNumber2=ValueNumber2\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.repeated section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{Item}={Value}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.or}\n" +
|
|
|
|
"this should not appear\n" +
|
|
|
|
"{.end}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "ItemNumber1=ValueNumber1\n" +
|
|
|
|
"ItemNumber2=ValueNumber2\n",
|
|
|
|
},
|
|
|
|
&Test{
|
|
|
|
in: "{.section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{.repeated section Empty }\n" +
|
|
|
|
"{Item}={Value}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.or}\n" +
|
|
|
|
"this should appear: empty field\n" +
|
|
|
|
"{.end}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "this should appear: empty field\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Pdata }\n" +
|
|
|
|
"{Item}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.alternates with}\n" +
|
|
|
|
"is\nover\nmultiple\nlines\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "ItemNumber1\n" +
|
|
|
|
"is\nover\nmultiple\nlines\n" +
|
|
|
|
"ItemNumber2\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Pdata }\n" +
|
|
|
|
"{Item}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.alternates with}\n" +
|
|
|
|
"is\nover\nmultiple\nlines\n" +
|
|
|
|
" {.end}\n",
|
|
|
|
|
|
|
|
out: "ItemNumber1\n" +
|
|
|
|
"is\nover\nmultiple\nlines\n" +
|
|
|
|
"ItemNumber2\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Pdata }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.repeated section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{Item}={Value}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.alternates with}DIVIDER\n" +
|
|
|
|
"{.or}\n" +
|
|
|
|
"this should not appear\n" +
|
|
|
|
"{.end}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "ItemNumber1=ValueNumber1\n" +
|
|
|
|
"DIVIDER\n" +
|
|
|
|
"ItemNumber2=ValueNumber2\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Vec }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{@}\n" +
|
|
|
|
"{.end}\n",
|
|
|
|
|
|
|
|
out: "elt1\n" +
|
|
|
|
"elt2\n",
|
|
|
|
},
|
|
|
|
// Same but with a space before {.end}: was a bug.
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Vec }\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{@} {.end}\n",
|
|
|
|
|
|
|
|
out: "elt1 elt2 \n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Integer}{.end}",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
2011-01-21 19:19:03 +01:00
|
|
|
err: "line 1: .repeated: cannot repeat Integer (type int)",
|
2010-12-03 05:34:57 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
// Nested names
|
|
|
|
&Test{
|
|
|
|
in: "{.section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{InnerT.Item}={InnerT.Value}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}",
|
|
|
|
|
|
|
|
out: "ItemNumber1=ValueNumber1\n",
|
|
|
|
},
|
|
|
|
&Test{
|
|
|
|
in: "{.section @ }\n" +
|
2011-01-21 19:19:03 +01:00
|
|
|
"{InnerT.Item}={.section InnerT}{.section Value}{@}{.end}{.end}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{.end}",
|
|
|
|
|
|
|
|
out: "ItemNumber1=ValueNumber1\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Emptystring}emptystring{.end}\n" +
|
|
|
|
"{.section Header}header{.end}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "\nheader\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section True}1{.or}2{.end}\n" +
|
|
|
|
"{.section False}3{.or}4{.end}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "1\n4\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
// Maps
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{Mp.mapkey}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "Ahoy!\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{Innermap.Mp.innerkey}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "55\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Innermap}{.section Mp}{innerkey}{.end}{.end}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "55\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section JSON}{.repeated section maps}{a}{b}{.end}{.end}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "1234\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{Stringmap.stringkey1}\n",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "stringresult\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Stringmap}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"{@}\n" +
|
|
|
|
"{.end}",
|
|
|
|
|
|
|
|
out: "stringresult\n" +
|
|
|
|
"stringresult\n",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Stringmap}\n" +
|
2010-12-03 05:34:57 +01:00
|
|
|
"\t{@}\n" +
|
|
|
|
"{.end}",
|
|
|
|
|
|
|
|
out: "\tstringresult\n" +
|
|
|
|
"\tstringresult\n",
|
|
|
|
},
|
2011-03-17 00:05:44 +01:00
|
|
|
&Test{
|
|
|
|
in: "{*Ptrmap.stringkey1}\n",
|
|
|
|
|
|
|
|
out: "pointedToString\n",
|
|
|
|
},
|
|
|
|
&Test{
|
|
|
|
in: "{.repeated section Ptrmap}\n" +
|
|
|
|
"{*@}\n" +
|
|
|
|
"{.end}",
|
|
|
|
|
|
|
|
out: "pointedToString\n" +
|
|
|
|
"pointedToString\n",
|
|
|
|
},
|
|
|
|
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
// Interface values
|
|
|
|
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{Iface}",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "[1 2 3]",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.repeated section Iface}{@}{.alternates with} {.end}",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "1 2 3",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Iface}{@}{.end}",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "[1 2 3]",
|
|
|
|
},
|
|
|
|
&Test{
|
2011-01-21 19:19:03 +01:00
|
|
|
in: "{.section Ifaceptr}{Item} {Value}{.end}",
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
out: "Item Value",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAll(t *testing.T) {
|
|
|
|
// Parse
|
|
|
|
testAll(t, func(test *Test) (*Template, os.Error) { return Parse(test.in, formatters) })
|
|
|
|
// ParseFile
|
|
|
|
testAll(t, func(test *Test) (*Template, os.Error) {
|
|
|
|
err := ioutil.WriteFile("_test/test.tmpl", []byte(test.in), 0600)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected write error:", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return ParseFile("_test/test.tmpl", formatters)
|
|
|
|
})
|
|
|
|
// tmpl.ParseFile
|
|
|
|
testAll(t, func(test *Test) (*Template, os.Error) {
|
|
|
|
err := ioutil.WriteFile("_test/test.tmpl", []byte(test.in), 0600)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected write error:", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tmpl := New(formatters)
|
|
|
|
return tmpl, tmpl.ParseFile("_test/test.tmpl")
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func testAll(t *testing.T, parseFunc func(*Test) (*Template, os.Error)) {
|
|
|
|
s := new(S)
|
|
|
|
// initialized by hand for clarity.
|
2011-01-21 19:19:03 +01:00
|
|
|
s.Header = "Header"
|
2011-03-17 00:05:44 +01:00
|
|
|
s.HeaderPtr = &s.Header
|
2011-01-21 19:19:03 +01:00
|
|
|
s.Integer = 77
|
2011-03-17 00:05:44 +01:00
|
|
|
s.IntegerPtr = &s.Integer
|
2011-01-21 19:19:03 +01:00
|
|
|
s.InnerT = t1
|
|
|
|
s.Data = []T{t1, t2}
|
|
|
|
s.Pdata = []*T{&t1, &t2}
|
|
|
|
s.Empty = []*T{}
|
|
|
|
s.Null = nil
|
|
|
|
s.Vec = new(vector.Vector)
|
|
|
|
s.Vec.Push("elt1")
|
|
|
|
s.Vec.Push("elt2")
|
|
|
|
s.True = true
|
|
|
|
s.False = false
|
|
|
|
s.Mp = make(map[string]string)
|
|
|
|
s.Mp["mapkey"] = "Ahoy!"
|
|
|
|
json.Unmarshal([]byte(`{"maps":[{"a":1,"b":2},{"a":3,"b":4}]}`), &s.JSON)
|
|
|
|
s.Innermap.Mp = make(map[string]int)
|
|
|
|
s.Innermap.Mp["innerkey"] = 55
|
|
|
|
s.Stringmap = make(map[string]string)
|
|
|
|
s.Stringmap["stringkey1"] = "stringresult" // the same value so repeated section is order-independent
|
|
|
|
s.Stringmap["stringkey2"] = "stringresult"
|
2011-03-17 00:05:44 +01:00
|
|
|
s.Ptrmap = make(map[string]*string)
|
|
|
|
x := "pointedToString"
|
|
|
|
s.Ptrmap["stringkey1"] = &x // the same value so repeated section is order-independent
|
|
|
|
s.Ptrmap["stringkey2"] = &x
|
2011-01-21 19:19:03 +01:00
|
|
|
s.Iface = []int{1, 2, 3}
|
|
|
|
s.Ifaceptr = &T{"Item", "Value"}
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
for _, test := range tests {
|
|
|
|
buf.Reset()
|
|
|
|
tmpl, err := parseFunc(test)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error: ", err)
|
|
|
|
continue
|
|
|
|
}
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&buf, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
if test.err == "" {
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected execute error:", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err == nil {
|
|
|
|
t.Errorf("expected execute error %q, got nil", test.err)
|
|
|
|
} else if err.String() != test.err {
|
|
|
|
t.Errorf("expected execute error %q, got %q", test.err, err.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if buf.String() != test.out {
|
|
|
|
t.Errorf("for %q: expected %q got %q", test.in, test.out, buf.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMapDriverType(t *testing.T) {
|
|
|
|
mp := map[string]string{"footer": "Ahoy!"}
|
|
|
|
tmpl, err := Parse("template: {footer}", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&b, mp)
|
2010-12-03 05:34:57 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected execute error:", err)
|
|
|
|
}
|
|
|
|
s := b.String()
|
2011-03-17 00:05:44 +01:00
|
|
|
expect := "template: Ahoy!"
|
|
|
|
if s != expect {
|
|
|
|
t.Errorf("failed passing string as data: expected %q got %q", expect, s)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMapNoEntry(t *testing.T) {
|
|
|
|
mp := make(map[string]int)
|
|
|
|
tmpl, err := Parse("template: {notthere}!", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
|
|
|
err = tmpl.Execute(&b, mp)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected execute error:", err)
|
|
|
|
}
|
|
|
|
s := b.String()
|
|
|
|
expect := "template: 0!"
|
|
|
|
if s != expect {
|
|
|
|
t.Errorf("failed passing string as data: expected %q got %q", expect, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStringDriverType(t *testing.T) {
|
|
|
|
tmpl, err := Parse("template: {@}", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&b, "hello")
|
2010-12-03 05:34:57 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected execute error:", err)
|
|
|
|
}
|
|
|
|
s := b.String()
|
2011-03-17 00:05:44 +01:00
|
|
|
expect := "template: hello"
|
|
|
|
if s != expect {
|
|
|
|
t.Errorf("failed passing string as data: expected %q got %q", expect, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestTwice(t *testing.T) {
|
|
|
|
tmpl, err := Parse("template: {@}", nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&b, "hello")
|
2010-12-03 05:34:57 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
s := b.String()
|
2011-03-17 00:05:44 +01:00
|
|
|
expect := "template: hello"
|
|
|
|
if s != expect {
|
|
|
|
t.Errorf("failed passing string as data: expected %q got %q", expect, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&b, "hello")
|
2010-12-03 05:34:57 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
}
|
|
|
|
s = b.String()
|
2011-03-17 00:05:44 +01:00
|
|
|
expect += expect
|
|
|
|
if s != expect {
|
|
|
|
t.Errorf("failed passing string as data: expected %q got %q", expect, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCustomDelims(t *testing.T) {
|
|
|
|
// try various lengths. zero should catch error.
|
|
|
|
for i := 0; i < 7; i++ {
|
|
|
|
for j := 0; j < 7; j++ {
|
|
|
|
tmpl := New(nil)
|
|
|
|
// first two chars deliberately the same to test equal left and right delims
|
|
|
|
ldelim := "$!#$%^&"[0:i]
|
|
|
|
rdelim := "$*&^%$!"[0:j]
|
|
|
|
tmpl.SetDelims(ldelim, rdelim)
|
|
|
|
// if braces, this would be template: {@}{.meta-left}{.meta-right}
|
|
|
|
text := "template: " +
|
|
|
|
ldelim + "@" + rdelim +
|
|
|
|
ldelim + ".meta-left" + rdelim +
|
|
|
|
ldelim + ".meta-right" + rdelim
|
|
|
|
err := tmpl.Parse(text)
|
|
|
|
if err != nil {
|
|
|
|
if i == 0 || j == 0 { // expected
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
} else if i == 0 || j == 0 {
|
|
|
|
t.Errorf("expected parse error for empty delimiter: %d %d %q %q", i, j, ldelim, rdelim)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
var b bytes.Buffer
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&b, "hello")
|
2010-12-03 05:34:57 +01:00
|
|
|
s := b.String()
|
|
|
|
if s != "template: hello"+ldelim+rdelim {
|
|
|
|
t.Errorf("failed delim check(%q %q) %q got %q", ldelim, rdelim, text, s)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test that a variable evaluates to the field itself and does not further indirection
|
|
|
|
func TestVarIndirection(t *testing.T) {
|
|
|
|
s := new(S)
|
|
|
|
// initialized by hand for clarity.
|
2011-01-21 19:19:03 +01:00
|
|
|
s.InnerPointerT = &t1
|
2010-12-03 05:34:57 +01:00
|
|
|
|
|
|
|
var buf bytes.Buffer
|
2011-01-21 19:19:03 +01:00
|
|
|
input := "{.section @}{InnerPointerT}{.end}"
|
2010-12-03 05:34:57 +01:00
|
|
|
tmpl, err := Parse(input, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("unexpected parse error:", err)
|
|
|
|
}
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&buf, s)
|
2010-12-03 05:34:57 +01:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal("unexpected execute error:", err)
|
|
|
|
}
|
|
|
|
expect := fmt.Sprintf("%v", &t1) // output should be hex address of t1
|
|
|
|
if buf.String() != expect {
|
|
|
|
t.Errorf("for %q: expected %q got %q", input, expect, buf.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHTMLFormatterWithByte(t *testing.T) {
|
|
|
|
s := "Test string."
|
|
|
|
b := []byte(s)
|
|
|
|
var buf bytes.Buffer
|
2011-01-21 19:19:03 +01:00
|
|
|
HTMLFormatter(&buf, "", b)
|
2010-12-03 05:34:57 +01:00
|
|
|
bs := buf.String()
|
|
|
|
if bs != s {
|
|
|
|
t.Errorf("munged []byte, expected: %s got: %s", s, bs)
|
|
|
|
}
|
|
|
|
}
|
2011-01-21 19:19:03 +01:00
|
|
|
|
|
|
|
type UF struct {
|
|
|
|
I int
|
|
|
|
s string
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestReferenceToUnexported(t *testing.T) {
|
|
|
|
u := &UF{3, "hello"}
|
|
|
|
var buf bytes.Buffer
|
|
|
|
input := "{.section @}{I}{s}{.end}"
|
|
|
|
tmpl, err := Parse(input, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("unexpected parse error:", err)
|
|
|
|
}
|
2011-03-17 00:05:44 +01:00
|
|
|
err = tmpl.Execute(&buf, u)
|
2011-01-21 19:19:03 +01:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("expected execute error, got none")
|
|
|
|
}
|
|
|
|
if strings.Index(err.String(), "not exported") < 0 {
|
|
|
|
t.Fatal("expected unexported error; got", err)
|
|
|
|
}
|
|
|
|
}
|
2011-03-17 00:05:44 +01:00
|
|
|
|
|
|
|
var formatterTests = []Test{
|
|
|
|
{
|
|
|
|
in: "{Header|uppercase}={Integer|+1}\n" +
|
|
|
|
"{Header|html}={Integer|str}\n",
|
|
|
|
|
|
|
|
out: "HEADER=78\n" +
|
|
|
|
"Header=77\n",
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
in: "{Header|uppercase}={Integer Header|multiword}\n" +
|
|
|
|
"{Header|html}={Header Integer|multiword}\n" +
|
|
|
|
"{Header|html}={Header Integer}\n",
|
|
|
|
|
|
|
|
out: "HEADER=<77><Header>\n" +
|
|
|
|
"Header=<Header><77>\n" +
|
|
|
|
"Header=Header77\n",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Raw}\n" +
|
|
|
|
"{Raw|html}\n",
|
|
|
|
|
|
|
|
out: "a <&> b\n" +
|
|
|
|
"a <&> b\n",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Bytes}",
|
|
|
|
out: "hello",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Raw|uppercase|html|html}",
|
|
|
|
out: "A &lt;&amp;&gt; B",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Header Integer|multiword|html}",
|
|
|
|
out: "<Header><77>",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Integer|no_formatter|html}",
|
|
|
|
err: `unknown formatter: "no_formatter"`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
in: "{Integer|||||}", // empty string is a valid formatter
|
|
|
|
out: "77",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFormatters(t *testing.T) {
|
|
|
|
data := map[string]interface{}{
|
|
|
|
"Header": "Header",
|
|
|
|
"Integer": 77,
|
|
|
|
"Raw": "a <&> b",
|
|
|
|
"Bytes": []byte("hello"),
|
|
|
|
}
|
|
|
|
for _, c := range formatterTests {
|
|
|
|
tmpl, err := Parse(c.in, formatters)
|
|
|
|
if err != nil {
|
|
|
|
if c.err == "" {
|
|
|
|
t.Error("unexpected parse error:", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if strings.Index(err.String(), c.err) < 0 {
|
|
|
|
t.Errorf("unexpected error: expected %q, got %q", c.err, err.String())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if c.err != "" {
|
|
|
|
t.Errorf("For %q, expected error, got none.", c.in)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
buf := bytes.NewBuffer(nil)
|
|
|
|
err = tmpl.Execute(buf, data)
|
|
|
|
if err != nil {
|
|
|
|
t.Error("unexpected Execute error: ", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
actual := buf.String()
|
|
|
|
if actual != c.out {
|
|
|
|
t.Errorf("for %q: expected %q but got %q.", c.in, c.out, actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|