xgo/script/script_test.go
Daniel e93f6f6325
limit max object allocations (#129)
- add object.NumObjects()
- add object allocation limit in VM
- delete objects.Break, objects.Continue, objects.ReturnValue
- add Script.SetMaxAllocs()
- update sandbox documentation
- add some tests
- remove duplicate values in compiled constants (fixes #96)
- option to limit the maximum number of objects in compiled bytecode constants
2019-03-06 17:20:05 -08:00

160 lines
4.4 KiB
Go

package script_test
import (
"errors"
"math"
"testing"
"github.com/d5/tengo/assert"
"github.com/d5/tengo/objects"
"github.com/d5/tengo/script"
)
func TestScript_Add(t *testing.T) {
s := script.New([]byte(`a := b; c := test(b); d := test(5)`))
assert.NoError(t, s.Add("b", 5)) // b = 5
assert.NoError(t, s.Add("b", "foo")) // b = "foo" (re-define before compilation)
assert.NoError(t, s.Add("test", func(args ...objects.Object) (ret objects.Object, err error) {
if len(args) > 0 {
switch arg := args[0].(type) {
case *objects.Int:
return &objects.Int{Value: arg.Value + 1}, nil
}
}
return &objects.Int{Value: 0}, nil
}))
c, err := s.Compile()
assert.NoError(t, err)
assert.NoError(t, c.Run())
assert.Equal(t, "foo", c.Get("a").Value())
assert.Equal(t, "foo", c.Get("b").Value())
assert.Equal(t, int64(0), c.Get("c").Value())
assert.Equal(t, int64(6), c.Get("d").Value())
}
func TestScript_Remove(t *testing.T) {
s := script.New([]byte(`a := b`))
err := s.Add("b", 5)
assert.NoError(t, err)
assert.True(t, s.Remove("b")) // b is removed
_, err = s.Compile() // should not compile because b is undefined
assert.Error(t, err)
}
func TestScript_Run(t *testing.T) {
s := script.New([]byte(`a := b`))
err := s.Add("b", 5)
assert.NoError(t, err)
c, err := s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", int64(5))
}
func TestScript_SetBuiltinFunctions(t *testing.T) {
s := script.New([]byte(`a := len([1, 2, 3])`))
c, err := s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", int64(3))
s = script.New([]byte(`a := len([1, 2, 3])`))
s.SetBuiltinFunctions([]*objects.BuiltinFunction{&objects.Builtins[3]})
c, err = s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", int64(3))
s.SetBuiltinFunctions([]*objects.BuiltinFunction{&objects.Builtins[0]})
_, err = s.Run()
assert.Error(t, err)
s.SetBuiltinFunctions(nil)
_, err = s.Run()
assert.Error(t, err)
s = script.New([]byte(`a := import("b")`))
s.SetUserModuleLoader(func(name string) ([]byte, error) {
if name == "b" {
return []byte(`export import("c")`), nil
} else if name == "c" {
return []byte("export len([1, 2, 3])"), nil
}
return nil, errors.New("module not found")
})
s.SetBuiltinFunctions([]*objects.BuiltinFunction{&objects.Builtins[3]})
c, err = s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", int64(3))
}
func TestScript_SetBuiltinModules(t *testing.T) {
s := script.New([]byte(`math := import("math"); a := math.abs(-19.84)`))
s.SetBuiltinModules(map[string]*objects.ImmutableMap{
"math": objectPtr(&objects.ImmutableMap{
Value: map[string]objects.Object{
"abs": &objects.UserFunction{Name: "abs", Value: func(args ...objects.Object) (ret objects.Object, err error) {
v, _ := objects.ToFloat64(args[0])
return &objects.Float{Value: math.Abs(v)}, nil
}},
},
}),
})
c, err := s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", 19.84)
c, err = s.Run()
assert.NoError(t, err)
assert.NotNil(t, c)
compiledGet(t, c, "a", 19.84)
s.SetBuiltinModules(map[string]*objects.ImmutableMap{"os": objectPtr(&objects.ImmutableMap{Value: map[string]objects.Object{}})})
_, err = s.Run()
assert.Error(t, err)
s.SetBuiltinModules(nil)
_, err = s.Run()
assert.Error(t, err)
}
func TestScript_SetMaxConstObjects(t *testing.T) {
// one constant '5'
s := script.New([]byte(`a := 5`))
s.SetMaxConstObjects(1) // limit = 1
_, err := s.Compile()
assert.NoError(t, err)
s.SetMaxConstObjects(0) // limit = 0
_, err = s.Compile()
assert.Equal(t, "exceeding constant objects limit: 1", err.Error())
// two constants '5' and '1'
s = script.New([]byte(`a := 5 + 1`))
s.SetMaxConstObjects(2) // limit = 2
_, err = s.Compile()
assert.NoError(t, err)
s.SetMaxConstObjects(1) // limit = 1
_, err = s.Compile()
assert.Equal(t, "exceeding constant objects limit: 2", err.Error())
// duplicates will be removed
s = script.New([]byte(`a := 5 + 5`))
s.SetMaxConstObjects(1) // limit = 1
_, err = s.Compile()
assert.NoError(t, err)
s.SetMaxConstObjects(0) // limit = 0
_, err = s.Compile()
assert.Equal(t, "exceeding constant objects limit: 1", err.Error())
// no limit set
s = script.New([]byte(`a := 1 + 2 + 3 + 4 + 5`))
_, err = s.Compile()
assert.NoError(t, err)
}
func objectPtr(o objects.Object) *objects.ImmutableMap {
return o.(*objects.ImmutableMap)
}