blob: 1e56edbfb8885f58ecbcae1c542f52a22db24396 [file] [log] [blame]
// Copyright 2020 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 validate
import (
"fmt"
"strings"
"testing"
"cuelang.org/go/cue/errors"
"cuelang.org/go/cue/parser"
"cuelang.org/go/internal/core/adt"
"cuelang.org/go/internal/core/compile"
"cuelang.org/go/internal/core/eval"
"cuelang.org/go/internal/core/runtime"
"github.com/google/go-cmp/cmp"
)
func TestValidate(t *testing.T) {
testCases := []struct {
desc string
in string
out string
lookup string
cfg *Config
}{{
desc: "no error, but not concrete, even with definition label",
cfg: &Config{Concrete: true},
in: `
#foo: { use: string }
`,
lookup: "#foo",
out: "incomplete\n#foo.use: incomplete value string",
}, {
desc: "definitions not considered for completeness",
cfg: &Config{Concrete: true},
in: `
#foo: { use: string }
`,
}, {
desc: "hidden fields not considered for completeness",
cfg: &Config{Concrete: true},
in: `
_foo: { use: string }
`,
}, {
desc: "hidden fields not considered for completeness",
in: `
_foo: { use: string }
`,
}, {
desc: "evaluation error at top",
in: `
1 & 2
`,
out: "eval\nconflicting values 2 and 1:\n test:2:3\n test:2:7",
}, {
desc: "evaluation error in field",
in: `
x: 1 & 2
`,
out: "eval\nx: conflicting values 2 and 1:\n test:2:6\n test:2:10",
}, {
desc: "first error",
in: `
x: 1 & 2
y: 2 & 4
`,
out: "eval\nx: conflicting values 2 and 1:\n test:2:6\n test:2:10",
}, {
desc: "all errors",
cfg: &Config{AllErrors: true},
in: `
x: 1 & 2
y: 2 & 4
`,
out: `eval
x: conflicting values 2 and 1:
test:2:6
test:2:10
y: conflicting values 4 and 2:
test:3:6
test:3:10`,
}, {
desc: "incomplete",
cfg: &Config{Concrete: true},
in: `
y: 2 + x
x: string
`,
out: "incomplete\ny: non-concrete value string in operand to +:\n test:2:6\n test:3:3",
}, {
desc: "allowed incomplete cycle",
in: `
y: x
x: y
`,
}, {
desc: "allowed incomplete when disallowing cycles",
cfg: &Config{DisallowCycles: true},
in: `
y: string
x: y
`,
}, {
desc: "disallow cycle",
cfg: &Config{DisallowCycles: true},
in: `
y: x + 1
x: y - 1
`,
out: "cycle\ncycle error:\n test:2:6",
}, {
desc: "disallow cycle",
cfg: &Config{DisallowCycles: true},
in: `
a: b - 100
b: a + 100
c: [c[1], c[0]] `,
out: "cycle\ncycle error:\n test:2:6",
}, {
desc: "treat cycles as incomplete when not disallowing",
cfg: &Config{},
in: `
y: x + 1
x: y - 1
`,
}, {
// Note: this is already handled by evaluation, as terminal errors
// are percolated up.
desc: "catch most serious error",
cfg: &Config{Concrete: true},
in: `
y: string
x: 1 & 2
`,
out: "eval\nx: conflicting values 2 and 1:\n test:3:6\n test:3:10",
}, {
desc: "consider defaults for concreteness",
cfg: &Config{Concrete: true},
in: `
x: *1 | 2
`,
}, {
desc: "allow non-concrete in definitions in concrete mode",
cfg: &Config{Concrete: true},
in: `
x: 2
#d: {
b: int
c: b + b
}
`,
}, {
desc: "pick up non-concrete value in default",
cfg: &Config{Concrete: true},
in: `
x: null | *{
a: int
}
`,
out: "incomplete\nx.a: incomplete value int",
}, {
desc: "pick up non-concrete value in default",
cfg: &Config{Concrete: true},
in: `
x: null | *{
a: 1 | 2
}
`,
out: "incomplete\nx.a: incomplete value 1 | 2",
}}
r := runtime.New()
ctx := eval.NewContext(r, nil)
for _, tc := range testCases {
t.Run(tc.desc, func(t *testing.T) {
f, err := parser.ParseFile("test", tc.in)
if err != nil {
t.Fatal(err)
}
v, err := compile.Files(nil, r, "", f)
if err != nil {
t.Fatal(err)
}
ctx.Unify(v, adt.Finalized)
if tc.lookup != "" {
v = v.Lookup(adt.MakeIdentLabel(r, tc.lookup, "main"))
}
b := Validate(ctx, v, tc.cfg)
w := &strings.Builder{}
if b != nil {
fmt.Fprintln(w, b.Code)
errors.Print(w, b.Err, nil)
}
got := strings.TrimSpace(w.String())
if tc.out != got {
t.Error(cmp.Diff(tc.out, got))
}
})
}
}