blob: 41c12385b924dfc01355b66b9503007383dc7a23 [file] [log] [blame]
// 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 (
"fmt"
"math/big"
"testing"
"cuelang.org/go/cue/ast"
"github.com/cockroachdb/apd/v2"
"github.com/google/go-cmp/cmp"
"github.com/google/go-cmp/cmp/cmpopts"
)
var defIntBase = newNumBase(&ast.BasicLit{}, newNumInfo(intKind, 0, 10, false))
var defRatBase = newNumBase(&ast.BasicLit{}, newNumInfo(floatKind, 0, 10, false))
func mkInt(a int64) *numLit {
x := &numLit{numBase: defIntBase}
x.v.SetInt64(a)
return x
}
func mkIntString(a string) *numLit {
x := &numLit{numBase: defIntBase}
x.v.SetString(a)
return x
}
func mkFloat(a string) *numLit {
x := &numLit{numBase: defRatBase}
x.v.SetString(a)
return x
}
func mkBigInt(a int64) (v apd.Decimal) { v.SetInt64(a); return }
func mkBigFloat(a string) (v apd.Decimal) { v.SetString(a); return }
var diffOpts = []cmp.Option{
cmp.Comparer(func(x, y big.Rat) bool {
return x.String() == y.String()
}),
cmp.Comparer(func(x, y big.Int) bool {
return x.String() == y.String()
}),
cmp.AllowUnexported(
nullLit{},
boolLit{},
stringLit{},
bytesLit{},
numLit{},
numBase{},
numInfo{},
),
cmpopts.IgnoreUnexported(
bottom{},
baseValue{},
baseValue{},
),
}
var (
nullSentinel = &nullLit{}
trueSentinel = &boolLit{b: true}
falseSentinel = &boolLit{b: false}
)
func TestLiterals(t *testing.T) {
mkMul := func(x int64, m multiplier, base int) *numLit {
return &numLit{
newNumBase(&ast.BasicLit{}, newNumInfo(intKind, m, base, false)),
mkBigInt(x),
}
}
hk := &numLit{
newNumBase(&ast.BasicLit{}, newNumInfo(intKind, 0, 10, true)),
mkBigInt(100000),
}
testCases := []struct {
lit string
node value
}{
{"0", mkInt(0)},
{"null", nullSentinel},
{"true", trueSentinel},
{"false", falseSentinel},
{"fls", &bottom{}},
{`"foo"`, &stringLit{str: "foo"}},
{`#"foo"#`, &stringLit{str: "foo"}},
{`#""foo"#`, &stringLit{str: `"foo`}},
{`#" ""#`, &stringLit{str: ` "`}},
{`"\"foo\""`, &stringLit{str: `"foo"`}},
{`"foo\u0032"`, &stringLit{str: `foo2`}},
{`"foo\U00000033"`, &stringLit{str: `foo3`}},
{`"foo\U0001f499"`, &stringLit{str: `foo💙`}},
{`"\a\b\f\n\r\t\v"`, &stringLit{str: "\a\b\f\n\r\t\v"}},
{`"""
"""`, &stringLit{str: ""}},
{`"""
abc
"""`, &stringLit{str: "abc"}},
{`"""
abc
def
"""`, &stringLit{str: "abc\ndef"}},
{`"""
abc
def
"""`, &stringLit{str: "abc\n\tdef"}},
{`'\xff'`, &bytesLit{b: []byte("\xff")}},
{"1", mkInt(1)},
{"100_000", hk},
{"1.", mkFloat("1")},
{"0.0", mkFloat("0.0")},
{".0", mkFloat(".0")},
{"012.34", mkFloat("012.34")},
{".01", mkFloat(".01")},
{".01e2", mkFloat("1")},
{"0.", mkFloat("0.")},
{"1K", mkMul(1000, mulK, 10)},
{".5K", mkMul(500, mulK, 10)},
{"1Mi", mkMul(1024*1024, mulMi, 10)},
{"1.5Mi", mkMul((1024+512)*1024, mulMi, 10)},
{"1.3Mi", &bottom{}}, // Cannot be accurately represented.
{"1.3G", mkMul(1300000000, mulG, 10)},
{"1.3e+20", mkFloat("1.3e+20")},
{"1.3e20", mkFloat("1.3e+20")},
{"1.3e-5", mkFloat("1.3e-5")},
{".3e-1", mkFloat("0.3e-1")},
{"0e-5", mkFloat("0e-5")},
{"0E-5", mkFloat("0e-5")},
{"5e-5", mkFloat("5e-5")},
{"5E-5", mkFloat("5e-5")},
{"0x1234", mkMul(0x1234, 0, 16)},
{"0xABCD", mkMul(0xABCD, 0, 16)},
{"0b11001000", mkMul(0xc8, 0, 2)},
{"0b1", mkMul(1, 0, 2)},
{"0o755", mkMul(0755, 0, 8)},
{"0755", mkMul(0755, 0, 8)},
}
p := litParser{
ctx: &context{Context: &apd.BaseContext},
}
for i, tc := range testCases {
t.Run(fmt.Sprintf("%d/%+q", i, tc.lit), func(t *testing.T) {
got := p.parse(&ast.BasicLit{Value: tc.lit})
if !cmp.Equal(got, tc.node, diffOpts...) {
t.Error(cmp.Diff(got, tc.node, diffOpts...))
t.Errorf("%#v, %#v\n", got, tc.node)
}
})
}
}
func TestLiteralErrors(t *testing.T) {
testCases := []struct {
lit string
}{
{`"foo\u"`},
{`"foo\u003"`},
{`"foo\U1234567"`},
{`"foo\U12345678"`},
{`"foo\Ug"`},
{`"\xff"`},
// not allowed in string literal, only binary
{`"foo\x00"`},
{`0x`},
{`0o`},
{`0b`},
{`0_`},
{"0128"},
{``},
{`"`},
{`"a`},
// wrong indentation
{`"""
abc
def
"""`},
// non-matching quotes
{`"""
abc
'''`},
{`"""
abc
"`},
{`"abc \( foo "`},
}
p := litParser{
ctx: &context{Context: &apd.BaseContext},
}
for _, tc := range testCases {
t.Run(fmt.Sprintf("%+q", tc.lit), func(t *testing.T) {
got := p.parse(&ast.BasicLit{Value: tc.lit})
if _, ok := got.(*bottom); !ok {
t.Fatalf("expected error but found none")
}
})
}
}