/src/pkg/exp/eval/eval_test.go

https://code.google.com/p/golang-on-cygwin/ · Go · 254 lines · 186 code · 43 blank · 25 comment · 24 complexity · dd832e42add30e7e92daa21bedb30a62 MD5 · raw file

  1. // Copyright 2009 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. package eval
  5. import (
  6. "bignum";
  7. "flag";
  8. "fmt";
  9. "log";
  10. "os";
  11. "reflect";
  12. "testing";
  13. )
  14. // Print each statement or expression before parsing it
  15. var noisy = false
  16. func init() { flag.BoolVar(&noisy, "noisy", false, "chatter during eval tests") }
  17. /*
  18. * Generic statement/expression test framework
  19. */
  20. type test []job
  21. type job struct {
  22. code string;
  23. cerr string;
  24. rterr string;
  25. val Value;
  26. noval bool;
  27. }
  28. func runTests(t *testing.T, baseName string, tests []test) {
  29. for i, test := range tests {
  30. name := fmt.Sprintf("%s[%d]", baseName, i);
  31. test.run(t, name);
  32. }
  33. }
  34. func (a test) run(t *testing.T, name string) {
  35. w := newTestWorld();
  36. for _, j := range a {
  37. src := j.code;
  38. if noisy {
  39. println("code:", src)
  40. }
  41. code, err := w.Compile(src);
  42. if err != nil {
  43. if j.cerr == "" {
  44. t.Errorf("%s: Compile %s: %v", name, src, err);
  45. break;
  46. }
  47. if !match(t, err, j.cerr) {
  48. t.Errorf("%s: Compile %s = error %s; want %v", name, src, err, j.cerr);
  49. break;
  50. }
  51. continue;
  52. }
  53. if j.cerr != "" {
  54. t.Errorf("%s: Compile %s succeeded; want %s", name, src, j.cerr);
  55. break;
  56. }
  57. val, err := code.Run();
  58. if err != nil {
  59. if j.rterr == "" {
  60. t.Errorf("%s: Run %s: %v", name, src, err);
  61. break;
  62. }
  63. if !match(t, err, j.rterr) {
  64. t.Errorf("%s: Run %s = error %s; want %v", name, src, err, j.rterr);
  65. break;
  66. }
  67. continue;
  68. }
  69. if j.rterr != "" {
  70. t.Errorf("%s: Run %s succeeded; want %s", name, src, j.rterr);
  71. break;
  72. }
  73. if !j.noval && !reflect.DeepEqual(val, j.val) {
  74. t.Errorf("%s: Run %s = %T(%v) want %T(%v)", name, src, val, val, j.val, j.val)
  75. }
  76. }
  77. }
  78. func match(t *testing.T, err os.Error, pat string) bool {
  79. ok, errstr := testing.MatchString(pat, err.String());
  80. if errstr != "" {
  81. t.Fatalf("compile regexp %s: %v", pat, errstr)
  82. }
  83. return ok;
  84. }
  85. /*
  86. * Test constructors
  87. */
  88. // Expression compile error
  89. func CErr(expr string, cerr string) test { return test([]job{job{code: expr, cerr: cerr}}) }
  90. // Expression runtime error
  91. func RErr(expr string, rterr string) test { return test([]job{job{code: expr, rterr: rterr}}) }
  92. // Expression value
  93. func Val(expr string, val interface{}) test {
  94. return test([]job{job{code: expr, val: toValue(val)}})
  95. }
  96. // Statement runs without error
  97. func Run(stmts string) test { return test([]job{job{code: stmts, noval: true}}) }
  98. // Two statements without error.
  99. // TODO(rsc): Should be possible with Run but the parser
  100. // won't let us do both top-level and non-top-level statements.
  101. func Run2(stmt1, stmt2 string) test {
  102. return test([]job{job{code: stmt1, noval: true}, job{code: stmt2, noval: true}})
  103. }
  104. // Statement runs and test one expression's value
  105. func Val1(stmts string, expr1 string, val1 interface{}) test {
  106. return test([]job{
  107. job{code: stmts, noval: true},
  108. job{code: expr1, val: toValue(val1)},
  109. })
  110. }
  111. // Statement runs and test two expressions' values
  112. func Val2(stmts string, expr1 string, val1 interface{}, expr2 string, val2 interface{}) test {
  113. return test([]job{
  114. job{code: stmts, noval: true},
  115. job{code: expr1, val: toValue(val1)},
  116. job{code: expr2, val: toValue(val2)},
  117. })
  118. }
  119. /*
  120. * Value constructors
  121. */
  122. type vstruct []interface{}
  123. type varray []interface{}
  124. type vslice struct {
  125. arr varray;
  126. len, cap int;
  127. }
  128. func toValue(val interface{}) Value {
  129. switch val := val.(type) {
  130. case bool:
  131. r := boolV(val);
  132. return &r;
  133. case uint8:
  134. r := uint8V(val);
  135. return &r;
  136. case uint:
  137. r := uintV(val);
  138. return &r;
  139. case int:
  140. r := intV(val);
  141. return &r;
  142. case *bignum.Integer:
  143. return &idealIntV{val}
  144. case float:
  145. r := floatV(val);
  146. return &r;
  147. case *bignum.Rational:
  148. return &idealFloatV{val}
  149. case string:
  150. r := stringV(val);
  151. return &r;
  152. case vstruct:
  153. elems := make([]Value, len(val));
  154. for i, e := range val {
  155. elems[i] = toValue(e)
  156. }
  157. r := structV(elems);
  158. return &r;
  159. case varray:
  160. elems := make([]Value, len(val));
  161. for i, e := range val {
  162. elems[i] = toValue(e)
  163. }
  164. r := arrayV(elems);
  165. return &r;
  166. case vslice:
  167. return &sliceV{Slice{toValue(val.arr).(ArrayValue), int64(val.len), int64(val.cap)}}
  168. case Func:
  169. return &funcV{val}
  170. }
  171. log.Crashf("toValue(%T) not implemented", val);
  172. panic();
  173. }
  174. /*
  175. * Default test scope
  176. */
  177. type testFunc struct{}
  178. func (*testFunc) NewFrame() *Frame { return &Frame{nil, &[2]Value{}} }
  179. func (*testFunc) Call(t *Thread) {
  180. n := t.f.Vars[0].(IntValue).Get(t);
  181. res := n + 1;
  182. t.f.Vars[1].(IntValue).Set(t, res);
  183. }
  184. type oneTwoFunc struct{}
  185. func (*oneTwoFunc) NewFrame() *Frame { return &Frame{nil, &[2]Value{}} }
  186. func (*oneTwoFunc) Call(t *Thread) {
  187. t.f.Vars[0].(IntValue).Set(t, 1);
  188. t.f.Vars[1].(IntValue).Set(t, 2);
  189. }
  190. type voidFunc struct{}
  191. func (*voidFunc) NewFrame() *Frame { return &Frame{nil, []Value{}} }
  192. func (*voidFunc) Call(t *Thread) {}
  193. func newTestWorld() *World {
  194. w := NewWorld();
  195. def := func(name string, t Type, val interface{}) { w.DefineVar(name, t, toValue(val)) };
  196. w.DefineConst("c", IdealIntType, toValue(bignum.Int(1)));
  197. def("i", IntType, 1);
  198. def("i2", IntType, 2);
  199. def("u", UintType, uint(1));
  200. def("f", FloatType, 1.0);
  201. def("s", StringType, "abc");
  202. def("t", NewStructType([]StructField{StructField{"a", IntType, false}}), vstruct{1});
  203. def("ai", NewArrayType(2, IntType), varray{1, 2});
  204. def("aai", NewArrayType(2, NewArrayType(2, IntType)), varray{varray{1, 2}, varray{3, 4}});
  205. def("aai2", NewArrayType(2, NewArrayType(2, IntType)), varray{varray{5, 6}, varray{7, 8}});
  206. def("fn", NewFuncType([]Type{IntType}, false, []Type{IntType}), &testFunc{});
  207. def("oneTwo", NewFuncType([]Type{}, false, []Type{IntType, IntType}), &oneTwoFunc{});
  208. def("void", NewFuncType([]Type{}, false, []Type{}), &voidFunc{});
  209. def("sli", NewSliceType(IntType), vslice{varray{1, 2, 3}, 2, 3});
  210. return w;
  211. }