adcf05d26f
* parsing variable function types * finished variadic functions * fix case where number of passed args to variadic function is 0 * remove extraneous OpVarArgs * allow multiple variables in variadic function declaration * fix IdentList string method to print multi-arg variadic functions correctly * round 2 of fix IdentList string method to print multi-arg variadic functions correctly round 2 of fix IdentList string method to print multi-arg variadic functions correctly * clean up tasks in OpCall handling, add tests for variadic closures * cleanup for pr - add syntax documentation - cleanup parseIdentList - cleanup OpCall handling for functions and closures - cleanup tests
288 lines
6.9 KiB
Go
288 lines
6.9 KiB
Go
package runtime_test
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/d5/tengo/objects"
|
|
)
|
|
|
|
func TestFunction(t *testing.T) {
|
|
// function with no "return" statement returns "invalid" value.
|
|
expect(t, `f1 := func() {}; out = f1();`, nil, objects.UndefinedValue)
|
|
expect(t, `f1 := func() {}; f2 := func() { return f1(); }; f1(); out = f2();`, nil, objects.UndefinedValue)
|
|
expect(t, `f := func(x) { x; }; out = f(5);`, nil, objects.UndefinedValue)
|
|
|
|
expect(t, `f := func(...x) { return x; }; out = f(1,2,3);`, nil, ARR{1, 2, 3})
|
|
|
|
expect(t, `f := func(a, b, ...x) { return [a, b, x]; }; out = f(8,9,1,2,3);`, nil, ARR{8, 9, ARR{1, 2, 3}})
|
|
|
|
expect(t, `f := func(v) { x := 2; return func(a, ...b){ return [a, b, v+x]}; }; out = f(5)("a", "b");`, nil,
|
|
ARR{"a", ARR{"b"}, 7})
|
|
|
|
expect(t, `f := func(...x) { return x; }; out = f();`, nil, &objects.Array{Value: []objects.Object{}})
|
|
|
|
expect(t, `f := func(a, b, ...x) { return [a, b, x]; }; out = f(8, 9);`, nil,
|
|
ARR{8, 9, ARR{}})
|
|
|
|
expect(t, `f := func(v) { x := 2; return func(a, ...b){ return [a, b, v+x]}; }; out = f(5)("a");`, nil,
|
|
ARR{"a", ARR{}, 7})
|
|
|
|
expectError(t, `f := func(a, b, ...x) { return [a, b, x]; }; f();`, nil,
|
|
"Runtime Error: wrong number of arguments: want>=2, got=0\n\tat test:1:46")
|
|
|
|
expectError(t, `f := func(a, b, ...x) { return [a, b, x]; }; f(1);`, nil,
|
|
"Runtime Error: wrong number of arguments: want>=2, got=1\n\tat test:1:46")
|
|
|
|
expect(t, `f := func(x) { return x; }; out = f(5);`, nil, 5)
|
|
expect(t, `f := func(x) { return x * 2; }; out = f(5);`, nil, 10)
|
|
expect(t, `f := func(x, y) { return x + y; }; out = f(5, 5);`, nil, 10)
|
|
expect(t, `f := func(x, y) { return x + y; }; out = f(5 + 5, f(5, 5));`, nil, 20)
|
|
expect(t, `out = func(x) { return x; }(5)`, nil, 5)
|
|
expect(t, `x := 10; f := func(x) { return x; }; f(5); out = x;`, nil, 10)
|
|
|
|
expect(t, `
|
|
f2 := func(a) {
|
|
f1 := func(a) {
|
|
return a * 2;
|
|
};
|
|
|
|
return f1(a) * 3;
|
|
};
|
|
|
|
out = f2(10);
|
|
`, nil, 60)
|
|
|
|
// closures
|
|
expect(t, `
|
|
newAdder := func(x) {
|
|
return func(y) { return x + y };
|
|
};
|
|
|
|
add2 := newAdder(2);
|
|
out = add2(5);
|
|
`, nil, 7)
|
|
|
|
// function as a argument
|
|
expect(t, `
|
|
add := func(a, b) { return a + b };
|
|
sub := func(a, b) { return a - b };
|
|
applyFunc := func(a, b, f) { return f(a, b) };
|
|
|
|
out = applyFunc(applyFunc(2, 2, add), 3, sub);
|
|
`, nil, 1)
|
|
|
|
expect(t, `f1 := func() { return 5 + 10; }; out = f1();`, nil, 15)
|
|
expect(t, `f1 := func() { return 1 }; f2 := func() { return 2 }; out = f1() + f2()`, nil, 3)
|
|
expect(t, `f1 := func() { return 1 }; f2 := func() { return f1() + 2 }; f3 := func() { return f2() + 3 }; out = f3()`, nil, 6)
|
|
expect(t, `f1 := func() { return 99; 100 }; out = f1();`, nil, 99)
|
|
expect(t, `f1 := func() { return 99; return 100 }; out = f1();`, nil, 99)
|
|
expect(t, `f1 := func() { return 33; }; f2 := func() { return f1 }; out = f2()();`, nil, 33)
|
|
expect(t, `one := func() { one = 1; return one }; out = one()`, nil, 1)
|
|
expect(t, `three := func() { one := 1; two := 2; return one + two }; out = three()`, nil, 3)
|
|
expect(t, `three := func() { one := 1; two := 2; return one + two }; seven := func() { three := 3; four := 4; return three + four }; out = three() + seven()`, nil, 10)
|
|
expect(t, `
|
|
foo1 := func() {
|
|
foo := 50
|
|
return foo
|
|
}
|
|
foo2 := func() {
|
|
foo := 100
|
|
return foo
|
|
}
|
|
out = foo1() + foo2()`, nil, 150)
|
|
expect(t, `
|
|
g := 50;
|
|
minusOne := func() {
|
|
n := 1;
|
|
return g - n;
|
|
};
|
|
minusTwo := func() {
|
|
n := 2;
|
|
return g - n;
|
|
};
|
|
out = minusOne() + minusTwo()
|
|
`, nil, 97)
|
|
expect(t, `
|
|
f1 := func() {
|
|
f2 := func() { return 1; }
|
|
return f2
|
|
};
|
|
out = f1()()
|
|
`, nil, 1)
|
|
|
|
expect(t, `
|
|
f1 := func(a) { return a; };
|
|
out = f1(4)`, nil, 4)
|
|
expect(t, `
|
|
f1 := func(a, b) { return a + b; };
|
|
out = f1(1, 2)`, nil, 3)
|
|
|
|
expect(t, `
|
|
sum := func(a, b) {
|
|
c := a + b;
|
|
return c;
|
|
};
|
|
out = sum(1, 2);`, nil, 3)
|
|
|
|
expect(t, `
|
|
sum := func(a, b) {
|
|
c := a + b;
|
|
return c;
|
|
};
|
|
out = sum(1, 2) + sum(3, 4);`, nil, 10)
|
|
|
|
expect(t, `
|
|
sum := func(a, b) {
|
|
c := a + b
|
|
return c
|
|
};
|
|
outer := func() {
|
|
return sum(1, 2) + sum(3, 4)
|
|
};
|
|
out = outer();`, nil, 10)
|
|
|
|
expect(t, `
|
|
g := 10;
|
|
|
|
sum := func(a, b) {
|
|
c := a + b;
|
|
return c + g;
|
|
}
|
|
|
|
outer := func() {
|
|
return sum(1, 2) + sum(3, 4) + g;
|
|
}
|
|
|
|
out = outer() + g
|
|
`, nil, 50)
|
|
|
|
expectError(t, `func() { return 1; }(1)`, nil, "wrong number of arguments")
|
|
expectError(t, `func(a) { return a; }()`, nil, "wrong number of arguments")
|
|
expectError(t, `func(a, b) { return a + b; }(1)`, nil, "wrong number of arguments")
|
|
|
|
expect(t, `
|
|
f1 := func(a) {
|
|
return func() { return a; };
|
|
};
|
|
f2 := f1(99);
|
|
out = f2()
|
|
`, nil, 99)
|
|
|
|
expect(t, `
|
|
f1 := func(a, b) {
|
|
return func(c) { return a + b + c };
|
|
};
|
|
|
|
f2 := f1(1, 2);
|
|
out = f2(8);
|
|
`, nil, 11)
|
|
expect(t, `
|
|
f1 := func(a, b) {
|
|
c := a + b;
|
|
return func(d) { return c + d };
|
|
};
|
|
f2 := f1(1, 2);
|
|
out = f2(8);
|
|
`, nil, 11)
|
|
expect(t, `
|
|
f1 := func(a, b) {
|
|
c := a + b;
|
|
return func(d) {
|
|
e := d + c;
|
|
return func(f) { return e + f };
|
|
}
|
|
};
|
|
f2 := f1(1, 2);
|
|
f3 := f2(3);
|
|
out = f3(8);
|
|
`, nil, 14)
|
|
expect(t, `
|
|
a := 1;
|
|
f1 := func(b) {
|
|
return func(c) {
|
|
return func(d) { return a + b + c + d }
|
|
};
|
|
};
|
|
f2 := f1(2);
|
|
f3 := f2(3);
|
|
out = f3(8);
|
|
`, nil, 14)
|
|
expect(t, `
|
|
f1 := func(a, b) {
|
|
one := func() { return a; };
|
|
two := func() { return b; };
|
|
return func() { return one() + two(); }
|
|
};
|
|
f2 := f1(9, 90);
|
|
out = f2();
|
|
`, nil, 99)
|
|
|
|
// global function recursion
|
|
expect(t, `
|
|
fib := func(x) {
|
|
if x == 0 {
|
|
return 0
|
|
} else if x == 1 {
|
|
return 1
|
|
} else {
|
|
return fib(x-1) + fib(x-2)
|
|
}
|
|
}
|
|
out = fib(15)`, nil, 610)
|
|
|
|
// local function recursion
|
|
expect(t, `
|
|
out = func() {
|
|
sum := func(x) {
|
|
return x == 0 ? 0 : x + sum(x-1)
|
|
}
|
|
return sum(5)
|
|
}()`, nil, 15)
|
|
|
|
expectError(t, `return 5`, nil, "return not allowed outside function")
|
|
|
|
// closure and block scopes
|
|
expect(t, `
|
|
func() {
|
|
a := 10
|
|
func() {
|
|
b := 5
|
|
if true {
|
|
out = a + 5
|
|
}
|
|
}()
|
|
}()`, nil, 15)
|
|
expect(t, `
|
|
func() {
|
|
a := 10
|
|
b := func() { return 5 }
|
|
func() {
|
|
if b() {
|
|
out = a + b()
|
|
}
|
|
}()
|
|
}()`, nil, 15)
|
|
expect(t, `
|
|
func() {
|
|
a := 10
|
|
func() {
|
|
b := func() { return 5 }
|
|
func() {
|
|
if true {
|
|
out = a + b()
|
|
}
|
|
}()
|
|
}()
|
|
}()`, nil, 15)
|
|
|
|
// function skipping return
|
|
expect(t, `out = func() {}()`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { if v { return true } }(1)`, nil, true)
|
|
expect(t, `out = func(v) { if v { return true } }(0)`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { if v { } else { return true } }(1)`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { if v { return } }(1)`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { if v { return } }(0)`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { if v { } else { return } }(1)`, nil, objects.UndefinedValue)
|
|
expect(t, `out = func(v) { for ;;v++ { if v == 3 { return true } } }(1)`, nil, true)
|
|
expect(t, `out = func(v) { for ;;v++ { if v == 3 { break } } }(1)`, nil, objects.UndefinedValue)
|
|
}
|