| // Copyright 2018 The CUE Authors |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| package cue |
| |
| import ( |
| "bytes" |
| "fmt" |
| "io/ioutil" |
| "math" |
| "math/big" |
| "reflect" |
| "strconv" |
| "strings" |
| "testing" |
| |
| "github.com/google/go-cmp/cmp" |
| |
| "cuelang.org/go/cue/ast" |
| "cuelang.org/go/internal/astinternal" |
| "cuelang.org/go/internal/core/adt" |
| "cuelang.org/go/internal/core/debug" |
| ) |
| |
| func getInstance(t *testing.T, body ...string) *Instance { |
| t.Helper() |
| |
| insts := Build(makeInstances([]*bimport{{files: body}})) |
| if insts[0].Err != nil { |
| t.Fatalf("unexpected parse error: %v", insts[0].Err) |
| } |
| return insts[0] |
| } |
| |
| func TestAPI(t *testing.T) { |
| testCases := []struct { |
| input string |
| fun func(i *Instance) Value |
| want string |
| skip bool |
| }{{ |
| // Issue #567 |
| input: ` |
| #runSpec: {action: foo: int} |
| |
| v: {ction: foo: 1} |
| `, |
| fun: func(i *Instance) Value { |
| runSpec := i.LookupDef("#runSpec") |
| v := i.Lookup("v") |
| res := runSpec.Unify(v) |
| return res |
| }, |
| want: "_|_ // #runSpec: field ction not allowed", |
| }, { |
| // Issue #567 |
| input: ` |
| #runSpec: {action: foo: int} |
| |
| v: {action: Foo: 1} |
| `, |
| fun: func(i *Instance) Value { |
| runSpec := i.LookupDef("#runSpec") |
| v := i.Lookup("v") |
| res := runSpec.Unify(v) |
| return res |
| }, |
| want: "_|_ // #runSpec.action: field Foo not allowed", |
| }, { |
| input: ` |
| #runSpec: v: {action: foo: int} |
| |
| w: {ction: foo: 1} |
| `, |
| fun: func(i *Instance) Value { |
| runSpec := i.LookupDef("#runSpec") |
| v := runSpec.Lookup("v") |
| w := i.Lookup("w") |
| res := w.Unify(v) |
| return res |
| }, |
| want: "_|_ // w: field ction not allowed", |
| }} |
| for _, tc := range testCases { |
| if tc.skip { |
| continue |
| } |
| t.Run("", func(t *testing.T) { |
| var r Runtime |
| inst, err := r.Compile("in", tc.input) |
| if err != nil { |
| t.Fatal(err) |
| } |
| v := tc.fun(inst) |
| got := fmt.Sprintf("%+v", v) |
| if got != tc.want { |
| t.Errorf("got:\n%s\nwant:\n%s", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func TestValueType(t *testing.T) { |
| testCases := []struct { |
| value string |
| kind Kind |
| incompleteKind Kind |
| json string |
| valid bool |
| concrete bool |
| closed bool |
| // pos token.Pos |
| }{{ // Not a concrete value. |
| value: `v: _`, |
| kind: BottomKind, |
| incompleteKind: TopKind, |
| }, { |
| value: `v: _|_`, |
| kind: BottomKind, |
| incompleteKind: BottomKind, |
| concrete: true, |
| }, { |
| value: `v: 1&2`, |
| kind: BottomKind, |
| incompleteKind: BottomKind, |
| concrete: true, |
| }, { |
| value: `v: b, b: 1&2`, |
| kind: BottomKind, |
| incompleteKind: BottomKind, |
| concrete: true, |
| }, { |
| value: `v: (b[a]), b: 1, a: 1`, |
| kind: BottomKind, |
| incompleteKind: BottomKind, |
| concrete: true, |
| }, { // TODO: should be error{ |
| value: `v: (b) |
| b: bool`, |
| kind: BottomKind, |
| incompleteKind: BoolKind, |
| }, { |
| value: `v: ([][b]), b: "d"`, |
| kind: BottomKind, |
| incompleteKind: BottomKind, |
| concrete: true, |
| }, { |
| value: `v: null`, |
| kind: NullKind, |
| incompleteKind: NullKind, |
| concrete: true, |
| }, { |
| value: `v: true`, |
| kind: BoolKind, |
| incompleteKind: BoolKind, |
| concrete: true, |
| }, { |
| value: `v: false`, |
| kind: BoolKind, |
| incompleteKind: BoolKind, |
| concrete: true, |
| }, { |
| value: `v: bool`, |
| kind: BottomKind, |
| incompleteKind: BoolKind, |
| }, { |
| value: `v: 2`, |
| kind: IntKind, |
| incompleteKind: IntKind, |
| concrete: true, |
| }, { |
| value: `v: 2.0`, |
| kind: FloatKind, |
| incompleteKind: FloatKind, |
| concrete: true, |
| }, { |
| value: `v: 2.0Mi`, |
| kind: IntKind, |
| incompleteKind: IntKind, |
| concrete: true, |
| }, { |
| value: `v: 14_000`, |
| kind: IntKind, |
| incompleteKind: IntKind, |
| concrete: true, |
| }, { |
| value: `v: >=0 & <5`, |
| kind: BottomKind, |
| incompleteKind: NumberKind, |
| }, { |
| value: `v: float`, |
| kind: BottomKind, |
| incompleteKind: FloatKind, |
| }, { |
| value: `v: "str"`, |
| kind: StringKind, |
| incompleteKind: StringKind, |
| concrete: true, |
| }, { |
| value: "v: '''\n'''", |
| kind: BytesKind, |
| incompleteKind: BytesKind, |
| concrete: true, |
| }, { |
| value: "v: string", |
| kind: BottomKind, |
| incompleteKind: StringKind, |
| }, { |
| value: `v: {}`, |
| kind: StructKind, |
| incompleteKind: StructKind, |
| concrete: true, |
| }, { |
| value: `v: close({})`, |
| kind: StructKind, |
| incompleteKind: StructKind, |
| concrete: true, |
| closed: true, |
| }, { |
| value: `v: []`, |
| kind: ListKind, |
| incompleteKind: ListKind, |
| concrete: true, |
| closed: true, |
| }, { |
| value: `v: [...int]`, |
| kind: BottomKind, |
| incompleteKind: ListKind, |
| concrete: false, |
| }, { |
| value: `v: {a: int, b: [1][a]}.b`, |
| kind: BottomKind, |
| concrete: false, |
| }, { |
| value: `import "time" |
| v: time.Time`, |
| kind: BottomKind, |
| incompleteKind: StringKind, |
| concrete: false, |
| }, { |
| value: `import "time" |
| v: {a: time.Time}.a`, |
| kind: BottomKind, |
| incompleteKind: StringKind, |
| concrete: false, |
| }, { |
| value: `import "time" |
| v: {a: time.Time & string}.a`, |
| kind: BottomKind, |
| incompleteKind: StringKind, |
| concrete: false, |
| }, { |
| value: `import "strings" |
| v: {a: strings.ContainsAny("D")}.a`, |
| kind: BottomKind, |
| incompleteKind: StringKind, |
| concrete: false, |
| }, { |
| value: `import "struct" |
| v: {a: struct.MaxFields(2) & {}}.a`, |
| kind: StructKind, // Can determine a valid struct already. |
| incompleteKind: StructKind, |
| concrete: true, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| inst := getInstance(t, tc.value) |
| v := inst.Lookup("v") |
| if got := v.Kind(); got != tc.kind { |
| t.Errorf("Kind: got %x; want %x", int(got), int(tc.kind)) |
| } |
| want := tc.incompleteKind | BottomKind |
| if got := v.IncompleteKind(); got != want { |
| t.Errorf("IncompleteKind: got %x; want %x", int(got), int(want)) |
| } |
| if got := v.IsConcrete(); got != tc.concrete { |
| t.Errorf("IsConcrete: got %v; want %v", got, tc.concrete) |
| } |
| if got := v.IsClosed(); got != tc.closed { |
| t.Errorf("IsClosed: got %v; want %v", got, tc.closed) |
| } |
| }) |
| } |
| } |
| |
| func TestInt(t *testing.T) { |
| testCases := []struct { |
| value string |
| int int64 |
| uint uint64 |
| base int |
| err string |
| errU string |
| notInt bool |
| }{{ |
| value: "1", |
| int: 1, |
| uint: 1, |
| }, { |
| value: "-1", |
| int: -1, |
| uint: 0, |
| errU: ErrAbove.Error(), |
| }, { |
| value: "-111222333444555666777888999000", |
| int: math.MinInt64, |
| uint: 0, |
| err: ErrAbove.Error(), |
| errU: ErrAbove.Error(), |
| }, { |
| value: "111222333444555666777888999000", |
| int: math.MaxInt64, |
| uint: math.MaxUint64, |
| err: ErrBelow.Error(), |
| errU: ErrBelow.Error(), |
| }, { |
| value: "1.0", |
| err: "cannot use value 1.0 (type float) as int", |
| errU: "cannot use value 1.0 (type float) as int", |
| notInt: true, |
| }, { |
| value: "int", |
| err: "non-concrete value int", |
| errU: "non-concrete value int", |
| notInt: true, |
| }, { |
| value: "_|_", |
| err: "explicit error (_|_ literal) in source", |
| errU: "explicit error (_|_ literal) in source", |
| notInt: true, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| n := getInstance(t, tc.value).Value() |
| base := 10 |
| if tc.base > 0 { |
| base = tc.base |
| } |
| b, err := n.AppendInt(nil, base) |
| if checkFailed(t, err, tc.err, "append") { |
| want := tc.value |
| if got := string(b); got != want { |
| t.Errorf("append: got %v; want %v", got, want) |
| } |
| } |
| |
| vi, err := n.Int64() |
| checkErr(t, err, tc.err, "Int64") |
| if vi != tc.int { |
| t.Errorf("Int64: got %v; want %v", vi, tc.int) |
| } |
| |
| vu, err := n.Uint64() |
| checkErr(t, err, tc.errU, "Uint64") |
| if vu != uint64(tc.uint) { |
| t.Errorf("Uint64: got %v; want %v", vu, tc.uint) |
| } |
| }) |
| } |
| } |
| |
| func TestFloat(t *testing.T) { |
| testCases := []struct { |
| value string |
| float string |
| float64 float64 |
| mant string |
| exp int |
| fmt byte |
| prec int |
| kind Kind |
| err string |
| }{{ |
| value: "1", |
| float: "1", |
| mant: "1", |
| exp: 0, |
| float64: 1, |
| fmt: 'g', |
| kind: IntKind, |
| }, { |
| value: "-1", |
| float: "-1", |
| mant: "-1", |
| exp: 0, |
| float64: -1, |
| fmt: 'g', |
| kind: IntKind, |
| }, { |
| value: "1.0", |
| float: "1.0", |
| mant: "10", |
| exp: -1, |
| float64: 1.0, |
| fmt: 'g', |
| kind: FloatKind, |
| }, { |
| value: "2.6", |
| float: "2.6", |
| mant: "26", |
| exp: -1, |
| float64: 2.6, |
| fmt: 'g', |
| kind: FloatKind, |
| }, { |
| value: "20.600", |
| float: "20.60", |
| mant: "20600", |
| exp: -3, |
| float64: 20.60, |
| prec: 2, |
| fmt: 'f', |
| kind: FloatKind, |
| }, { |
| value: "1/0", |
| float: "", |
| float64: 0, |
| prec: 2, |
| fmt: 'f', |
| err: "division by zero", |
| kind: BottomKind, |
| }, { |
| value: "1.797693134862315708145274237317043567982e+308", |
| float: "1.8e+308", |
| mant: "1797693134862315708145274237317043567982", |
| exp: 269, |
| float64: math.Inf(1), |
| prec: 2, |
| fmt: 'g', |
| err: ErrAbove.Error(), |
| kind: FloatKind, |
| }, { |
| value: "-1.797693134862315708145274237317043567982e+308", |
| float: "-1.8e+308", |
| mant: "-1797693134862315708145274237317043567982", |
| exp: 269, |
| float64: math.Inf(-1), |
| prec: 2, |
| fmt: 'g', |
| kind: FloatKind, |
| err: ErrBelow.Error(), |
| }, { |
| value: "4.940656458412465441765687928682213723650e-324", |
| float: "4.941e-324", |
| mant: "4940656458412465441765687928682213723650", |
| exp: -363, |
| float64: 0, |
| prec: 4, |
| fmt: 'g', |
| kind: FloatKind, |
| err: ErrBelow.Error(), |
| }, { |
| value: "-4.940656458412465441765687928682213723650e-324", |
| float: "-4.940656458412465441765687928682213723650e-324", |
| mant: "-4940656458412465441765687928682213723650", |
| exp: -363, |
| float64: 0, |
| prec: -1, |
| fmt: 'g', |
| kind: FloatKind, |
| err: ErrAbove.Error(), |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| n := getInstance(t, tc.value).Value() |
| if n.Kind() != tc.kind { |
| t.Fatal("Not a number") |
| } |
| |
| var mant big.Int |
| exp, err := n.MantExp(&mant) |
| mstr := "" |
| if err == nil { |
| mstr = mant.String() |
| } |
| if exp != tc.exp || mstr != tc.mant { |
| t.Errorf("mantExp: got %s %d; want %s %d", mstr, exp, tc.mant, tc.exp) |
| } |
| |
| b, _ := n.AppendFloat(nil, tc.fmt, tc.prec) |
| want := tc.float |
| if got := string(b); got != want { |
| t.Errorf("append: got %v; want %v", got, want) |
| } |
| |
| f, err := n.Float64() |
| checkErr(t, err, tc.err, "Float64") |
| if f != tc.float64 { |
| t.Errorf("Float64: got %v; want %v", f, tc.float64) |
| } |
| }) |
| } |
| } |
| |
| func TestString(t *testing.T) { |
| testCases := []struct { |
| value string |
| str string |
| err string |
| }{{ |
| value: `""`, |
| str: ``, |
| }, { |
| value: `"Hello world!"`, |
| str: `Hello world!`, |
| }, { |
| value: `"Hello \(#world)!" |
| #world: "world"`, |
| str: `Hello world!`, |
| }, { |
| value: `string`, |
| err: "non-concrete value string", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| str, err := getInstance(t, tc.value).Value().String() |
| checkFatal(t, err, tc.err, "init") |
| if str != tc.str { |
| t.Errorf("String: got %q; want %q", str, tc.str) |
| } |
| |
| b, err := getInstance(t, tc.value).Value().Bytes() |
| checkFatal(t, err, tc.err, "init") |
| if got := string(b); got != tc.str { |
| t.Errorf("Bytes: got %q; want %q", got, tc.str) |
| } |
| |
| r, err := getInstance(t, tc.value).Value().Reader() |
| checkFatal(t, err, tc.err, "init") |
| b, _ = ioutil.ReadAll(r) |
| if got := string(b); got != tc.str { |
| t.Errorf("Reader: got %q; want %q", got, tc.str) |
| } |
| }) |
| } |
| } |
| |
| func TestError(t *testing.T) { |
| testCases := []struct { |
| value string |
| err string |
| }{{ |
| value: `_|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `"Hello world!"`, |
| }, { |
| value: `string`, |
| err: "", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| err := getInstance(t, tc.value).Value().Err() |
| checkErr(t, err, tc.err, "init") |
| }) |
| } |
| } |
| |
| func TestNull(t *testing.T) { |
| testCases := []struct { |
| value string |
| err string |
| }{{ |
| value: `v: _|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `v: "str"`, |
| err: "cannot use value \"str\" (type string) as null", |
| }, { |
| value: `v: null`, |
| }, { |
| value: `v: _`, |
| err: "non-concrete value _", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| err := getInstance(t, tc.value).Lookup("v").Null() |
| checkErr(t, err, tc.err, "init") |
| }) |
| } |
| } |
| |
| func TestBool(t *testing.T) { |
| testCases := []struct { |
| value string |
| bool bool |
| err string |
| }{{ |
| value: `_|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `"str"`, |
| err: "cannot use value \"str\" (type string) as bool", |
| }, { |
| value: `true`, |
| bool: true, |
| }, { |
| value: `false`, |
| }, { |
| value: `bool`, |
| err: "non-concrete value bool", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| got, err := getInstance(t, tc.value).Value().Bool() |
| if checkErr(t, err, tc.err, "init") { |
| if got != tc.bool { |
| t.Errorf("got %v; want %v", got, tc.bool) |
| } |
| } |
| }) |
| } |
| } |
| |
| func TestList(t *testing.T) { |
| testCases := []struct { |
| value string |
| res string |
| err string |
| }{{ |
| value: `_|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `"str"`, |
| err: "cannot use value \"str\" (type string) as list", |
| }, { |
| value: `[]`, |
| res: "[]", |
| }, { |
| value: `[1,2,3]`, |
| res: "[1,2,3,]", |
| }, { |
| value: `[for x in #y if x > 1 { x }] |
| #y: [1,2,3]`, |
| res: "[2,3,]", |
| }, { |
| value: `[int]`, |
| err: "cannot convert incomplete value", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| l, err := getInstance(t, tc.value).Value().List() |
| checkFatal(t, err, tc.err, "init") |
| |
| buf := []byte{'['} |
| for l.Next() { |
| b, err := l.Value().MarshalJSON() |
| checkFatal(t, err, tc.err, "list.Value") |
| buf = append(buf, b...) |
| buf = append(buf, ',') |
| } |
| buf = append(buf, ']') |
| if got := string(buf); got != tc.res { |
| t.Errorf("got %v; want %v", got, tc.res) |
| } |
| }) |
| } |
| } |
| |
| func TestFields(t *testing.T) { |
| testCases := []struct { |
| value string |
| res string |
| err string |
| }{{ |
| value: `{ #def: 1, _hidden: 2, opt?: 3, reg: 4 }`, |
| res: "{reg:4,}", |
| }, { |
| value: `_|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `"str"`, |
| err: "cannot use value \"str\" (type string) as struct", |
| }, { |
| value: `{}`, |
| res: "{}", |
| }, { |
| value: `{a:1,b:2,c:3}`, |
| res: "{a:1,b:2,c:3,}", |
| }, { |
| value: `{a:1,"_b":2,c:3,_d:4}`, |
| res: `{a:1,"_b":2,c:3,}`, |
| }, { |
| value: `{_a:"a"}`, |
| res: "{}", |
| }, { |
| value: `{ for k, v in #y if v > 1 {"\(k)": v} } |
| #y: {a:1,b:2,c:3}`, |
| res: "{b:2,c:3,}", |
| }, { |
| value: `{ #def: 1, _hidden: 2, opt?: 3, reg: 4 }`, |
| res: "{reg:4,}", |
| }, { |
| value: `{a:1,b:2,c:int}`, |
| err: "cannot convert incomplete value", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| obj := getInstance(t, tc.value).Value() |
| |
| iter, err := obj.Fields() |
| checkFatal(t, err, tc.err, "init") |
| |
| buf := []byte{'{'} |
| for iter.Next() { |
| buf = append(buf, iter.Selector().String()...) |
| buf = append(buf, ':') |
| b, err := iter.Value().MarshalJSON() |
| checkFatal(t, err, tc.err, "Obj.At") |
| buf = append(buf, b...) |
| buf = append(buf, ',') |
| } |
| buf = append(buf, '}') |
| if got := string(buf); got != tc.res { |
| t.Errorf("got %v; want %v", got, tc.res) |
| } |
| |
| iter, _ = obj.Fields() |
| for iter.Next() { |
| want, err := iter.Value().MarshalJSON() |
| checkFatal(t, err, tc.err, "Obj.At2") |
| |
| got, err := obj.Lookup(iter.Label()).MarshalJSON() |
| checkFatal(t, err, tc.err, "Obj.At2") |
| |
| if !bytes.Equal(got, want) { |
| t.Errorf("Lookup: got %q; want %q", got, want) |
| } |
| } |
| v := obj.Lookup("non-existing") |
| checkErr(t, v.Err(), "not found", "non-existing") |
| }) |
| } |
| } |
| |
| func TestAllFields(t *testing.T) { |
| testCases := []struct { |
| value string |
| res string |
| err string |
| }{{ |
| value: `{a:1,"_b":2,c:3,_d:4}`, |
| res: "{a:1,_b:2,c:3,_d:4,}", |
| }, { |
| value: `{_a:"a"}`, |
| res: `{_a:"a",}`, |
| }, { |
| value: `{_a:"a", b?: "b", #c: 3}`, |
| res: `{_a:"a",b?:"b",#c:3,}`, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| obj := getInstance(t, tc.value).Value() |
| |
| var iter *Iterator // Verify that the returned iterator is a pointer. |
| iter, err := obj.Fields(All()) |
| checkFatal(t, err, tc.err, "init") |
| |
| buf := []byte{'{'} |
| for iter.Next() { |
| buf = append(buf, iter.Label()...) |
| if iter.IsOptional() { |
| buf = append(buf, '?') |
| } |
| buf = append(buf, ':') |
| b, err := iter.Value().MarshalJSON() |
| checkFatal(t, err, tc.err, "Obj.At") |
| buf = append(buf, b...) |
| buf = append(buf, ',') |
| } |
| buf = append(buf, '}') |
| if got := string(buf); got != tc.res { |
| t.Errorf("got %v; want %v", got, tc.res) |
| } |
| }) |
| } |
| } |
| |
| func TestLookup(t *testing.T) { |
| var runtime = new(Runtime) |
| inst, err := runtime.Compile("x.cue", ` |
| #V: { |
| x: int |
| } |
| #X: { |
| [string]: int64 |
| } & #V |
| v: #X |
| `) |
| if err != nil { |
| t.Fatalf("compile: %v", err) |
| } |
| // expr, err := parser.ParseExpr("lookup.cue", `v`, parser.DeclarationErrors, parser.AllErrors) |
| // if err != nil { |
| // log.Fatalf("parseExpr: %v", err) |
| // } |
| // v := inst.Eval(expr) |
| testCases := []struct { |
| ref []string |
| raw string |
| eval string |
| }{{ |
| ref: []string{"v", "x"}, |
| raw: ">=-9223372036854775808 & <=9223372036854775807 & int", |
| eval: "int64", |
| }} |
| for _, tc := range testCases { |
| v := inst.Lookup(tc.ref...) |
| |
| if got := fmt.Sprintf("%#v", v); got != tc.raw { |
| t.Errorf("got %v; want %v", got, tc.raw) |
| } |
| |
| got := fmt.Sprint(astinternal.DebugStr(v.Eval().Syntax())) |
| if got != tc.eval { |
| t.Errorf("got %v; want %v", got, tc.eval) |
| } |
| |
| v = inst.Lookup() |
| for _, ref := range tc.ref { |
| s, err := v.Struct() |
| if err != nil { |
| t.Fatal(err) |
| } |
| fi, err := s.FieldByName(ref, false) |
| if err != nil { |
| t.Fatal(err) |
| } |
| v = fi.Value |
| } |
| |
| if got := fmt.Sprintf("%#v", v); got != tc.raw { |
| t.Errorf("got %v; want %v", got, tc.raw) |
| } |
| |
| got = fmt.Sprint(astinternal.DebugStr(v.Eval().Syntax())) |
| if got != tc.eval { |
| t.Errorf("got %v; want %v", got, tc.eval) |
| } |
| } |
| } |
| |
| func compileT(t *testing.T, r *Runtime, s string) *Instance { |
| t.Helper() |
| inst, err := r.Compile("", s) |
| if err != nil { |
| t.Fatal(err) |
| } |
| return inst |
| } |
| |
| func goValue(v Value) interface{} { |
| var x interface{} |
| err := v.Decode(&x) |
| if err != nil { |
| return err |
| } |
| return x |
| } |
| |
| // TODO: Exporting of Vertex as Conjunct |
| func TestFill(t *testing.T) { |
| r := &Runtime{} |
| |
| inst, err := r.CompileExpr(ast.NewStruct("bar", ast.NewString("baz"))) |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| testCases := []struct { |
| in string |
| x interface{} |
| path string // comma-separated path |
| out string |
| }{{ |
| in: ` |
| foo: int |
| bar: foo |
| `, |
| x: 3, |
| path: "foo", |
| out: ` |
| foo: 3 |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| string |
| `, |
| x: "foo", |
| path: "", |
| out: ` |
| "foo" |
| `, |
| }, { |
| in: ` |
| foo: _ |
| `, |
| x: inst.Value(), |
| path: "foo", |
| out: ` |
| {foo: {bar: "baz"}} |
| `, |
| }} |
| |
| for _, tc := range testCases { |
| var path []string |
| if tc.path != "" { |
| path = strings.Split(tc.path, ",") |
| } |
| |
| v := compileT(t, r, tc.in).Value() |
| v = v.Fill(tc.x, path...) |
| |
| w := compileT(t, r, tc.out).Value() |
| |
| if !cmp.Equal(goValue(v), goValue(w)) { |
| t.Error(cmp.Diff(goValue(v), goValue(w))) |
| t.Errorf("\ngot: %s\nwant: %s", v, w) |
| } |
| } |
| } |
| |
| func TestFill2(t *testing.T) { |
| r := &Runtime{} |
| |
| root, err := r.Compile("test", ` |
| #Provider: { |
| ID: string |
| notConcrete: bool |
| a: int |
| b: int |
| } |
| `) |
| |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| spec := root.LookupDef("#Provider") |
| providerInstance := spec.Fill("12345", "ID") |
| root, err = root.Fill(providerInstance, "providers", "myprovider") |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| got := fmt.Sprintf("%#v", root.Value()) |
| want := `#Provider: { |
| ID: string |
| notConcrete: bool |
| a: int |
| b: int |
| } |
| providers: { |
| myprovider: { |
| ID: "12345" |
| notConcrete: bool |
| a: int |
| b: int |
| } |
| }` |
| if got != want { |
| t.Errorf("got: %s\nwant: %s", got, want) |
| } |
| } |
| |
| func TestFillPath(t *testing.T) { |
| r := &Runtime{} |
| |
| inst, err := r.CompileExpr(ast.NewStruct("bar", ast.NewString("baz"))) |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| testCases := []struct { |
| in string |
| x interface{} |
| path Path |
| out string |
| }{{ |
| in: ` |
| foo: int |
| bar: foo |
| `, |
| x: 3, |
| path: ParsePath("foo"), |
| out: ` |
| foo: 3 |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| X="#foo": int |
| bar: X |
| `, |
| x: 3, |
| path: ParsePath(`"#foo"`), |
| out: ` |
| "#foo": 3 |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| X="#foo": foo: int |
| bar: X.foo |
| `, |
| x: 3, |
| path: ParsePath(`"#foo".foo`), |
| out: ` |
| "#foo": foo: 3 |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| foo: #foo: int |
| bar: foo.#foo |
| `, |
| x: 3, |
| path: ParsePath("foo.#foo"), |
| out: ` |
| foo: { |
| #foo: 3 |
| } |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| foo: _foo: int |
| bar: foo._foo |
| `, |
| x: 3, |
| path: MakePath(Str("foo"), Hid("_foo", "_")), |
| out: ` |
| foo: { |
| _foo: 3 |
| } |
| bar: 3 |
| `, |
| }, { |
| in: ` |
| string |
| `, |
| x: "foo", |
| path: ParsePath(""), |
| out: ` |
| "foo" |
| `, |
| }, { |
| in: ` |
| foo: _ |
| `, |
| x: inst.Value(), |
| path: ParsePath("foo"), |
| out: ` |
| {foo: {bar: "baz"}} |
| `, |
| }, { |
| // Resolve to enclosing |
| in: ` |
| foo: _ |
| x: 1 |
| `, |
| x: ast.NewIdent("x"), |
| path: ParsePath("foo"), |
| out: ` |
| {foo: 1, x: 1} |
| `, |
| }, { |
| in: ` |
| foo: { |
| bar: _ |
| x: 1 |
| } |
| `, |
| x: ast.NewIdent("x"), |
| path: ParsePath("foo.bar"), |
| out: ` |
| {foo: {bar: 1, x: 1}} |
| `, |
| }, { |
| // Resolve one scope up |
| in: ` |
| x: 1 |
| foo: { |
| bar: _ |
| } |
| `, |
| x: ast.NewIdent("x"), |
| path: ParsePath("foo.bar"), |
| out: ` |
| {foo: {bar: 1}, x: 1} |
| `, |
| }, { |
| // Resolve within ast expression |
| in: ` |
| foo: { |
| bar: _ |
| } |
| `, |
| x: ast.NewStruct( |
| ast.NewIdent("x"), ast.NewString("1"), |
| ast.NewIdent("y"), ast.NewIdent("x"), |
| ), |
| path: ParsePath("foo.bar"), |
| out: ` |
| {foo: {bar: {x: "1", y: "1"}}} |
| `, |
| }, { |
| // Resolve in non-existing |
| in: ` |
| foo: x: 1 |
| `, |
| x: ast.NewIdent("x"), |
| path: ParsePath("foo.bar.baz"), |
| out: ` |
| {foo: {x: 1, bar: baz: 1}} |
| `, |
| }, { |
| // empty path |
| in: ` |
| _ |
| #foo: 1 |
| `, |
| x: ast.NewIdent("#foo"), |
| out: `{1, #foo: 1}`, |
| }, { |
| in: `[...int]`, |
| x: 1, |
| path: ParsePath("0"), |
| out: `[1]`, |
| }, { |
| in: `[1, ...int]`, |
| x: 1, |
| path: ParsePath("1"), |
| out: `[1, 1]`, |
| }, { |
| in: `a: {b: v: int, c: v: int}`, |
| x: 1, |
| path: MakePath(Str("a"), AnyString, Str("v")), |
| out: `{ |
| a: { |
| b: { |
| v: 1 |
| } |
| c: { |
| v: 1 |
| } |
| } |
| }`, |
| }, { |
| in: `a: [_]`, |
| x: 1, |
| path: MakePath(Str("a"), AnyIndex, Str("b")), |
| out: `{ |
| a: [{ |
| b: 1 |
| }] |
| }`, |
| }, { |
| in: `a: 1`, |
| x: 1, |
| path: MakePath(Str("b").Optional()), |
| out: `{a: 1}`, |
| }, { |
| in: `b: int`, |
| x: 1, |
| path: MakePath(Str("b").Optional()), |
| out: `{b: 1}`, |
| }} |
| |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| v := compileT(t, r, tc.in).Value() |
| v = v.FillPath(tc.path, tc.x) |
| |
| w := compileT(t, r, tc.out).Value() |
| |
| if !cmp.Equal(goValue(v), goValue(w)) { |
| t.Error(cmp.Diff(goValue(v), goValue(w))) |
| t.Errorf("\ngot: %s\nwant: %s", v, w) |
| } |
| }) |
| } |
| } |
| |
| func TestFillPathError(t *testing.T) { |
| r := &Runtime{} |
| |
| type key struct{ a int } |
| |
| testCases := []struct { |
| in string |
| x interface{} |
| path Path |
| err string |
| }{{ |
| // unsupported type. |
| in: `_`, |
| x: make(chan int), |
| err: "unsupported Go type (chan int)", |
| }} |
| |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| v := compileT(t, r, tc.in).Value() |
| v = v.FillPath(tc.path, tc.x) |
| |
| err := v.Err() |
| if err == nil { |
| t.Errorf("unexpected success") |
| } |
| |
| if got := err.Error(); !strings.Contains(got, tc.err) { |
| t.Errorf("\ngot: %s\nwant: %s", got, tc.err) |
| } |
| }) |
| } |
| } |
| |
| func TestAllows(t *testing.T) { |
| r := &Runtime{} |
| |
| testCases := []struct { |
| desc string |
| in string |
| sel Selector |
| allow bool |
| }{{ |
| desc: "allow new field in open struct", |
| in: ` |
| x: { |
| a: int |
| } |
| `, |
| sel: Str("b"), |
| allow: true, |
| }, { |
| desc: "disallow new field in definition", |
| in: ` |
| x: #Def |
| #Def: { |
| a: int |
| } |
| `, |
| sel: Str("b"), |
| }, { |
| desc: "disallow new field in explicitly closed struct", |
| in: ` |
| x: close({ |
| a: int |
| }) |
| `, |
| sel: Str("b"), |
| }, { |
| desc: "allow index in open list", |
| in: ` |
| x: [...int] |
| `, |
| sel: Index(100), |
| allow: true, |
| }, { |
| desc: "disallow index in closed list", |
| in: ` |
| x: [] |
| `, |
| sel: Index(0), |
| }, { |
| desc: "allow existing index in closed list", |
| in: ` |
| x: [1] |
| `, |
| sel: Index(0), |
| allow: true, |
| }, { |
| desc: "definition in non-def closed list", |
| in: ` |
| x: [1] |
| `, |
| sel: Def("#foo"), |
| allow: true, |
| }, { |
| // TODO(disallow) |
| desc: "definition in def open list", |
| in: ` |
| x: #Def |
| x: [1] |
| #Def: [...int] |
| `, |
| sel: Def("#foo"), |
| allow: true, |
| }, { |
| desc: "field in def open list", |
| in: ` |
| x: #Def |
| x: [1] |
| #Def: [...int] |
| `, |
| sel: Str("foo"), |
| }, { |
| desc: "definition in open scalar", |
| in: ` |
| x: 1 |
| `, |
| sel: Def("#foo"), |
| allow: true, |
| }, { |
| desc: "field in scalar", |
| in: ` |
| x: #Def |
| x: 1 |
| #Def: int |
| `, |
| sel: Str("foo"), |
| }, { |
| desc: "any index in closed list", |
| in: ` |
| x: [1] |
| `, |
| sel: AnyIndex, |
| }, { |
| desc: "any index in open list", |
| in: ` |
| x: [...int] |
| `, |
| sel: AnyIndex, |
| allow: true, |
| }, { |
| desc: "definition in open scalar", |
| in: ` |
| x: 1 |
| `, |
| sel: anyDefinition, |
| allow: true, |
| }, { |
| desc: "field in open scalar", |
| in: ` |
| x: 1 |
| `, |
| sel: AnyString, |
| |
| // TODO(v0.6.0) |
| // }, { |
| // desc: "definition in closed scalar", |
| // in: ` |
| // x: #Def |
| // x: 1 |
| // #Def: int |
| // `, |
| // sel: AnyDefinition, |
| // allow: true, |
| }, { |
| desc: "allow field in any", |
| in: ` |
| x: _ |
| `, |
| sel: AnyString, |
| allow: true, |
| }, { |
| desc: "allow index in any", |
| in: ` |
| x: _ |
| `, |
| sel: AnyIndex, |
| allow: true, |
| }, { |
| desc: "allow index in disjunction", |
| in: ` |
| x: [...int] | 1 |
| `, |
| sel: AnyIndex, |
| allow: true, |
| }, { |
| desc: "allow index in disjunction", |
| in: ` |
| x: [] | [...int] |
| `, |
| sel: AnyIndex, |
| allow: true, |
| }, { |
| desc: "disallow index in disjunction", |
| in: ` |
| x: [1, 2] | [3, 2] |
| `, |
| sel: AnyIndex, |
| }, { |
| desc: "disallow index in non-list disjunction", |
| in: ` |
| x: "foo" | 1 |
| `, |
| sel: AnyIndex, |
| }, { |
| desc: "allow label in disjunction", |
| in: ` |
| x: {} | 1 |
| `, |
| sel: AnyString, |
| allow: true, |
| }, { |
| desc: "allow label in disjunction", |
| in: ` |
| x: #Def |
| #Def: { a: 1 } | { b: 1, ... } |
| `, |
| sel: AnyString, |
| allow: true, |
| }, { |
| desc: "disallow label in disjunction", |
| in: ` |
| x: #Def |
| #Def: { a: 1 } | { b: 1 } |
| `, |
| sel: AnyString, |
| }, { |
| desc: "pattern constraint", |
| in: ` |
| x: #PC |
| #PC: [>"m"]: int |
| `, |
| sel: Str(""), |
| }, { |
| desc: "pattern constraint", |
| in: ` |
| x: #PC |
| #PC: [>"m"]: int |
| `, |
| sel: Str("z"), |
| allow: true, |
| }, { |
| desc: "any in pattern constraint", |
| in: ` |
| x: #PC |
| #PC: [>"m"]: int |
| `, |
| sel: AnyString, |
| }, { |
| desc: "any in pattern constraint", |
| in: ` |
| x: #PC |
| #PC: [>" "]: int |
| `, |
| sel: AnyString, |
| }} |
| |
| path := ParsePath("x") |
| |
| for _, tc := range testCases { |
| t.Run(tc.desc, func(t *testing.T) { |
| v := compileT(t, r, tc.in).Value() |
| v = v.LookupPath(path) |
| |
| got := v.Allows(tc.sel) |
| if got != tc.allow { |
| t.Errorf("got %v; want %v", got, tc.allow) |
| } |
| }) |
| } |
| } |
| |
| func TestFillFloat(t *testing.T) { |
| // This tests panics for issue #749 |
| |
| want := `{ |
| x: 3.14 |
| }` |
| |
| filltest := func(x interface{}) { |
| r := &Runtime{} |
| i, err := r.Compile("test", ` |
| x: number |
| `) |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| i, err = i.Fill(x, "x") |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| got := fmt.Sprint(i.Value()) |
| if got != want { |
| t.Errorf("got: %s\nwant: %s", got, want) |
| } |
| } |
| |
| filltest(float32(3.14)) |
| filltest(float64(3.14)) |
| filltest(big.NewFloat(3.14)) |
| } |
| |
| func TestValue_LookupDef(t *testing.T) { |
| r := &Runtime{} |
| |
| testCases := []struct { |
| in string |
| def string // comma-separated path |
| exists bool |
| out string |
| }{{ |
| in: `#foo: 3`, |
| def: "#foo", |
| out: `3`, |
| }, { |
| in: `_foo: 3`, |
| def: "_foo", |
| out: `_|_ // field "#_foo" not found`, |
| }, { |
| in: `_#foo: 3`, |
| def: "_#foo", |
| out: `_|_ // field "_#foo" not found`, |
| }, { |
| in: `"foo", #foo: 3`, |
| def: "#foo", |
| out: `3`, |
| }} |
| |
| for _, tc := range testCases { |
| t.Run(tc.def, func(t *testing.T) { |
| v := compileT(t, r, tc.in).Value() |
| v = v.LookupDef(tc.def) |
| got := fmt.Sprint(v) |
| |
| if got != tc.out { |
| t.Errorf("\ngot: %s\nwant: %s", got, tc.out) |
| } |
| }) |
| } |
| } |
| |
| // TODO: trim down to individual defaults? |
| func TestDefaults(t *testing.T) { |
| testCases := []struct { |
| value string |
| def string |
| val string |
| ok bool |
| }{{ |
| value: `number | *1`, |
| def: "1", |
| val: "number", |
| ok: true, |
| }, { |
| value: `1 | 2 | *3`, |
| def: "3", |
| val: "1|2|3", |
| ok: true, |
| }, { |
| value: `*{a:1,b:2}|{a:1}|{b:2}`, |
| def: "{a:1,b:2}", |
| val: "{a: 1}|{b: 2}", |
| ok: true, |
| }, { |
| value: `{a:1}&{b:2}`, |
| def: `({a:1} & {b:2})`, |
| val: ``, |
| ok: false, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| v := getInstance(t, "a: "+tc.value).Lookup("a") |
| |
| d, ok := v.Default() |
| if ok != tc.ok { |
| t.Errorf("hasDefault: got %v; want %v", ok, tc.ok) |
| } |
| |
| if got := compactRawStr(d); got != tc.def { |
| t.Errorf("default: got %v; want %v", got, tc.def) |
| } |
| |
| op, val := d.Expr() |
| if op != OrOp { |
| return |
| } |
| vars := []string{} |
| for _, v := range val { |
| vars = append(vars, fmt.Sprint(v)) |
| } |
| if got := strings.Join(vars, "|"); got != tc.val { |
| t.Errorf("value: got %v; want %v", got, tc.val) |
| } |
| }) |
| } |
| } |
| |
| func TestLen(t *testing.T) { |
| testCases := []struct { |
| input string |
| length string |
| }{{ |
| input: "[1, 3]", |
| length: "2", |
| }, { |
| input: "[1, 3, ...]", |
| length: "int & >=2", |
| }, { |
| input: `"foo"`, |
| length: "3", |
| }, { |
| input: `'foo'`, |
| length: "3", |
| // TODO: Currently not supported. |
| // }, { |
| // input: "{a:1, b:3, a:1, c?: 3, _hidden: 4}", |
| // length: "2", |
| }, { |
| input: "3", |
| length: "_|_ // len not supported for type int", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.input, func(t *testing.T) { |
| v := getInstance(t, "a: "+tc.input).Lookup("a") |
| |
| length := v.Len() |
| if got := fmt.Sprint(length); got != tc.length { |
| t.Errorf("length: got %v; want %v", got, tc.length) |
| } |
| }) |
| } |
| } |
| |
| func TestTemplate(t *testing.T) { |
| testCases := []struct { |
| value string |
| path []string |
| want string |
| }{{ |
| value: ` |
| a: [Name=string]: Name |
| `, |
| path: []string{"a", ""}, |
| want: `"label"`, |
| }, { |
| value: ` |
| [Name=string]: { a: Name } |
| `, |
| path: []string{"", "a"}, |
| want: `"label"`, |
| }, { |
| value: ` |
| [Name=string]: { a: Name } |
| `, |
| path: []string{""}, |
| want: `{"a":"label"}`, |
| }, { |
| value: ` |
| a: [Foo=string]: [Bar=string]: { b: Foo+Bar } |
| `, |
| path: []string{"a", "", ""}, |
| want: `{"b":"labellabel"}`, |
| }, { |
| value: ` |
| a: [Foo=string]: b: [Bar=string]: { c: Foo+Bar } |
| a: foo: b: [Bar=string]: { d: Bar } |
| `, |
| path: []string{"a", "foo", "b", ""}, |
| want: `{"c":"foolabel","d":"label"}`, |
| }} |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| v := getInstance(t, tc.value).Value() |
| for _, p := range tc.path { |
| if p == "" { |
| v = v.Template()("label") |
| } else { |
| v = v.Lookup(p) |
| } |
| } |
| b, err := v.MarshalJSON() |
| if err != nil { |
| t.Fatal(err) |
| } |
| if got := string(b); got != tc.want { |
| t.Errorf("\n got: %q\nwant: %q", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func TestElem(t *testing.T) { |
| testCases := []struct { |
| value string |
| path []string |
| want string |
| }{{ |
| value: ` |
| a: [...int] |
| `, |
| path: []string{"a", ""}, |
| want: `int`, |
| }, { |
| value: ` |
| [Name=string]: { a: Name } |
| `, |
| path: []string{"", "a"}, |
| want: `string`, |
| }, { |
| value: ` |
| [Name=string]: { a: Name } |
| `, |
| path: []string{""}, |
| want: "{\n\ta: string\n}", |
| }, { |
| value: ` |
| a: [Foo=string]: [Bar=string]: { b: Foo+Bar } |
| `, |
| path: []string{"a", "", ""}, |
| want: "{\n\tb: string + string\n}", |
| }, { |
| value: ` |
| a: [Foo=string]: b: [Bar=string]: { c: Foo+Bar } |
| a: foo: b: [Bar=string]: { d: Bar } |
| `, |
| path: []string{"a", "foo", "b", ""}, |
| want: "{\n\tc: string + string\n\td: string\n}", |
| }} |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| v := getInstance(t, tc.value).Value() |
| v.v.Finalize(v.ctx()) // TODO: do in instance. |
| for _, p := range tc.path { |
| if p == "" { |
| var ok bool |
| v, ok = v.Elem() |
| if !ok { |
| t.Fatal("expected element") |
| } |
| } else { |
| v = v.Lookup(p) |
| } |
| } |
| got := fmt.Sprint(v) |
| // got := debug.NodeString(v.ctx(), v.v, &debug.Config{Compact: true}) |
| if got != tc.want { |
| t.Errorf("\n got: %q\nwant: %q", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func TestSubsume(t *testing.T) { |
| a := ParsePath("a") |
| b := ParsePath("b") |
| testCases := []struct { |
| value string |
| pathA Path |
| pathB Path |
| options []Option |
| want bool |
| }{{ |
| value: `4`, |
| want: true, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: b, |
| pathB: a, |
| want: false, |
| }, { |
| value: `a: {a: string, b: 4}, b: {a: "foo", b: 4}`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [string, 4], b: ["foo", 4]`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [...string], b: ["foo"]`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [...int], b: ["foo"]`, |
| pathA: a, |
| pathB: b, |
| want: false, |
| }, { |
| // Issue #566 |
| // Closed struct subsuming open struct. |
| value: ` |
| #Run: { action: "run", command: [...string] } |
| b: { action: "run", command: ["echo", "hello"] } |
| `, |
| pathA: ParsePath("#Run"), |
| pathB: b, |
| |
| // NOTE: this is for v0.2 compatibility. Logically a closed struct |
| // does not subsume an open struct. One could argue that the default |
| // of an open struct is the closed struct with the minimal number |
| // of fields that is an instance of it, though. |
| want: true, // open struct is not subsumed by closed if not final. |
| }, { |
| // Issue #566 |
| // Closed struct subsuming open struct. |
| value: ` |
| #Run: { action: "run", command: [...string] } |
| b: { action: "run", command: ["echo", "hello"] } |
| `, |
| pathA: ParsePath("#Run"), |
| pathB: b, |
| options: []Option{Final()}, |
| want: true, |
| }, { |
| // default |
| value: ` |
| a: <5 |
| b: *3 | int |
| `, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| // Disable default elimination. |
| value: ` |
| a: <5 |
| b: *3 | int |
| `, |
| pathA: a, |
| pathB: b, |
| options: []Option{Raw()}, |
| want: false, |
| }, { |
| value: ` |
| #A: { |
| exact: string |
| } | { |
| regex: string |
| } |
| #B: { |
| exact: string |
| } | { |
| regex: string |
| } |
| `, |
| pathA: ParsePath("#A"), |
| pathB: ParsePath("#B"), |
| options: []Option{}, |
| want: true, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| v := getInstance(t, tc.value) |
| a := v.Value().LookupPath(tc.pathA) |
| b := v.Value().LookupPath(tc.pathB) |
| got := a.Subsume(b, tc.options...) == nil |
| if got != tc.want { |
| t.Errorf("got %v (%v); want %v (%v)", got, a, tc.want, b) |
| } |
| }) |
| } |
| } |
| |
| func TestSubsumes(t *testing.T) { |
| a := []string{"a"} |
| b := []string{"b"} |
| testCases := []struct { |
| value string |
| pathA []string |
| pathB []string |
| want bool |
| }{{ |
| value: `4`, |
| want: true, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: b, |
| pathB: a, |
| want: false, |
| }, { |
| value: `a: {a: string, b: 4}, b: {a: "foo", b: 4}`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [string, 4], b: ["foo", 4]`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [...string], b: ["foo"]`, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }, { |
| value: `a: [...int], b: ["foo"]`, |
| pathA: a, |
| pathB: b, |
| want: false, |
| }, { |
| value: ` |
| a: { action: "run", command: [...string] } |
| b: { action: "run", command: ["echo", "hello"] } |
| `, |
| pathA: a, |
| pathB: b, |
| want: true, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| v := getInstance(t, tc.value) |
| a := v.Lookup(tc.pathA...) |
| b := v.Lookup(tc.pathB...) |
| got := a.Subsumes(b) |
| if got != tc.want { |
| t.Errorf("got %v (%v); want %v (%v)", got, a, tc.want, b) |
| } |
| }) |
| } |
| } |
| |
| func TestUnify(t *testing.T) { |
| a := []string{"a"} |
| b := []string{"b"} |
| testCases := []struct { |
| value string |
| pathA []string |
| pathB []string |
| want string |
| }{{ |
| value: `4`, |
| want: `4`, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: a, |
| pathB: b, |
| want: `"foo"`, |
| }, { |
| value: `a: string, b: "foo"`, |
| pathA: b, |
| pathB: a, |
| want: `"foo"`, |
| }, { |
| value: `a: {a: string, b: 4}, b: {a: "foo", b: 4}`, |
| pathA: a, |
| pathB: b, |
| want: `{"a":"foo","b":4}`, |
| }, { |
| value: `a: [string, 4], b: ["foo", 4]`, |
| pathA: a, |
| pathB: b, |
| want: `["foo",4]`, |
| }, { |
| value: `a: {a: string, _hidden: int, _#hidden: int}, b: close({a: "foo"})`, |
| pathA: a, |
| pathB: b, |
| want: `{"a":"foo"}`, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.value, func(t *testing.T) { |
| v := getInstance(t, tc.value).Value() |
| x := v.Lookup(tc.pathA...) |
| y := v.Lookup(tc.pathB...) |
| b, err := x.Unify(y).MarshalJSON() |
| if err != nil { |
| t.Fatal(err) |
| } |
| if got := string(b); got != tc.want { |
| t.Errorf("got %v; want %v", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func TestEquals(t *testing.T) { |
| testCases := []struct { |
| a, b string |
| want bool |
| }{{ |
| `4`, `4`, true, |
| }, { |
| `"str"`, `2`, false, |
| }, { |
| `2`, `3`, false, |
| }, { |
| `[1]`, `[3]`, false, |
| }, { |
| `[{a: 1,...}]`, `[{a: 1,...}]`, true, |
| }, { |
| `[]`, `[]`, true, |
| }, { |
| `{ |
| a: b, |
| b: a, |
| }`, |
| `{ |
| a: b, |
| b: a, |
| }`, |
| true, |
| }, { |
| `{ |
| a: "foo", |
| b: "bar", |
| }`, |
| `{ |
| a: "foo", |
| }`, |
| false, |
| }, { |
| // Ignore closedness |
| `{ #Foo: { k: 1 }, a: #Foo }`, |
| `{ #Foo: { k: 1 }, a: { k: 1 } }`, |
| true, |
| }, { |
| // Ignore optional fields |
| `{ #Foo: { k: 1 }, a: #Foo }`, |
| `{ #Foo: { k: 1 }, a: { #Foo, i?: 1 } }`, |
| true, |
| }, { |
| // Treat embedding as equal |
| `{ a: 2, b: { 3 } }`, |
| `{ a: { 2 }, b: 3 }`, |
| true, |
| }} |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| var r Runtime |
| a, err := r.Compile("a", tc.a) |
| if err != nil { |
| t.Fatal(err) |
| } |
| b, err := r.Compile("b", tc.b) |
| if err != nil { |
| t.Fatal(err) |
| } |
| got := a.Value().Equals(b.Value()) |
| if got != tc.want { |
| t.Errorf("got %v; want %v", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| // TODO: options: disallow cycles. |
| func TestValidate(t *testing.T) { |
| testCases := []struct { |
| desc string |
| in string |
| err bool |
| opts []Option |
| }{{ |
| desc: "issue #51", |
| in: ` |
| a: [string]: foo |
| a: b: {} |
| `, |
| err: true, |
| }, { |
| desc: "concrete", |
| in: ` |
| a: 1 |
| b: { c: 2, d: 3 } |
| c: d: e: f: 5 |
| g?: int |
| `, |
| opts: []Option{Concrete(true)}, |
| }, { |
| desc: "definition error", |
| in: ` |
| #b: 1 & 2 |
| `, |
| opts: []Option{}, |
| err: true, |
| }, { |
| desc: "definition error okay if optional", |
| in: ` |
| #b?: 1 & 2 |
| `, |
| opts: []Option{}, |
| }, { |
| desc: "definition with optional", |
| in: ` |
| #b: { |
| a: int |
| b?: >=0 |
| } |
| `, |
| opts: []Option{Concrete(true)}, |
| }, { |
| desc: "disjunction", |
| in: `a: 1 | 2`, |
| }, { |
| desc: "disjunction concrete", |
| in: `a: 1 | 2`, |
| opts: []Option{Concrete(true)}, |
| err: true, |
| }, { |
| desc: "incomplete concrete", |
| in: `a: string`, |
| }, { |
| desc: "incomplete", |
| in: `a: string`, |
| opts: []Option{Concrete(true)}, |
| err: true, |
| }, { |
| desc: "list", |
| in: `a: [{b: string}, 3]`, |
| }, { |
| desc: "list concrete", |
| in: `a: [{b: string}, 3]`, |
| opts: []Option{Concrete(true)}, |
| err: true, |
| }, { |
| desc: "allow cycles", |
| in: ` |
| a: b - 100 |
| b: a + 100 |
| c: [c[1], c[0]] |
| `, |
| }, { |
| desc: "disallow cycles", |
| in: ` |
| a: b - 100 |
| b: a + 100 |
| c: [c[1], c[0]] |
| `, |
| opts: []Option{DisallowCycles(true)}, |
| err: true, |
| }, { |
| desc: "builtins are okay", |
| in: ` |
| import "time" |
| |
| a: { b: time.Duration } | { c: time.Duration } |
| `, |
| }, { |
| desc: "comprehension error", |
| in: ` |
| a: { if b == "foo" { field: 2 } } |
| `, |
| err: true, |
| }, { |
| desc: "ignore optional in schema", |
| in: ` |
| #Schema1: { |
| a?: int |
| } |
| instance1: #Schema1 |
| `, |
| opts: []Option{Concrete(true)}, |
| }, { |
| desc: "issue324", |
| in: ` |
| import "encoding/yaml" |
| |
| x: string |
| a: b: c: *["\(x)"] | _ |
| d: yaml.Marshal(a.b) |
| `, |
| }, { |
| desc: "allow non-concrete values for definitions", |
| in: ` |
| variables: #variables |
| |
| {[!~"^[.]"]: #job} |
| |
| #variables: [string]: int | string |
| |
| #job: ({a: int} | {b: int}) & { |
| "variables"?: #variables |
| } |
| `, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.desc, func(t *testing.T) { |
| r := Runtime{} |
| inst, err := r.Parse("validate", tc.in) |
| if err == nil { |
| err = inst.Value().Validate(tc.opts...) |
| } |
| if gotErr := err != nil; gotErr != tc.err { |
| t.Errorf("got %v; want %v", err, tc.err) |
| } |
| }) |
| } |
| } |
| |
| func TestPath(t *testing.T) { |
| config := ` |
| a: b: c: 5 |
| b: { |
| b1: 3 |
| b2: 4 |
| "b 3": 5 |
| "4b": 6 |
| l: [ |
| {a: 2}, |
| {c: 2}, |
| ] |
| } |
| ` |
| mkpath := func(p ...string) []string { return p } |
| testCases := [][]string{ |
| mkpath("a", "b", "c"), |
| mkpath("b", "l", "1", "c"), |
| mkpath("b", `"b 3"`), |
| mkpath("b", `"4b"`), |
| } |
| for _, tc := range testCases { |
| r := Runtime{} |
| inst, err := r.Parse("config", config) |
| if err != nil { |
| t.Fatal(err) |
| } |
| t.Run(strings.Join(tc, "."), func(t *testing.T) { |
| v := inst.Lookup(tc[0]) |
| for _, e := range tc[1:] { |
| if '0' <= e[0] && e[0] <= '9' { |
| i, err := strconv.Atoi(e) |
| if err != nil { |
| t.Fatal(err) |
| } |
| iter, err := v.List() |
| if err != nil { |
| t.Fatal(err) |
| } |
| for c := 0; iter.Next(); c++ { |
| if c == i { |
| v = iter.Value() |
| break |
| } |
| } |
| } else if e[0] == '"' { |
| v = v.Lookup(e[1 : len(e)-1]) |
| } else { |
| v = v.Lookup(e) |
| } |
| } |
| got := pathToStrings(v.Path()) |
| if !reflect.DeepEqual(got, tc) { |
| t.Errorf("got %v; want %v", got, tc) |
| } |
| }) |
| } |
| } |
| |
| func TestValueLookup(t *testing.T) { |
| config := ` |
| a: { |
| a: 0 |
| b: 1 |
| c: 2 |
| } |
| b: { |
| d: a.a |
| e: int |
| } |
| ` |
| |
| strList := func(s ...string) []string { return s } |
| |
| testCases := []struct { |
| config string |
| path []string |
| str string |
| notExists bool |
| }{{ |
| config: "_|_", |
| path: strList(""), |
| str: "explicit error (_|_ literal) in source", |
| }, { |
| config: "_|_", |
| path: strList("a"), |
| str: "explicit error (_|_ literal) in source", |
| }, { |
| config: config, |
| path: strList(), |
| str: "{a:{a:0,b:1,c:2},b:{d:0,e:int}", |
| }, { |
| config: config, |
| path: strList("a", "a"), |
| str: "0", |
| }, { |
| config: config, |
| path: strList("a"), |
| str: "{a:0,b:1,c:2}", |
| }, { |
| config: config, |
| path: strList("b", "d"), |
| str: "0", |
| }, { |
| config: config, |
| path: strList("c", "non-existing"), |
| str: "not found", |
| notExists: true, |
| }, { |
| config: config, |
| path: strList("b", "d", "lookup in non-struct"), |
| str: "cannot use value 0 (type int) as struct", |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.str, func(t *testing.T) { |
| v := getInstance(t, tc.config).Value().Lookup(tc.path...) |
| if got := !v.Exists(); got != tc.notExists { |
| t.Errorf("exists: got %v; want %v", got, tc.notExists) |
| } |
| |
| got := v.ctx().Str(v.v) |
| if tc.str == "" { |
| t.Fatalf("str empty, got %q", got) |
| } |
| if !strings.Contains(got, tc.str) { |
| t.Errorf("\n got %v\nwant %v", got, tc.str) |
| } |
| }) |
| } |
| } |
| |
| func cmpError(a, b error) bool { |
| if a == nil { |
| return b == nil |
| } |
| if b == nil { |
| return a == nil |
| } |
| return a.Error() == b.Error() |
| } |
| |
| // TODO: duplicate docs. |
| func TestValueDoc(t *testing.T) { |
| const config = ` |
| // foobar defines at least foo. |
| package foobar |
| |
| // A Foo fooses stuff. |
| Foo: { |
| // field1 is an int. |
| field1: int |
| |
| field2: int |
| |
| // duplicate field comment |
| dup3: int |
| } |
| |
| // foos are instances of Foo. |
| foos: [string]: Foo |
| |
| // My first little foo. |
| foos: MyFoo: { |
| // local field comment. |
| field1: 0 |
| |
| // Dangling comment. |
| |
| // other field comment. |
| field2: 1 |
| |
| // duplicate field comment |
| dup3: int |
| } |
| |
| bar: { |
| // comment from bar on field 1 |
| field1: int |
| // comment from bar on field 2 |
| field2: int // don't include this |
| } |
| |
| baz: bar & { |
| // comment from baz on field 1 |
| field1: int |
| field2: int |
| } |
| ` |
| config2 := ` |
| // Another Foo. |
| Foo: {} |
| ` |
| var r Runtime |
| getInst := func(name, body string) *Instance { |
| inst, err := r.Compile("dir/file1.cue", body) |
| if err != nil { |
| t.Fatal(err) |
| } |
| return inst |
| } |
| |
| inst := getInst("config", config) |
| |
| v1 := inst.Value() |
| v2 := getInst("config2", config2).Value() |
| both := v1.Unify(v2) |
| |
| testCases := []struct { |
| val Value |
| path string |
| doc string |
| }{{ |
| val: v1, |
| path: "foos", |
| doc: "foos are instances of Foo.\n", |
| }, { |
| val: v1, |
| path: "foos MyFoo", |
| doc: "My first little foo.\n", |
| }, { |
| val: v1, |
| path: "foos MyFoo field1", |
| doc: `local field comment. |
| |
| field1 is an int. |
| `, |
| }, { |
| val: v1, |
| path: "foos MyFoo field2", |
| doc: "other field comment.\n", |
| }, { |
| // Duplicates are now removed. |
| val: v1, |
| path: "foos MyFoo dup3", |
| doc: "duplicate field comment\n", |
| }, { |
| val: v1, |
| path: "bar field1", |
| doc: "comment from bar on field 1\n", |
| }, { |
| val: v1, |
| path: "baz field1", |
| doc: `comment from bar on field 1 |
| |
| comment from baz on field 1 |
| `, |
| }, { |
| val: v1, |
| path: "baz field2", |
| doc: "comment from bar on field 2\n", |
| }, { |
| val: v2, |
| path: "Foo", |
| doc: `Another Foo. |
| `, |
| }, { |
| val: both, |
| path: "Foo", |
| doc: `A Foo fooses stuff. |
| |
| Another Foo. |
| `, |
| }} |
| for _, tc := range testCases { |
| t.Run("field:"+tc.path, func(t *testing.T) { |
| v := tc.val.Lookup(strings.Split(tc.path, " ")...) |
| doc := docStr(v.Doc()) |
| if doc != tc.doc { |
| t.Errorf("doc: got:\n%vwant:\n%v", doc, tc.doc) |
| } |
| }) |
| } |
| want := "foobar defines at least foo.\n" |
| if got := docStr(inst.Value().Doc()); got != want { |
| t.Errorf("pkg: got:\n%vwant:\n%v", got, want) |
| } |
| } |
| |
| func docStr(docs []*ast.CommentGroup) string { |
| doc := "" |
| for _, d := range docs { |
| if doc != "" { |
| doc += "\n" |
| } |
| doc += d.Text() |
| } |
| return doc |
| } |
| |
| // TODO: unwrap marshal error |
| // TODO: improve error messages |
| func TestMarshalJSON(t *testing.T) { |
| testCases := []struct { |
| value string |
| json string |
| err string |
| }{{ |
| value: `""`, |
| json: `""`, |
| }, { |
| value: `null`, |
| json: `null`, |
| }, { |
| value: `_|_`, |
| err: "explicit error (_|_ literal) in source", |
| }, { |
| value: `(a.b) |
| a: {}`, |
| err: "undefined field", |
| }, { |
| value: `true`, |
| json: `true`, |
| }, { |
| value: `false`, |
| json: `false`, |
| }, { |
| value: `bool`, |
| err: "cannot convert incomplete value", |
| }, { |
| value: `"str"`, |
| json: `"str"`, |
| }, { |
| value: `12_000`, |
| json: `12000`, |
| }, { |
| value: `12.000`, |
| json: `12.000`, |
| }, { |
| value: `12M`, |
| json: `12000000`, |
| }, { |
| value: `3.0e100`, |
| json: `3.0E+100`, |
| }, { |
| value: `0/0`, |
| err: "division undefined", |
| }, { |
| value: `[]`, |
| json: `[]`, |
| }, { |
| value: `[1, 2, 3]`, |
| json: `[1,2,3]`, |
| }, { |
| value: `[int]`, |
| err: `0: cannot convert incomplete value`, |
| }, { |
| value: `{}`, |
| json: `{}`, |
| }, { |
| value: `{a: 2, b: 3, c: ["A", "B"]}`, |
| json: `{"a":2,"b":3,"c":["A","B"]}`, |
| }, { |
| value: `{a: 2, b: 3, c: [string, "B"]}`, |
| err: `c.0: cannot convert incomplete value`, |
| }, { |
| value: `{a: [{b: [0, {c: string}] }] }`, |
| err: `a.0.b.1.c: cannot convert incomplete value`, |
| }, { |
| value: `{foo?: 1, bar?: 2, baz: 3}`, |
| json: `{"baz":3}`, |
| }, { |
| // Has an unresolved cycle, but should not matter as all fields involved |
| // are optional |
| value: `{foo?: bar, bar?: foo, baz: 3}`, |
| json: `{"baz":3}`, |
| }, { |
| // Issue #107 |
| value: `a: 1.0/1`, |
| json: `{"a":1.0}`, |
| }, { |
| // Issue #108 |
| value: ` |
| a: int |
| a: >0 |
| a: <2 |
| |
| b: int |
| b: >=0.9 |
| b: <1.1 |
| |
| c: int |
| c: >1 |
| c: <=2 |
| |
| d: int |
| d: >=1 |
| d: <=1.5 |
| |
| e: int |
| e: >=1 |
| e: <=1.32 |
| |
| f: >=1.1 & <=1.1 |
| `, |
| json: `{"a":1,"b":1,"c":2,"d":1,"e":1,"f":1.1}`, |
| }, { |
| value: ` |
| #Task: { |
| { |
| op: "pull" |
| tag: *"latest" | string |
| tagInString: tag + "dd" |
| } | { |
| op: "scratch" |
| } |
| } |
| |
| foo: #Task & {"op": "pull"} |
| `, |
| json: `{"foo":{"op":"pull","tag":"latest","tagInString":"latestdd"}}`, |
| }, { |
| // Issue #326 |
| value: `x: "\(string)": "v"`, |
| err: `x: invalid interpolation`, |
| }, { |
| // Issue #326 |
| value: `x: "\(bool)": "v"`, |
| err: `invalid interpolation`, |
| }, { |
| // Issue #326 |
| value: ` |
| x: { |
| for k, v in y { |
| "\(k)": v |
| } |
| } |
| y: {} |
| `, |
| json: `{"x":{},"y":{}}`, |
| }, { |
| // Issue #326 |
| value: ` |
| x: { |
| for k, v in y { |
| "\(k)": v |
| } |
| } |
| y: _ |
| `, |
| err: `x: cannot range over y (incomplete type _)`, |
| }} |
| for i, tc := range testCases { |
| t.Run(fmt.Sprintf("%d/%v", i, tc.value), func(t *testing.T) { |
| inst := getInstance(t, tc.value) |
| b, err := inst.Value().MarshalJSON() |
| checkFatal(t, err, tc.err, "init") |
| |
| if got := string(b); got != tc.json { |
| t.Errorf("\n got %v;\nwant %v", got, tc.json) |
| } |
| }) |
| } |
| } |
| |
| func TestWalk(t *testing.T) { |
| testCases := []struct { |
| value string |
| out string |
| }{{ |
| value: `""`, |
| out: `""`, |
| }, { |
| value: `null`, |
| out: `null`, |
| }, { |
| value: `_|_`, |
| out: "_|_(explicit error (_|_ literal) in source)", |
| }, { |
| value: `(a.b) |
| a: {}`, |
| out: `_|_(undefined field b)`, |
| }, { |
| value: `true`, |
| out: `true`, |
| }, { |
| value: `false`, |
| out: `false`, |
| }, { |
| value: `bool`, |
| out: "bool", |
| }, { |
| value: `"str"`, |
| out: `"str"`, |
| }, { |
| value: `12_000`, |
| out: `12000`, |
| // out: `12_000`, |
| }, { |
| value: `12.000`, |
| out: `12.000`, |
| }, { |
| value: `12M`, |
| out: `12000000`, |
| // out: `12M`, |
| }, { |
| value: `3.0e100`, |
| out: `3.0e+100`, |
| // out: `3.0e100`, |
| }, { |
| value: `[]`, |
| out: `[]`, |
| }, { |
| value: `[1, 2, 3]`, |
| out: `[1,2,3]`, |
| }, { |
| value: `[int]`, |
| out: `[int]`, |
| }, { |
| value: `3 * [1, 2]`, |
| out: `[1,2,1,2,1,2]`, |
| }, { |
| value: `{}`, |
| out: `{}`, |
| }, { |
| value: `{a: 2, b: 3, c: ["A", "B"]}`, |
| out: `{a:2,b:3,c:["A","B"]}`, |
| }} |
| for i, tc := range testCases { |
| t.Run(fmt.Sprintf("%d/%v", i, tc.value), func(t *testing.T) { |
| inst := getInstance(t, tc.value) |
| buf := []byte{} |
| stripComma := func() { |
| if n := len(buf) - 1; buf[n] == ',' { |
| buf = buf[:n] |
| } |
| } |
| inst.Value().Walk(func(v Value) bool { |
| v = v.Eval() |
| if !v.v.Label.IsInt() { |
| if k, ok := v.Label(); ok { |
| buf = append(buf, k+":"...) |
| } |
| } |
| switch v.Kind() { |
| case StructKind: |
| buf = append(buf, '{') |
| case ListKind: |
| buf = append(buf, '[') |
| default: |
| if b, _ := v.v.BaseValue.(*adt.Bottom); b != nil { |
| s := debugStr(v.ctx(), b) |
| buf = append(buf, fmt.Sprint(s, ",")...) |
| return true |
| } |
| buf = append(buf, fmt.Sprint(v, ",")...) |
| } |
| return true |
| }, func(v Value) { |
| switch v.Kind() { |
| case StructKind: |
| stripComma() |
| buf = append(buf, "},"...) |
| case ListKind: |
| stripComma() |
| buf = append(buf, "],"...) |
| } |
| }) |
| stripComma() |
| if got := string(buf); got != tc.out { |
| t.Errorf("\n got %v;\nwant %v", got, tc.out) |
| } |
| }) |
| } |
| } |
| |
| func TestTrimZeros(t *testing.T) { |
| testCases := []struct { |
| in string |
| out string |
| }{ |
| {"", ""}, |
| {"2", "2"}, |
| {"2.0", "2.0"}, |
| {"2.000000000000", "2.0"}, |
| {"2000000000000", "2e+12"}, |
| {"2000000", "2e+6"}, |
| } |
| for _, tc := range testCases { |
| t.Run(tc.in, func(t *testing.T) { |
| if got := trimZeros(tc.in); got != tc.out { |
| t.Errorf("got %q; want %q", got, tc.out) |
| } |
| }) |
| } |
| } |
| |
| func TestReferencePath(t *testing.T) { |
| testCases := []struct { |
| input string |
| want string |
| alt string |
| }{{ |
| input: "v: w: x: _|_", |
| want: "", |
| }, { |
| input: "v: w: x: 2", |
| want: "", |
| }, { |
| input: "v: w: x: a, a: 1", |
| want: "a", |
| }, { |
| input: "v: w: x: a.b.c, a: b: c: 1", |
| want: "a.b.c", |
| }, { |
| input: "v: w: x: w.a.b.c, v: w: a: b: c: 1", |
| want: "v.w.a.b.c", |
| }, { |
| input: `v: w: x: w.a.b.c, v: w: a: b: c: 1, #D: 3, opt?: 3, "v\(#D)": 3, X: {a: 3}, X`, |
| want: "v.w.a.b.c", |
| }, { |
| input: ` |
| v: w: x: w.a[bb]["c"] |
| v: w: a: b: c: 1 |
| bb: "b"`, |
| want: "v.w.a.b.c", |
| }, { |
| input: ` |
| X="\(y)": 1 |
| v: w: x: X // TODO: Move up for crash |
| y: "foo"`, |
| want: "foo", |
| }, { |
| input: ` |
| v: w: _ |
| v: [X=string]: x: a[X] |
| a: w: 1`, |
| want: "a.w", |
| }, { |
| input: `v: { |
| for t in src { |
| w: "t\(t)": 1 |
| w: "\(t)": w["t\(t)"] |
| } |
| }, |
| src: ["x", "y"]`, |
| want: "v.w.tx", |
| }, { |
| input: ` |
| v: w: x: a |
| a: 1 |
| for i in [] { |
| } |
| `, |
| want: "a", |
| }, { |
| input: ` |
| v: w: close({x: a}) |
| a: 1 |
| `, |
| want: "a", |
| }, { |
| input: ` |
| import "math" |
| |
| v: w: x: math.Pi |
| `, |
| want: "Pi", |
| alt: "3.14159265358979323846264338327950288419716939937510582097494459", |
| }} |
| for _, tc := range testCases { |
| t.Run("", func(t *testing.T) { |
| var r Runtime |
| inst, _ := r.Compile("in", tc.input) // getInstance(t, tc.input) |
| v := inst.Lookup("v", "w", "x") |
| |
| root, path := v.ReferencePath() |
| if got := path.String(); got != tc.want { |
| t.Errorf("\n got %v;\nwant %v", got, tc.want) |
| } |
| |
| if tc.want != "" { |
| want := "1" |
| if tc.alt != "" { |
| want = tc.alt |
| } |
| v := fmt.Sprint(root.LookupPath(path)) |
| if v != want { |
| t.Errorf("path resolved to %s; want %s", v, want) |
| } |
| } |
| |
| inst, a := v.Reference() |
| if got := strings.Join(a, "."); got != tc.want { |
| t.Errorf("\n got %v;\nwant %v", got, tc.want) |
| } |
| |
| if tc.want != "" { |
| want := "1" |
| if tc.alt != "" { |
| want = tc.alt |
| } |
| v := fmt.Sprint(inst.Lookup(a...)) |
| if v != want { |
| t.Errorf("path resolved to %s; want %s", v, want) |
| } |
| } |
| }) |
| } |
| } |
| |
| func TestPathCorrection(t *testing.T) { |
| testCases := []struct { |
| input string |
| lookup func(i *Instance) Value |
| want string |
| skip bool |
| }{{ |
| input: ` |
| a: b: { |
| c: d: b |
| } |
| `, |
| lookup: func(i *Instance) Value { |
| op, a := i.Lookup("a", "b", "c", "d").Expr() |
| _ = op |
| return a[0] // structural cycle errors. |
| }, |
| want: "a", |
| }, { |
| |
| // TODO: embedding: have field operators. |
| input: ` |
| a: { |
| {x: c} |
| c: 3 |
| } |
| `, |
| lookup: func(i *Instance) Value { |
| op, a := i.Lookup("a").Expr() |
| _ = op |
| return a[0].Lookup("x") |
| }, |
| want: "a.c", |
| }, { |
| |
| // TODO: implement proper Elem() |
| input: ` |
| a: b: [...T] |
| a: b: [...T] |
| T: int |
| `, |
| lookup: func(i *Instance) Value { |
| v, _ := i.Lookup("a", "b").Elem() |
| _, a := v.Expr() |
| return a[0] |
| }, |
| want: "T", |
| }, { |
| input: ` |
| #S: { |
| b?: [...#T] |
| b?: [...#T] |
| } |
| #T: int |
| `, |
| lookup: func(i *Instance) Value { |
| v := i.LookupDef("#S") |
| f, _ := v.LookupField("b") |
| v, _ = f.Value.Elem() |
| _, a := v.Expr() |
| return a[0] |
| }, |
| want: "#T", |
| }, { |
| input: ` |
| #S: { |
| a?: [...#T] |
| b?: [...#T] |
| } |
| #T: int |
| `, |
| lookup: func(i *Instance) Value { |
| v := i.LookupDef("#S") |
| f, _ := v.LookupField("a") |
| x := f.Value |
| f, _ = v.LookupField("b") |
| y := f.Value |
| u := x.Unify(y) |
| v, _ = u.Elem() |
| _, a := v.Expr() |
| return a[0] |
| }, |
| want: "#T", |
| }, { |
| input: ` |
| #a: { |
| close({}) | close({c: #T}) | close({d: string}) |
| #T: {b: 3} |
| } |
| `, |
| lookup: func(i *Instance) Value { |
| f, _ := i.LookupField("#a") |
| _, a := f.Value.Expr() // & |
| _, a = a[0].Expr() // | |
| return a[1].Lookup("c") |
| }, |
| want: "#a.#T", |
| }, { |
| input: ` |
| package foo |
| |
| #Struct: { |
| #T: int |
| |
| {b?: #T} |
| }`, |
| want: "#Struct.#T", |
| lookup: func(inst *Instance) Value { |
| // Locate Struct |
| i, _ := inst.Value().Fields(Definitions(true)) |
| if !i.Next() { |
| t.Fatal("no fields") |
| } |
| // Locate b |
| i, _ = i.Value().Fields(Definitions(true), Optional(true)) |
| if !(i.Next() && i.Next()) { |
| t.Fatal("no fields") |
| } |
| v := i.Value() |
| return v |
| }, |
| }, { |
| |
| input: ` |
| package foo |
| |
| #A: #B: #T |
| |
| #T: { |
| a: #S.#U |
| #S: #U: {} |
| } |
| `, |
| want: "#T.#S.#U", |
| lookup: func(inst *Instance) Value { |
| f, _ := inst.Value().LookupField("#A") |
| f, _ = f.Value.LookupField("#B") |
| v := f.Value |
| v = Dereference(v) |
| v = v.Lookup("a") |
| return v |
| }, |
| }, { |
| |
| // TODO: record additionalItems in list |
| input: ` |
| package foo |
| |
| #A: #B: #T |
| |
| #T: { |
| a: [...#S] |
| #S: {} |
| } |
| `, |
| want: "#T.#S", |
| lookup: func(inst *Instance) Value { |
| f, _ := inst.Value().LookupField("#A") |
| f, _ = f.Value.LookupField("#B") |
| v := f.Value |
| v = Dereference(v) |
| v, _ = v.Lookup("a").Elem() |
| return v |
| }, |
| }, { |
| input: ` |
| #A: { |
| b: #T |
| } |
| |
| #T: { |
| a: #S |
| #S: {} |
| } |
| `, |
| want: "#T.#S", |
| lookup: func(inst *Instance) Value { |
| f, _ := inst.Value().LookupField("#A") |
| v := f.Value.Lookup("b") |
| v = Dereference(v) |
| v = v.Lookup("a") |
| return v |
| }, |
| }, { |
| input: ` |
| #Tracing: { |
| #T: { address?: string } |
| #S: { ip?: string } |
| |
| close({}) | close({ |
| t: #T |
| }) | close({ |
| s: #S |
| }) |
| } |
| #X: {} |
| #X // Disconnect top-level struct from the one visible by close. |
| `, |
| want: "#Tracing.#T", |
| lookup: func(inst *Instance) Value { |
| f, _ := inst.Value().LookupField("#Tracing") |
| v := f.Value.Eval() |
| _, args := v.Expr() |
| v = args[1] |
| v = v.Lookup("t") |
| return v |
| }, |
| }} |
| for _, tc := range testCases { |
| if tc.skip { |
| continue |
| } |
| t.Run("", func(t *testing.T) { |
| var r Runtime |
| inst, err := r.Compile("in", tc.input) |
| if err != nil { |
| t.Fatal(err) |
| } |
| v := tc.lookup(inst) |
| gotInst, ref := v.Reference() |
| if gotInst != inst { |
| t.Error("reference not in original instance") |
| } |
| gotPath := strings.Join(ref, ".") |
| if gotPath != tc.want { |
| t.Errorf("got path %s; want %s", gotPath, tc.want) |
| } |
| |
| x, p := v.ReferencePath() |
| if x.Value() != inst.Value() { |
| t.Error("reference not in original instance") |
| } |
| gotPath = p.String() |
| if gotPath != tc.want { |
| t.Errorf("got path %s; want %s", gotPath, tc.want) |
| } |
| |
| }) |
| } |
| } |
| |
| // func TestReferences(t *testing.T) { |
| // config1 := ` |
| // a: { |
| // b: 3 |
| // } |
| // c: { |
| // d: a.b |
| // e: c.d |
| // f: a |
| // } |
| // ` |
| // config2 := ` |
| // a: { c: 3 } |
| // b: { c: int, d: 4 } |
| // r: (a & b).c |
| // c: {args: s1 + s2}.args |
| // s1: string |
| // s2: string |
| // d: ({arg: b}).arg.c |
| // e: f.arg.c |
| // f: {arg: b} |
| // ` |
| // testCases := []struct { |
| // config string |
| // in string |
| // out string |
| // }{ |
| // {config1, "c.d", "a.b"}, |
| // {config1, "c.e", "c.d"}, |
| // {config1, "c.f", "a"}, |
| |
| // {config2, "r", "a.c b.c"}, |
| // {config2, "c", "s1 s2"}, |
| // // {config2, "d", "b.c"}, // TODO: make this work as well. |
| // {config2, "e", "f.arg.c"}, // TODO: should also report b.c. |
| // } |
| // for _, tc := range testCases { |
| // t.Run(tc.in, func(t *testing.T) { |
| // ctx, st := compileFile(t, tc.config) |
| // v := newValueRoot(ctx, st) |
| // for _, k := range strings.Split(tc.in, ".") { |
| // obj, err := v.structValFull(ctx) |
| // if err != nil { |
| // t.Fatal(err) |
| // } |
| // v = obj.Lookup(k) |
| // } |
| // got := []string{} |
| // for _, r := range v.References() { |
| // got = append(got, strings.Join(r, ".")) |
| // } |
| // want := strings.Split(tc.out, " ") |
| // if !reflect.DeepEqual(got, want) { |
| // t.Errorf("got %v; want %v", got, want) |
| // } |
| // }) |
| // } |
| // } |
| |
| func checkErr(t *testing.T, err error, str, name string) bool { |
| t.Helper() |
| if err == nil { |
| if str != "" { |
| t.Errorf(`err:%s: got ""; want %q`, name, str) |
| } |
| return true |
| } |
| return checkFailed(t, err, str, name) |
| } |
| |
| func checkFatal(t *testing.T, err error, str, name string) { |
| t.Helper() |
| if !checkFailed(t, err, str, name) { |
| t.SkipNow() |
| } |
| } |
| |
| func checkFailed(t *testing.T, err error, str, name string) bool { |
| t.Helper() |
| if err != nil { |
| got := err.Error() |
| if str == "" { |
| t.Fatalf(`err:%s: got %q; want ""`, name, got) |
| } |
| if !strings.Contains(got, str) { |
| t.Errorf(`err:%s: got %q; want %q`, name, got, str) |
| } |
| return false |
| } |
| return true |
| } |
| |
| func TestExpr(t *testing.T) { |
| testCases := []struct { |
| input string |
| want string |
| }{{ |
| input: "v: 3", |
| want: "3", |
| }, { |
| input: "v: 3 + 4", |
| want: "+(3 4)", |
| }, { |
| input: "v: !a, a: bool", |
| want: `!(.(〈〉 "a"))`, |
| }, { |
| input: "v: !a, a: 3", // TODO: Should still look up. |
| want: `!(.(〈〉 "a"))`, |
| }, { |
| input: "v: 1 | 2 | 3 | *4", |
| want: "|(1 2 3 4)", |
| }, { |
| input: "v: 2 & 5", // Allow even with error. |
| want: "&(2 5)", |
| }, { |
| input: "v: 2 | 5", |
| want: "|(2 5)", |
| }, { |
| input: "v: 2 && 5", |
| want: "&&(2 5)", |
| }, { |
| input: "v: 2 || 5", |
| want: "||(2 5)", |
| }, { |
| input: "v: 2 == 5", |
| want: "==(2 5)", |
| }, { |
| input: "v: !b, b: true", |
| want: `!(.(〈〉 "b"))`, |
| }, { |
| input: "v: 2 != 5", |
| want: "!=(2 5)", |
| }, { |
| input: "v: <5", |
| want: "<(5)", |
| }, { |
| input: "v: 2 <= 5", |
| want: "<=(2 5)", |
| }, { |
| input: "v: 2 > 5", |
| want: ">(2 5)", |
| }, { |
| input: "v: 2 >= 5", |
| want: ">=(2 5)", |
| }, { |
| input: "v: 2 =~ 5", |
| want: "=~(2 5)", |
| }, { |
| input: "v: 2 !~ 5", |
| want: "!~(2 5)", |
| }, { |
| input: "v: 2 + 5", |
| want: "+(2 5)", |
| }, { |
| input: "v: 2 - 5", |
| want: "-(2 5)", |
| }, { |
| input: "v: 2 * 5", |
| want: "*(2 5)", |
| }, { |
| input: "v: 2 / 5", |
| want: "/(2 5)", |
| }, { |
| input: "v: 2 quo 5", |
| want: "quo(2 5)", |
| }, { |
| input: "v: 2 rem 5", |
| want: "rem(2 5)", |
| }, { |
| input: "v: 2 div 5", |
| want: "div(2 5)", |
| }, { |
| input: "v: 2 mod 5", |
| want: "mod(2 5)", |
| }, { |
| input: "v: a.b, a: b: 4", |
| want: `.(.(〈〉 "a") "b")`, |
| }, { |
| input: `v: a["b"], a: b: 3 `, |
| want: `[](.(〈〉 "a") "b")`, |
| }, { |
| input: "v: a[2:5], a: [1, 2, 3, 4, 5]", |
| want: `[:](.(〈〉 "a") 2 5)`, |
| }, { |
| input: "v: len([])", |
| want: "()(len [])", |
| }, { |
| input: "v: a.b, a: { b: string }", |
| want: `.(.(〈〉 "a") "b")`, |
| }, { |
| input: `v: "Hello, \(x)! Welcome to \(place)", place: string, x: string`, |
| want: `\()("Hello, " .(〈〉 "x") "! Welcome to " .(〈〉 "place") "")`, |
| }, { |
| input: `v: { a, b: 1 }, a: 2`, |
| want: `&(.(〈〉 "a") {b:1})`, |
| }, { |
| input: `v: { {c: a}, b: a }, a: int`, |
| want: `&({c:a} {b:a})`, |
| }, { |
| input: `v: [...number] | *[1, 2, 3]`, |
| want: `([...number]|*[1,2,3])`, |
| }, { |
| input: `v: or([1, 2, 3])`, |
| want: `|(1 2 3)`, |
| }, { |
| input: `v: and([1, 2, 3])`, |
| want: `&(1 2 3)`, |
| }} |
| for _, tc := range testCases { |
| t.Run(tc.input, func(t *testing.T) { |
| v := getInstance(t, tc.input).Lookup("v") |
| got := exprStr(v) |
| if got != tc.want { |
| t.Errorf("\n got %v;\nwant %v", got, tc.want) |
| } |
| }) |
| } |
| } |
| |
| func exprStr(v Value) string { |
| op, operands := v.Expr() |
| if op == NoOp { |
| return compactRawStr(v) |
| } |
| s := op.String() |
| s += "(" |
| for i, v := range operands { |
| if i > 0 { |
| s += " " |
| } |
| s += exprStr(v) |
| } |
| s += ")" |
| return s |
| } |
| |
| func compactRawStr(v Value) string { |
| ctx := v.ctx() |
| cfg := &debug.Config{Compact: true, Raw: true} |
| return debug.NodeString(ctx, v.v, cfg) |
| } |