2019-01-09 23:39:40 +03:00
|
|
|
package script_test
|
|
|
|
|
|
|
|
import (
|
2019-01-20 15:59:43 +03:00
|
|
|
"context"
|
2019-01-09 23:39:40 +03:00
|
|
|
"testing"
|
2019-01-20 15:59:43 +03:00
|
|
|
"time"
|
2019-01-09 23:39:40 +03:00
|
|
|
|
|
|
|
"github.com/d5/tengo/assert"
|
|
|
|
"github.com/d5/tengo/script"
|
|
|
|
)
|
|
|
|
|
|
|
|
type M map[string]interface{}
|
|
|
|
|
|
|
|
func TestCompiled_Get(t *testing.T) {
|
|
|
|
// simple script
|
2019-01-11 09:34:28 +03:00
|
|
|
c := compile(t, `a := 5`, nil)
|
2019-01-09 23:39:40 +03:00
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGet(t, c, "a", int64(5))
|
|
|
|
|
|
|
|
// user-defined variables
|
2019-01-11 09:34:28 +03:00
|
|
|
compileError(t, `a := b`, nil) // compile error because "b" is not defined
|
|
|
|
c = compile(t, `a := b`, M{"b": "foo"}) // now compile with b = "foo" defined
|
|
|
|
compiledGet(t, c, "a", nil) // a = undefined; because it's before Compiled.Run()
|
|
|
|
compiledRun(t, c) // Compiled.Run()
|
|
|
|
compiledGet(t, c, "a", "foo") // a = "foo"
|
2019-01-09 23:39:40 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestCompiled_GetAll(t *testing.T) {
|
2019-01-11 09:34:28 +03:00
|
|
|
c := compile(t, `a := 5`, nil)
|
2019-01-09 23:39:40 +03:00
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGetAll(t, c, M{"a": int64(5)})
|
|
|
|
|
2019-01-11 09:34:28 +03:00
|
|
|
c = compile(t, `a := b`, M{"b": "foo"})
|
2019-01-09 23:39:40 +03:00
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGetAll(t, c, M{"a": "foo", "b": "foo"})
|
|
|
|
|
2019-01-11 09:34:28 +03:00
|
|
|
c = compile(t, `a := b; b = 5`, M{"b": "foo"})
|
2019-01-09 23:39:40 +03:00
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGetAll(t, c, M{"a": "foo", "b": int64(5)})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCompiled_IsDefined(t *testing.T) {
|
2019-01-11 09:34:28 +03:00
|
|
|
c := compile(t, `a := 5`, nil)
|
2019-01-09 23:39:40 +03:00
|
|
|
compiledIsDefined(t, c, "a", false) // a is not defined before Run()
|
|
|
|
compiledRun(t, c)
|
|
|
|
compiledIsDefined(t, c, "a", true)
|
|
|
|
compiledIsDefined(t, c, "b", false)
|
|
|
|
}
|
|
|
|
|
2019-02-01 22:09:12 +03:00
|
|
|
func TestCompiled_Set(t *testing.T) {
|
|
|
|
c := compile(t, `a := b`, M{"b": "foo"})
|
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGet(t, c, "a", "foo")
|
|
|
|
|
|
|
|
// replace value of 'b'
|
|
|
|
err := c.Set("b", "bar")
|
|
|
|
assert.NoError(t, err)
|
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGet(t, c, "a", "bar")
|
|
|
|
|
|
|
|
// try to replace undefined variable
|
|
|
|
err = c.Set("c", 1984)
|
|
|
|
assert.Error(t, err) // 'c' is not defined
|
|
|
|
|
|
|
|
// case #2
|
|
|
|
c = compile(t, `
|
|
|
|
a := func() {
|
|
|
|
return func() {
|
|
|
|
return b + 5
|
|
|
|
}()
|
|
|
|
}()`, M{"b": 5})
|
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGet(t, c, "a", int64(10))
|
|
|
|
err = c.Set("b", 10)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
compiledRun(t, c)
|
|
|
|
compiledGet(t, c, "a", int64(15))
|
|
|
|
}
|
|
|
|
|
2019-01-20 15:59:43 +03:00
|
|
|
func TestCompiled_RunContext(t *testing.T) {
|
|
|
|
// machine completes normally
|
|
|
|
c := compile(t, `a := 5`, nil)
|
|
|
|
err := c.RunContext(context.Background())
|
|
|
|
assert.NoError(t, err)
|
|
|
|
compiledGet(t, c, "a", int64(5))
|
|
|
|
|
|
|
|
// cancelled
|
|
|
|
c = compile(t, `for true {}`, nil)
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
go func() {
|
|
|
|
time.Sleep(1 * time.Millisecond)
|
|
|
|
cancel()
|
|
|
|
}()
|
|
|
|
err = c.RunContext(ctx)
|
|
|
|
assert.Equal(t, context.Canceled, err)
|
|
|
|
|
|
|
|
// timeout
|
|
|
|
c = compile(t, `for true {}`, nil)
|
|
|
|
ctx, cancel = context.WithTimeout(context.Background(), 1*time.Millisecond)
|
|
|
|
defer cancel()
|
|
|
|
err = c.RunContext(ctx)
|
|
|
|
assert.Equal(t, context.DeadlineExceeded, err)
|
|
|
|
}
|
|
|
|
|
2019-01-09 23:39:40 +03:00
|
|
|
func compile(t *testing.T, input string, vars M) *script.Compiled {
|
|
|
|
s := script.New([]byte(input))
|
|
|
|
for vn, vv := range vars {
|
|
|
|
err := s.Add(vn, vv)
|
|
|
|
if !assert.NoError(t, err) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := s.Compile()
|
|
|
|
if !assert.NoError(t, err) || !assert.NotNil(t, c) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
func compileError(t *testing.T, input string, vars M) bool {
|
|
|
|
s := script.New([]byte(input))
|
|
|
|
for vn, vv := range vars {
|
|
|
|
err := s.Add(vn, vv)
|
|
|
|
if !assert.NoError(t, err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_, err := s.Compile()
|
|
|
|
|
|
|
|
return assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func compiledRun(t *testing.T, c *script.Compiled) bool {
|
|
|
|
err := c.Run()
|
|
|
|
|
|
|
|
return assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func compiledGet(t *testing.T, c *script.Compiled, name string, expected interface{}) bool {
|
|
|
|
v := c.Get(name)
|
|
|
|
if !assert.NotNil(t, v) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return assert.Equal(t, expected, v.Value())
|
|
|
|
}
|
|
|
|
|
|
|
|
func compiledGetAll(t *testing.T, c *script.Compiled, expected M) bool {
|
|
|
|
vars := c.GetAll()
|
|
|
|
|
|
|
|
if !assert.Equal(t, len(expected), len(vars)) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
for k, v := range expected {
|
|
|
|
var found bool
|
|
|
|
for _, e := range vars {
|
|
|
|
if e.Name() == k {
|
|
|
|
if !assert.Equal(t, v, e.Value()) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !found {
|
|
|
|
assert.Fail(t, "variable '%s' not found", k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func compiledIsDefined(t *testing.T, c *script.Compiled, name string, expected bool) bool {
|
|
|
|
return assert.Equal(t, expected, c.IsDefined(name))
|
|
|
|
}
|