OLD | NEW |
(Empty) | |
| 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 package template |
| 6 |
| 7 import ( |
| 8 "bytes" |
| 9 "fmt" |
| 10 "os" |
| 11 "sort" |
| 12 "strings" |
| 13 "testing" |
| 14 ) |
| 15 |
| 16 // T has lots of interesting pieces to use to test execution. |
| 17 type T struct { |
| 18 // Basics |
| 19 I int |
| 20 U16 uint16 |
| 21 X string |
| 22 // Nested structs. |
| 23 U *U |
| 24 // Slices |
| 25 SI []int |
| 26 SEmpty []int |
| 27 // Maps |
| 28 MSI map[string]int |
| 29 MSIEmpty map[string]int |
| 30 } |
| 31 |
| 32 // Simple methods with and without arguments. |
| 33 func (t *T) Method0() string { |
| 34 return "resultOfMethod0" |
| 35 } |
| 36 |
| 37 func (t *T) Method1(a int) int { |
| 38 return a |
| 39 } |
| 40 |
| 41 func (t *T) Method2(a uint16, b string) string { |
| 42 return fmt.Sprintf("Method2: %d %s", a, b) |
| 43 } |
| 44 |
| 45 func (t *T) MAdd(a int, b []int) []int { |
| 46 v := make([]int, len(b)) |
| 47 for i, x := range b { |
| 48 v[i] = x + a |
| 49 } |
| 50 return v |
| 51 } |
| 52 |
| 53 // MSort is used to sort map keys for stable output. (Nice trick!) |
| 54 func (t *T) MSort(m map[string]int) []string { |
| 55 keys := make([]string, len(m)) |
| 56 i := 0 |
| 57 for k := range m { |
| 58 keys[i] = k |
| 59 i++ |
| 60 } |
| 61 sort.SortStrings(keys) |
| 62 return keys |
| 63 } |
| 64 |
| 65 // EPERM returns a value and an os.Error according to its argument. |
| 66 func (t *T) EPERM(a int) (int, os.Error) { |
| 67 if a == 0 { |
| 68 return 0, os.EPERM |
| 69 } |
| 70 return a, nil |
| 71 } |
| 72 |
| 73 type U struct { |
| 74 V string |
| 75 } |
| 76 |
| 77 var tVal = &T{ |
| 78 I: 17, |
| 79 U16: 16, |
| 80 X: "x", |
| 81 U: &U{"v"}, |
| 82 SI: []int{3, 4, 5}, |
| 83 MSI: map[string]int{"one": 1, "two": 2, "three": 3}, |
| 84 } |
| 85 |
| 86 type execTest struct { |
| 87 name string |
| 88 input string |
| 89 output string |
| 90 data interface{} |
| 91 ok bool |
| 92 } |
| 93 |
| 94 var execTests = []execTest{ |
| 95 {"empty", "", "", nil, true}, |
| 96 {"text", "some text", "some text", nil, true}, |
| 97 {".X", "-{{.X}}-", "-x-", tVal, true}, |
| 98 {".U.V", "-{{.U.V}}-", "-v-", tVal, true}, |
| 99 {".Method0", "-{{.Method0}}-", "-resultOfMethod0-", tVal, true}, |
| 100 {".Method1(1234)", "-{{.Method1 1234}}-", "-1234-", tVal, true}, |
| 101 {".Method1(.I)", "-{{.Method1 .I}}-", "-17-", tVal, true}, |
| 102 {".Method2(3, .X)", "-{{.Method2 3 .X}}-", "-Method2: 3 x-", tVal, true}
, |
| 103 {".Method2(.U16, `str`)", "-{{.Method2 .U16 `str`}}-", "-Method2: 16 str
-", tVal, true}, |
| 104 {"pipeline", "-{{.Method0 | .Method2 .U16}}-", "-Method2: 16 resultOfMet
hod0-", tVal, true}, |
| 105 {"range []int", "{{range .SI}}-{{.}}-{{end}}", "-3--4--5-", tVal, true}, |
| 106 {"range empty no else", "{{range .SEmpty}}-{{.}}-{{end}}", "", tVal, tru
e}, |
| 107 {"range []int else", "{{range .SI}}-{{.}}-{{else}}EMPTY{{end}}", "-3--4-
-5-", tVal, true}, |
| 108 {"range empty else", "{{range .SEmpty}}-{{.}}-{{else}}EMPTY{{end}}", "EM
PTY", tVal, true}, |
| 109 {"range []int method", "{{range .SI | .MAdd .I}}-{{.}}-{{end}}", "-20--2
1--22-", tVal, true}, |
| 110 {"range map", "{{range .MSI | .MSort}}-{{.}}-{{end}}", "-one--three--two
-", tVal, true}, |
| 111 {"range empty map no else", "{{range .MSIEmpty}}-{{.}}-{{end}}", "", tVa
l, true}, |
| 112 {"range map else", "{{range .MSI | .MSort}}-{{.}}-{{else}}EMPTY{{end}}",
"-one--three--two-", tVal, true}, |
| 113 {"range empty map else", "{{range .MSIEmpty}}-{{.}}-{{else}}EMPTY{{end}}
", "EMPTY", tVal, true}, |
| 114 {"error method, no error", "{{.EPERM 1}}", "1", tVal, true}, |
| 115 {"error method, error", "{{.EPERM 0}}", "1", tVal, false}, |
| 116 } |
| 117 |
| 118 func TestExecute(t *testing.T) { |
| 119 b := new(bytes.Buffer) |
| 120 for _, test := range execTests { |
| 121 tmpl := New(test.name) |
| 122 err := tmpl.Parse(test.input) |
| 123 if err != nil { |
| 124 t.Errorf("%s: parse error: %s", test.name, err) |
| 125 continue |
| 126 } |
| 127 b.Reset() |
| 128 err = tmpl.Execute(b, test.data) |
| 129 switch { |
| 130 case !test.ok && err == nil: |
| 131 t.Errorf("%s: expected error; got none", test.name) |
| 132 continue |
| 133 case test.ok && err != nil: |
| 134 t.Errorf("%s: unexpected execute error: %s", test.name,
err) |
| 135 continue |
| 136 case !test.ok && err != nil: |
| 137 continue |
| 138 } |
| 139 result := b.String() |
| 140 if result != test.output { |
| 141 t.Errorf("%s: expected\n\t%q\ngot\n\t%q", test.name, tes
t.output, result) |
| 142 } |
| 143 } |
| 144 } |
| 145 |
| 146 // Check that an error from a method flows back to the top. |
| 147 func TestExecuteError(t *testing.T) { |
| 148 b := new(bytes.Buffer) |
| 149 tmpl := New("error") |
| 150 err := tmpl.Parse("{{.EPERM 0}}") |
| 151 if err != nil { |
| 152 t.Fatalf("parse error: %s", err) |
| 153 } |
| 154 err = tmpl.Execute(b, tVal) |
| 155 if err == nil { |
| 156 t.Errorf("expected error; got none") |
| 157 } else if !strings.Contains(err.String(), os.EPERM.String()) { |
| 158 t.Errorf("expected os.EPERM; got %s %s", err) |
| 159 } |
| 160 } |
OLD | NEW |