2019-01-09 23:39:40 +03:00
|
|
|
package script_test
|
|
|
|
|
|
|
|
import (
|
2019-03-01 05:41:29 +03:00
|
|
|
"errors"
|
2019-03-04 21:21:39 +03:00
|
|
|
"math"
|
2019-01-09 23:39:40 +03:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/d5/tengo/assert"
|
2019-02-28 19:26:25 +03:00
|
|
|
"github.com/d5/tengo/objects"
|
2019-01-09 23:39:40 +03:00
|
|
|
"github.com/d5/tengo/script"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestScript_Add(t *testing.T) {
|
2019-03-01 20:28:41 +03:00
|
|
|
s := script.New([]byte(`a := b; c := test(b); d := test(5)`))
|
2019-01-09 23:39:40 +03:00
|
|
|
assert.NoError(t, s.Add("b", 5)) // b = 5
|
|
|
|
assert.NoError(t, s.Add("b", "foo")) // b = "foo" (re-define before compilation)
|
2019-03-01 20:28:41 +03:00
|
|
|
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
|
|
|
|
}))
|
2019-01-09 23:39:40 +03:00
|
|
|
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())
|
2019-03-01 20:28:41 +03:00
|
|
|
assert.Equal(t, int64(0), c.Get("c").Value())
|
|
|
|
assert.Equal(t, int64(6), c.Get("d").Value())
|
2019-01-09 23:39:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestScript_Remove(t *testing.T) {
|
2019-01-11 09:34:28 +03:00
|
|
|
s := script.New([]byte(`a := b`))
|
2019-01-09 23:39:40 +03:00
|
|
|
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)
|
|
|
|
}
|
2019-01-15 21:14:16 +03:00
|
|
|
|
|
|
|
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))
|
|
|
|
}
|
2019-01-25 00:08:53 +03:00
|
|
|
|
2019-02-28 19:26:25 +03:00
|
|
|
func TestScript_SetBuiltinFunctions(t *testing.T) {
|
2019-01-25 00:08:53 +03:00
|
|
|
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))
|
2019-02-28 19:26:25 +03:00
|
|
|
|
|
|
|
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)
|
2019-01-25 00:08:53 +03:00
|
|
|
_, err = s.Run()
|
|
|
|
assert.Error(t, err)
|
2019-03-01 05:41:29 +03:00
|
|
|
|
|
|
|
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))
|
2019-01-25 00:08:53 +03:00
|
|
|
}
|
|
|
|
|
2019-02-28 19:26:25 +03:00
|
|
|
func TestScript_SetBuiltinModules(t *testing.T) {
|
2019-01-25 00:08:53 +03:00
|
|
|
s := script.New([]byte(`math := import("math"); a := math.abs(-19.84)`))
|
2019-03-04 21:21:39 +03:00
|
|
|
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
|
|
|
|
}},
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
})
|
2019-01-25 00:08:53 +03:00
|
|
|
c, err := s.Run()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, c)
|
|
|
|
compiledGet(t, c, "a", 19.84)
|
2019-02-28 19:26:25 +03:00
|
|
|
|
|
|
|
c, err = s.Run()
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.NotNil(t, c)
|
|
|
|
compiledGet(t, c, "a", 19.84)
|
|
|
|
|
2019-03-04 21:21:39 +03:00
|
|
|
s.SetBuiltinModules(map[string]*objects.ImmutableMap{"os": objectPtr(&objects.ImmutableMap{Value: map[string]objects.Object{}})})
|
2019-01-25 00:08:53 +03:00
|
|
|
_, err = s.Run()
|
|
|
|
assert.Error(t, err)
|
2019-02-28 19:26:25 +03:00
|
|
|
|
|
|
|
s.SetBuiltinModules(nil)
|
|
|
|
_, err = s.Run()
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
2019-03-07 04:20:05 +03:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2019-02-28 19:26:25 +03:00
|
|
|
func objectPtr(o objects.Object) *objects.ImmutableMap {
|
|
|
|
return o.(*objects.ImmutableMap)
|
2019-01-25 00:08:53 +03:00
|
|
|
}
|