3 Star 1 Fork 0

Gitee 极速下载/otto

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
此仓库是为了提升国内下载速度的镜像仓库,每日同步一次。 原始仓库: https://github.com/robertkrimen/otto
克隆/下载
reflect_test.go 17.71 KB
一键复制 编辑 原始数据 按行查看 历史
Steven Hartland 提交于 2024-04-13 17:05 . chore: update ci versions (#519)
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940
package otto
import (
"bytes"
"encoding/json"
"fmt"
"math"
"reflect"
"strings"
"testing"
"github.com/stretchr/testify/require"
)
type abcStruct struct {
Jkl interface{}
Pqr map[string]int8
Ghi string
Mno _mnoStruct
Def int
Abc bool
}
func (abc abcStruct) String() string {
return abc.Ghi
}
func (abc *abcStruct) FuncPointer() string {
return "abc"
}
func (abc abcStruct) Func() {
}
func (abc abcStruct) FuncReturn1() string {
return "abc"
}
func (abc abcStruct) FuncReturn2() (string, error) {
return "def", nil
}
func (abc abcStruct) Func1Return1(a string) string {
return a
}
func (abc abcStruct) Func2Return1(x, y string) string {
return x + y
}
func (abc abcStruct) FuncEllipsis(xyz ...string) int {
return len(xyz)
}
func (abc abcStruct) FuncReturnStruct() _mnoStruct {
return _mnoStruct{}
}
func (abc abcStruct) Func1Int(i int) int {
return i + 1
}
func (abc abcStruct) Func1Int8(i int8) int8 {
return i + 1
}
func (abc abcStruct) Func1Int16(i int16) int16 {
return i + 1
}
func (abc abcStruct) Func1Int32(i int32) int32 {
return i + 1
}
func (abc abcStruct) Func1Int64(i int64) int64 {
return i + 1
}
func (abc abcStruct) Func1Uint(i uint) uint {
return i + 1
}
func (abc abcStruct) Func1Uint8(i uint8) uint8 {
return i + 1
}
func (abc abcStruct) Func1Uint16(i uint16) uint16 {
return i + 1
}
func (abc abcStruct) Func1Uint32(i uint32) uint32 {
return i + 1
}
func (abc abcStruct) Func1Uint64(i uint64) uint64 {
return i + 1
}
func (abc abcStruct) Func2Int(i, j int) int {
return i + j
}
func (abc abcStruct) Func2StringInt(s string, i int) string {
return fmt.Sprintf("%v:%v", s, i)
}
func (abc abcStruct) Func1IntVariadic(a ...int) int {
t := 0
for _, i := range a {
t += i
}
return t
}
func (abc abcStruct) Func2IntVariadic(s string, a ...int) string {
t := 0
for _, i := range a {
t += i
}
return fmt.Sprintf("%v:%v", s, t)
}
func (abc abcStruct) Func2IntArrayVariadic(s string, a ...[]int) string {
t := 0
for _, i := range a {
for _, j := range i {
t += j
}
}
return fmt.Sprintf("%v:%v", s, t)
}
type _mnoStruct struct {
Ghi string
}
func (mno _mnoStruct) Func() string {
return "mno"
}
func TestReflect(t *testing.T) {
tt(t, func() {
// Testing dbgf
// These should panic
str := "test"
require.Panics(t, func() {
_, err := toValue("Xyzzy").toReflectValue(reflect.ValueOf(&str).Type())
require.NoError(t, err)
})
require.Panics(t, func() {
_, err := stringToReflectValue("Xyzzy", reflect.Ptr)
require.NoError(t, err)
})
})
}
func Test_reflectStruct(t *testing.T) {
tt(t, func() {
test, vm := test()
// _abcStruct
{
abc := &abcStruct{}
vm.Set("abc", abc)
test(`
[ abc.Abc, abc.Ghi ];
`, "false,")
abc.Abc = true
abc.Ghi = "Nothing happens."
test(`
[ abc.Abc, abc.Ghi ];
`, "true,Nothing happens.")
*abc = abcStruct{}
test(`
[ abc.Abc, abc.Ghi ];
`, "false,")
abc.Abc = true
abc.Ghi = "Xyzzy"
vm.Set("abc", abc)
test(`
[ abc.Abc, abc.Ghi ];
`, "true,Xyzzy")
is(abc.Abc, true)
test(`
abc.Abc = false;
abc.Def = 451;
abc.Ghi = "Nothing happens.";
abc.abc = "Something happens.";
[ abc.Def, abc.abc ];
`, "451,Something happens.")
is(abc.Abc, false)
is(abc.Def, 451)
is(abc.Ghi, "Nothing happens.")
test(`
delete abc.Def;
delete abc.abc;
[ abc.Def, abc.abc ];
`, "451,")
is(abc.Def, 451)
test(`
abc.FuncPointer();
`, "abc")
test(`
abc.Func();
`, "undefined")
test(`
abc.FuncReturn1();
`, "abc")
test(`
abc.Func1Return1("abc");
`, "abc")
test(`
abc.Func2Return1("abc", "def");
`, "abcdef")
test(`
abc.FuncEllipsis("abc", "def", "ghi");
`, 3)
test(`
ret = abc.FuncReturn2();
if (ret && ret.length && ret.length == 2 && ret[0] == "def" && ret[1] === undefined) {
true;
} else {
false;
}
`, true)
test(`
abc.FuncReturnStruct();
`, "[object Object]")
test(`
abc.FuncReturnStruct().Func();
`, "mno")
test(`
abc.Func1Int(1);
`, 2)
test(`
abc.Func1Int(0x01 & 0x01);
`, 2)
test(`raise:
abc.Func1Int(1.1);
`, "RangeError: converting float64 to int would cause loss of precision")
test(`
var v = 1;
abc.Func1Int(v + 1);
`, 3)
test(`
abc.Func2Int(1, 2);
`, 3)
test(`
abc.Func1Int8(1);
`, 2)
test(`
abc.Func1Int16(1);
`, 2)
test(`
abc.Func1Int32(1);
`, 2)
test(`
abc.Func1Int64(1);
`, 2)
test(`
abc.Func1Uint(1);
`, 2)
test(`
abc.Func1Uint8(1);
`, 2)
test(`
abc.Func1Uint16(1);
`, 2)
test(`
abc.Func1Uint32(1);
`, 2)
test(`
abc.Func1Uint64(1);
`, 2)
test(`
abc.Func2StringInt("test", 1);
`, "test:1")
test(`
abc.Func1IntVariadic(1, 2);
`, 3)
test(`
abc.Func2IntVariadic("test", 1, 2);
`, "test:3")
test(`
abc.Func2IntVariadic("test", [1, 2]);
`, "test:3")
test(`
abc.Func2IntArrayVariadic("test", [1, 2]);
`, "test:3")
test(`
abc.Func2IntArrayVariadic("test", [1, 2], [3, 4]);
`, "test:10")
test(`
abc.Func2IntArrayVariadic("test", [[1, 2], [3, 4]]);
`, "test:10")
}
})
}
func Test_reflectMap(t *testing.T) {
tt(t, func() {
test, vm := test()
// map[string]string
{
abc := map[string]string{
"Xyzzy": "Nothing happens.",
"def": "1",
}
vm.Set("abc", abc)
test(`
abc.xyz = "pqr";
[ abc.Xyzzy, abc.def, abc.ghi ];
`, "Nothing happens.,1,")
is(abc["xyz"], "pqr")
}
// map[string]float64
{
abc := map[string]float64{
"Xyzzy": math.Pi,
"def": 1,
}
vm.Set("abc", abc)
test(`
abc.xyz = "pqr";
abc.jkl = 10;
[ abc.Xyzzy, abc.def, abc.ghi ];
`, "3.141592653589793,1,")
is(abc["xyz"], math.NaN())
is(abc["jkl"], float64(10))
}
// map[string]int32
{
abc := map[string]int32{
"Xyzzy": 3,
"def": 1,
}
vm.Set("abc", abc)
test(`
abc.xyz = "pqr";
abc.jkl = 10;
[ abc.Xyzzy, abc.def, abc.ghi ];
`, "3,1,")
is(abc["xyz"], 0)
is(abc["jkl"], int32(10))
test(`
delete abc["Xyzzy"];
`)
_, exists := abc["Xyzzy"]
is(exists, false)
is(abc["Xyzzy"], 0)
}
// map[int32]string
{
abc := map[int32]string{
0: "abc",
1: "def",
}
vm.Set("abc", abc)
test(`
abc[2] = "pqr";
//abc.jkl = 10;
abc[3] = 10;
[ abc[0], abc[1], abc[2], abc[3] ]
`, "abc,def,pqr,10")
is(abc[2], "pqr")
is(abc[3], "10")
test(`
delete abc[2];
`)
_, exists := abc[2]
is(exists, false)
}
})
}
func Test_reflectMapIterateKeys(t *testing.T) {
tt(t, func() {
test, vm := test()
// map[string]interface{}
{
abc := map[string]interface{}{
"Xyzzy": "Nothing happens.",
"def": 1,
}
vm.Set("abc", abc)
test(`
var keys = [];
for (var key in abc) {
keys.push(key);
}
keys.sort();
keys;
`, "Xyzzy,def")
}
// map[uint]interface{}
{
abc := map[uint]interface{}{
456: "Nothing happens.",
123: 1,
}
vm.Set("abc", abc)
test(`
var keys = [];
for (var key in abc) {
keys.push(key);
}
keys.sort();
keys;
`, "123,456")
}
// map[byte]interface{}
{
abc := map[byte]interface{}{
10: "Nothing happens.",
20: 1,
}
vm.Set("abc", abc)
test(`
for (var key in abc) {
abc[key] = "123";
}
`)
is(abc[10], "123")
is(abc[20], "123")
}
})
}
func Test_reflectSlice(t *testing.T) {
tt(t, func() {
test, vm := test()
// []bool
{
abc := []bool{
false,
true,
true,
false,
}
vm.Set("abc", abc)
test(`
abc;
`, "false,true,true,false")
test(`
abc[0] = true;
abc[abc.length-1] = true;
delete abc[2];
abc;
`, "true,true,false,true")
is(abc, []bool{true, true, false, true})
is(abc[len(abc)-1], true)
}
// []int32
{
abc := make([]int32, 4)
vm.Set("abc", abc)
test(`
abc;
`, "0,0,0,0")
test(`raise:
abc[0] = "42";
abc[1] = 4.2;
abc[2] = 3.14;
abc;
`, "RangeError: 4.2 to reflect.Kind: int32")
is(abc, []int32{42, 0, 0, 0})
test(`
delete abc[1];
delete abc[2];
`)
is(abc[1], 0)
is(abc[2], 0)
}
})
}
func Test_reflectArray(t *testing.T) {
tt(t, func() {
test, vm := test()
// []bool
{
abc := [4]bool{
false,
true,
true,
false,
}
vm.Set("abc", abc)
test(`
abc;
`, "false,true,true,false")
// Unaddressable array
test(`
abc[0] = true;
abc[abc.length-1] = true;
abc;
`, "false,true,true,false")
// Again, unaddressable array
is(abc, [4]bool{false, true, true, false})
is(abc[len(abc)-1], false)
// ...
}
// []int32
{
abc := make([]int32, 4)
vm.Set("abc", abc)
test(`
abc;
`, "0,0,0,0")
test(`raise:
abc[0] = "42";
abc[1] = 4.2;
abc[2] = 3.14;
abc;
`, "RangeError: 4.2 to reflect.Kind: int32")
is(abc, []int32{42, 0, 0, 0})
}
// []bool
{
abc := [4]bool{
false,
true,
true,
false,
}
vm.Set("abc", &abc)
test(`
abc;
`, "false,true,true,false")
test(`
abc[0] = true;
abc[abc.length-1] = true;
delete abc[2];
abc;
`, "true,true,false,true")
is(abc, [4]bool{true, true, false, true})
is(abc[len(abc)-1], true)
}
// no common type
{
test(`
abc = [1, 2.2, "str"];
abc;
`, "1,2.2,str")
val, err := vm.Get("abc")
is(err, nil)
abc, err := val.Export()
is(err, nil)
is(abc, []interface{}{int64(1), 2.2, "str"})
}
// common type int
{
test(`
abc = [1, 2, 3];
abc;
`, "1,2,3")
val, err := vm.Get("abc")
is(err, nil)
abc, err := val.Export()
is(err, nil)
is(abc, []int64{1, 2, 3})
}
// common type string
{
test(`
abc = ["str1", "str2", "str3"];
abc;
`, "str1,str2,str3")
val, err := vm.Get("abc")
is(err, nil)
abc, err := val.Export()
is(err, nil)
is(abc, []string{"str1", "str2", "str3"})
}
// issue #269
{
called := false
vm.Set("blah", func(c FunctionCall) Value {
v, err := c.Argument(0).Export()
is(err, nil)
is(v, []int64{3})
called = true
return UndefinedValue()
})
is(called, false)
test(`var x = 3; blah([x])`)
is(called, true)
}
})
}
func Test_reflectArray_concat(t *testing.T) {
tt(t, func() {
test, vm := test()
vm.Set("ghi", []string{"jkl", "mno"})
vm.Set("pqr", []interface{}{"jkl", 42, 3.14159, true})
test(`
var def = {
"abc": ["abc"],
"xyz": ["xyz"]
};
xyz = pqr.concat(ghi, def.abc, def, def.xyz);
[ xyz, xyz.length ];
`, "jkl,42,3.14159,true,jkl,mno,abc,[object Object],xyz,9")
})
}
func Test_reflectMapInterface(t *testing.T) {
tt(t, func() {
test, vm := test()
{
abc := map[string]interface{}{
"Xyzzy": "Nothing happens.",
"def": "1",
"jkl": "jkl",
}
vm.Set("abc", abc)
vm.Set("mno", &abcStruct{})
test(`
abc.xyz = "pqr";
abc.ghi = {};
abc.jkl = 3.14159;
abc.mno = mno;
mno.Abc = true;
mno.Ghi = "Something happens.";
[ abc.Xyzzy, abc.def, abc.ghi, abc.mno ];
`, "Nothing happens.,1,[object Object],[object Object]")
is(abc["xyz"], "pqr")
is(abc["ghi"], map[string]interface{}{})
is(abc["jkl"], float64(3.14159))
mno, valid := abc["mno"].(*abcStruct)
is(valid, true)
is(mno.Abc, true)
is(mno.Ghi, "Something happens.")
}
})
}
func TestPassthrough(t *testing.T) {
tt(t, func() {
test, vm := test()
{
abc := &abcStruct{
Mno: _mnoStruct{
Ghi: "<Mno.Ghi>",
},
}
vm.Set("abc", abc)
test(`
abc.Mno.Ghi;
`, "<Mno.Ghi>")
vm.Set("pqr", map[string]int8{
"xyzzy": 0,
"Nothing happens.": 1,
})
test(`
abc.Ghi = "abc";
abc.Pqr = pqr;
abc.Pqr["Nothing happens."];
`, 1)
mno := _mnoStruct{
Ghi: "<mno.Ghi>",
}
vm.Set("mno", mno)
test(`
abc.Mno = mno;
abc.Mno.Ghi;
`, "<mno.Ghi>")
}
})
}
type TestDynamicFunctionReturningInterfaceMyStruct1 struct{} //nolint:errname
func (m *TestDynamicFunctionReturningInterfaceMyStruct1) Error() string { return "MyStruct1" }
type TestDynamicFunctionReturningInterfaceMyStruct2 struct{} //nolint:errname
func (m *TestDynamicFunctionReturningInterfaceMyStruct2) Error() string { return "MyStruct2" }
func TestDynamicFunctionReturningInterface(t *testing.T) {
tt(t, func() {
test, vm := test()
var l []func() error
vm.Set("r", func(cb func() error) { l = append(l, cb) })
vm.Set("e1", func() error { return &TestDynamicFunctionReturningInterfaceMyStruct1{} })
vm.Set("e2", func() error { return &TestDynamicFunctionReturningInterfaceMyStruct2{} })
vm.Set("e3", func() error { return nil })
test("r(function() { return e1(); })", UndefinedValue())
test("r(function() { return e2(); })", UndefinedValue())
test("r(function() { return e3(); })", UndefinedValue())
test("r(function() { return null; })", UndefinedValue())
if l[0]() == nil {
t.Fail()
}
if l[1]() == nil {
t.Fail()
}
if l[2]() != nil {
t.Fail()
}
if l[3]() != nil {
t.Fail()
}
})
}
func TestStructCallParameterConversion(t *testing.T) {
tt(t, func() {
test, vm := test()
type T struct {
StringValue string `json:"s"`
BooleanValue bool `json:"b"`
IntegerValue int `json:"i"`
}
var x T
vm.Set("f", func(t T) bool { return t == x })
// test set properties
x = T{"A", true, 1}
test("f({s: 'A', b: true, i: 1})", true)
// test zero-value properties
x = T{"", false, 0}
test("f({s: '', b: false, i: 0})", true)
// test missing properties
x = T{"", true, 1}
test("f({b: true, i: 1})", true)
x = T{"A", false, 1}
test("f({s: 'A', i: 1})", true)
x = T{"A", true, 0}
test("f({s: 'A', b: true})", true)
x = T{"", false, 0}
test("f({})", true)
// make sure it fails with extra properties
x = T{"", false, 0}
if _, err := vm.Run("f({x: true})"); err == nil {
t.Fail()
}
})
}
type TestTextUnmarshallerCallParameterConversionMyStruct struct{}
func (m *TestTextUnmarshallerCallParameterConversionMyStruct) UnmarshalText(b []byte) error {
if string(b) == "good" {
return nil
}
return fmt.Errorf("NOT_GOOD: %s", string(b))
}
func TestTextUnmarshallerCallParameterConversion(t *testing.T) {
tt(t, func() {
test, vm := test()
vm.Set("f", func(t TestTextUnmarshallerCallParameterConversionMyStruct) bool { return true })
// success
test("f('good')", true)
// explicit failure, should pass error message up
if _, err := vm.Run("f('bad')"); err == nil || !strings.Contains(err.Error(), "NOT_GOOD: bad") {
t.Fail()
}
// wrong input
if _, err := vm.Run("f(null)"); err == nil {
t.Fail()
}
// no input
if _, err := vm.Run("f()"); err == nil {
t.Fail()
}
})
}
func TestJSONRawMessageCallParameterConversion(t *testing.T) {
for _, e := range []struct {
c string
r string
e bool
}{
{"f({a:1})", `{"a":1}`, false},
{"f(null)", `null`, false},
{"f(1)", `1`, false},
{"f('x')", `"x"`, false},
{"f([1,2,3])", `[1,2,3]`, false},
{"f(function(){})", `{}`, false},
{"f()", `[1,2,3]`, true},
} {
t.Run(e.c, func(t *testing.T) {
vm := New()
err := vm.Set("f", func(m json.RawMessage) json.RawMessage { return m })
require.NoError(t, err)
r, err := vm.Run(e.c)
if err != nil {
if !e.e {
t.Error("err should be nil")
t.Fail()
}
return
}
if e.e {
t.Error("err should not be nil")
t.Fail()
return
}
v, err := r.Export()
if err != nil {
t.Error(err)
t.Fail()
}
m, ok := v.(json.RawMessage)
if !ok {
t.Error("result should be json.RawMessage")
t.Fail()
}
if !bytes.Equal(m, json.RawMessage(e.r)) {
t.Errorf("output is wrong\nexpected: %s\nactual: %s\n", e.r, m)
t.Fail()
}
})
}
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/mirrors/otto.git
git@gitee.com:mirrors/otto.git
mirrors
otto
otto
master

搜索帮助

0d507c66 1850385 C8b1a773 1850385