xgo/compiler_test.go
2019-12-20 11:40:38 -08:00

1340 lines
43 KiB
Go

package tengo_test
import (
"fmt"
"strings"
"testing"
"github.com/d5/tengo"
"github.com/d5/tengo/internal"
"github.com/d5/tengo/internal/require"
)
func TestCompiler_Compile(t *testing.T) {
expectCompile(t, `1 + 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1; 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1 - 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 12),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1 * 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 13),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `2 / 1`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 14),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(2),
intObject(1))))
expectCompile(t, `true`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `false`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpFalse),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `1 > 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 39),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1 < 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 39),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(2),
intObject(1))))
expectCompile(t, `1 >= 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 44),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1 <= 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 44),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(2),
intObject(1))))
expectCompile(t, `1 == 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpEqual),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `1 != 2`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpNotEqual),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `true == false`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpFalse),
internal.MakeInstruction(internal.OpEqual),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `true != false`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpFalse),
internal.MakeInstruction(internal.OpNotEqual),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `-1`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpMinus),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1))))
expectCompile(t, `!true`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpLNot),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `if true { 10 }; 3333`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue), // 0000
internal.MakeInstruction(internal.OpJumpFalsy, 8), // 0001
internal.MakeInstruction(internal.OpConstant, 0), // 0004
internal.MakeInstruction(internal.OpPop), // 0007
internal.MakeInstruction(internal.OpConstant, 1), // 0008
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)), // 0011
objectsArray(
intObject(10),
intObject(3333))))
expectCompile(t, `if (true) { 10 } else { 20 }; 3333;`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpTrue), // 0000
internal.MakeInstruction(internal.OpJumpFalsy, 11), // 0001
internal.MakeInstruction(internal.OpConstant, 0), // 0004
internal.MakeInstruction(internal.OpPop), // 0007
internal.MakeInstruction(internal.OpJump, 15), // 0008
internal.MakeInstruction(internal.OpConstant, 1), // 0011
internal.MakeInstruction(internal.OpPop), // 0014
internal.MakeInstruction(internal.OpConstant, 2), // 0015
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)), // 0018
objectsArray(
intObject(10),
intObject(20),
intObject(3333))))
expectCompile(t, `"kami"`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
stringObject("kami"))))
expectCompile(t, `"ka" + "mi"`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
stringObject("ka"),
stringObject("mi"))))
expectCompile(t, `a := 1; b := 2; a += b`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetGlobal, 1),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `a := 1; b := 2; a /= b`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetGlobal, 1),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 1),
internal.MakeInstruction(internal.OpBinaryOp, 14),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2))))
expectCompile(t, `[]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpArray, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `[1, 2, 3]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3))))
expectCompile(t, `[1 + 2, 3 - 4, 5 * 6]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpBinaryOp, 12),
internal.MakeInstruction(internal.OpConstant, 4),
internal.MakeInstruction(internal.OpConstant, 5),
internal.MakeInstruction(internal.OpBinaryOp, 13),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3),
intObject(4),
intObject(5),
intObject(6))))
expectCompile(t, `{}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpMap, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `{a: 2, b: 4, c: 6}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpConstant, 4),
internal.MakeInstruction(internal.OpConstant, 5),
internal.MakeInstruction(internal.OpMap, 6),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
stringObject("a"),
intObject(2),
stringObject("b"),
intObject(4),
stringObject("c"),
intObject(6))))
expectCompile(t, `{a: 2 + 3, b: 5 * 6}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpConstant, 4),
internal.MakeInstruction(internal.OpConstant, 5),
internal.MakeInstruction(internal.OpBinaryOp, 13),
internal.MakeInstruction(internal.OpMap, 4),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
stringObject("a"),
intObject(2),
intObject(3),
stringObject("b"),
intObject(5),
intObject(6))))
expectCompile(t, `[1, 2, 3][1 + 1]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3))))
expectCompile(t, `{a: 2}[2 - 1]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpMap, 2),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpBinaryOp, 12),
internal.MakeInstruction(internal.OpIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
stringObject("a"),
intObject(2),
intObject(1))))
expectCompile(t, `[1, 2, 3][:]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpNull),
internal.MakeInstruction(internal.OpNull),
internal.MakeInstruction(internal.OpSliceIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3))))
expectCompile(t, `[1, 2, 3][0 : 2]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSliceIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3),
intObject(0))))
expectCompile(t, `[1, 2, 3][:2]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpNull),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSliceIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3))))
expectCompile(t, `[1, 2, 3][0:]`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpArray, 3),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpNull),
internal.MakeInstruction(internal.OpSliceIndex),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3),
intObject(0))))
expectCompile(t, `func() { return 5 + 10 }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(5),
intObject(10),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `func() { 5 + 10 }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(5),
intObject(10),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `func() { 1; 2 }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `func() { 1; return 2 }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `func() { if(true) { return 1 } else { return 2 } }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpTrue), // 0000
internal.MakeInstruction(internal.OpJumpFalsy, 9), // 0001
internal.MakeInstruction(internal.OpConstant, 0), // 0004
internal.MakeInstruction(internal.OpReturn, 1), // 0007
internal.MakeInstruction(internal.OpConstant, 1), // 0009
internal.MakeInstruction(internal.OpReturn, 1))))) // 0012
expectCompile(t, `func() { 1; if(true) { 2 } else { 3 }; 4 }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 4),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3),
intObject(4),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0), // 0000
internal.MakeInstruction(internal.OpPop), // 0003
internal.MakeInstruction(internal.OpTrue), // 0004
internal.MakeInstruction(internal.OpJumpFalsy, 15), // 0005
internal.MakeInstruction(internal.OpConstant, 1), // 0008
internal.MakeInstruction(internal.OpPop), // 0011
internal.MakeInstruction(internal.OpJump, 19), // 0012
internal.MakeInstruction(internal.OpConstant, 2), // 0015
internal.MakeInstruction(internal.OpPop), // 0018
internal.MakeInstruction(internal.OpConstant, 3), // 0019
internal.MakeInstruction(internal.OpPop), // 0022
internal.MakeInstruction(internal.OpReturn, 0))))) // 0023
expectCompile(t, `func() { }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `func() { 24 }()`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpCall, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(24),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `func() { return 24 }()`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpCall, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(24),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `noArg := func() { 24 }; noArg();`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpCall, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(24),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `noArg := func() { return 24 }; noArg();`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpCall, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(24),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `n := 55; func() { n };`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(55),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `func() { n := 55; return n }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(55),
compiledFunction(1, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `func() { a := 55; b := 77; return a + b }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(55),
intObject(77),
compiledFunction(2, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpDefineLocal, 1),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `f1 := func(a) { return a }; f1(24);`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpCall, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpReturn, 1)),
intObject(24))))
expectCompile(t, `varTest := func(...a) { return a }; varTest(1,2,3);`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpCall, 3),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpReturn, 1)),
intObject(1), intObject(2), intObject(3))))
expectCompile(t, `f1 := func(a, b, c) { a; b; return c; }; f1(24, 25, 26);`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpCall, 3),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(3, 3,
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpGetLocal, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpGetLocal, 2),
internal.MakeInstruction(internal.OpReturn, 1)),
intObject(24),
intObject(25),
intObject(26))))
expectCompile(t, `func() { n := 55; n = 23; return n }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(55),
intObject(23),
compiledFunction(1, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `len([]);`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpGetBuiltin, 0),
internal.MakeInstruction(internal.OpArray, 0),
internal.MakeInstruction(internal.OpCall, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `func() { return len([]) }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpGetBuiltin, 0),
internal.MakeInstruction(internal.OpArray, 0),
internal.MakeInstruction(internal.OpCall, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `func(a) { func(b) { return a + b } }`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetFree, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpReturn, 1)),
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetLocalPtr, 0),
internal.MakeInstruction(internal.OpClosure, 0, 1),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpReturn, 0)))))
expectCompile(t, `
func(a) {
return func(b) {
return func(c) {
return a + b + c
}
}
}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetFree, 0),
internal.MakeInstruction(internal.OpGetFree, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpReturn, 1)),
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetFreePtr, 0),
internal.MakeInstruction(internal.OpGetLocalPtr, 0),
internal.MakeInstruction(internal.OpClosure, 0, 2),
internal.MakeInstruction(internal.OpReturn, 1)),
compiledFunction(1, 1,
internal.MakeInstruction(internal.OpGetLocalPtr, 0),
internal.MakeInstruction(internal.OpClosure, 1, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `
g := 55;
func() {
a := 66;
return func() {
b := 77;
return func() {
c := 88;
return g + a + b + c;
}
}
}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 6),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(55),
intObject(66),
intObject(77),
intObject(88),
compiledFunction(1, 0,
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpGetFree, 0),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpGetFree, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpReturn, 1)),
compiledFunction(1, 0,
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetFreePtr, 0),
internal.MakeInstruction(internal.OpGetLocalPtr, 0),
internal.MakeInstruction(internal.OpClosure, 4, 2),
internal.MakeInstruction(internal.OpReturn, 1)),
compiledFunction(1, 0,
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetLocalPtr, 0),
internal.MakeInstruction(internal.OpClosure, 5, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `for i:=0; i<10; i++ {}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpBinaryOp, 39),
internal.MakeInstruction(internal.OpJumpFalsy, 31),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpJump, 6),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(0),
intObject(10),
intObject(1))))
expectCompile(t, `m := {}; for k, v in m {}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpMap, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpIteratorInit),
internal.MakeInstruction(internal.OpSetGlobal, 1),
internal.MakeInstruction(internal.OpGetGlobal, 1),
internal.MakeInstruction(internal.OpIteratorNext),
internal.MakeInstruction(internal.OpJumpFalsy, 37),
internal.MakeInstruction(internal.OpGetGlobal, 1),
internal.MakeInstruction(internal.OpIteratorKey),
internal.MakeInstruction(internal.OpSetGlobal, 2),
internal.MakeInstruction(internal.OpGetGlobal, 1),
internal.MakeInstruction(internal.OpIteratorValue),
internal.MakeInstruction(internal.OpSetGlobal, 3),
internal.MakeInstruction(internal.OpJump, 13),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray()))
expectCompile(t, `a := 0; a == 0 && a != 1 || a < 1`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpEqual),
internal.MakeInstruction(internal.OpAndJump, 23),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpNotEqual),
internal.MakeInstruction(internal.OpOrJump, 34),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpBinaryOp, 39),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(0),
intObject(1))))
// unknown module name
expectCompileError(t, `import("user1")`, "module 'user1' not found")
// too many errors
expectCompileError(t, `
r["x"] = {
@a:1,
@b:1,
@c:1,
@d:1,
@e:1,
@f:1,
@g:1,
@h:1,
@i:1,
@j:1,
@k:1
}
`, "Parse Error: illegal character U+0040 '@'\n\tat test:3:5 (and 10 more errors)")
expectCompileError(t, `import("")`, "empty module name")
}
func TestCompilerErrorReport(t *testing.T) {
expectCompileError(t, `import("user1")`,
"Compile Error: module 'user1' not found\n\tat test:1:1")
expectCompileError(t, `a = 1`,
"Compile Error: unresolved reference 'a'\n\tat test:1:1")
expectCompileError(t, `a, b := 1, 2`,
"Compile Error: tuple assignment not allowed\n\tat test:1:1")
expectCompileError(t, `a.b := 1`,
"not allowed with selector")
expectCompileError(t, `a:=1; a:=3`,
"Compile Error: 'a' redeclared in this block\n\tat test:1:7")
expectCompileError(t, `return 5`,
"Compile Error: return not allowed outside function\n\tat test:1:1")
expectCompileError(t, `func() { break }`,
"Compile Error: break not allowed outside loop\n\tat test:1:10")
expectCompileError(t, `func() { continue }`,
"Compile Error: continue not allowed outside loop\n\tat test:1:10")
expectCompileError(t, `func() { export 5 }`,
"Compile Error: export not allowed inside function\n\tat test:1:10")
}
func TestCompilerDeadCode(t *testing.T) {
expectCompile(t, `
func() {
a := 4
return a
b := 5 // dead code from here
c := a
return b
}`,
bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(4),
intObject(5),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `
func() {
if true {
return 5
a := 4 // dead code from here
b := a
return b
} else {
return 4
c := 5 // dead code from here
d := c
return d
}
}`, bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(5),
intObject(4),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpJumpFalsy, 9),
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpReturn, 1),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `
func() {
a := 1
for {
if a == 5 {
return 10
}
5 + 5
return 20
b := a
return b
}
}`, bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 4),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(5),
intObject(10),
intObject(20),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpEqual),
internal.MakeInstruction(internal.OpJumpFalsy, 19),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpReturn, 1),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpBinaryOp, 11),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpReturn, 1)))))
expectCompile(t, `
func() {
if true {
return 5
a := 4 // dead code from here
b := a
return b
} else {
return 4
c := 5 // dead code from here
d := c
return d
}
}`, bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(5),
intObject(4),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpTrue),
internal.MakeInstruction(internal.OpJumpFalsy, 9),
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpReturn, 1),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpReturn, 1)))))
}
func TestCompilerScopes(t *testing.T) {
expectCompile(t, `
if a := 1; a {
a = 2
b := a
} else {
a = 3
b := a
}`, bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpJumpFalsy, 27),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpSetGlobal, 1),
internal.MakeInstruction(internal.OpJump, 39),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpSetGlobal, 0),
internal.MakeInstruction(internal.OpGetGlobal, 0),
internal.MakeInstruction(internal.OpSetGlobal, 1),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3))))
expectCompile(t, `
func() {
if a := 1; a {
a = 2
b := a
} else {
a = 3
b := a
}
}`, bytecode(
concatInsts(
internal.MakeInstruction(internal.OpConstant, 3),
internal.MakeInstruction(internal.OpPop),
internal.MakeInstruction(internal.OpSuspend)),
objectsArray(
intObject(1),
intObject(2),
intObject(3),
compiledFunction(0, 0,
internal.MakeInstruction(internal.OpConstant, 0),
internal.MakeInstruction(internal.OpDefineLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpJumpFalsy, 22),
internal.MakeInstruction(internal.OpConstant, 1),
internal.MakeInstruction(internal.OpSetLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpDefineLocal, 1),
internal.MakeInstruction(internal.OpJump, 31),
internal.MakeInstruction(internal.OpConstant, 2),
internal.MakeInstruction(internal.OpSetLocal, 0),
internal.MakeInstruction(internal.OpGetLocal, 0),
internal.MakeInstruction(internal.OpDefineLocal, 1),
internal.MakeInstruction(internal.OpReturn, 0)))))
}
func concatInsts(instructions ...[]byte) []byte {
var concat []byte
for _, i := range instructions {
concat = append(concat, i...)
}
return concat
}
func bytecode(
instructions []byte,
constants []tengo.Object,
) *tengo.Bytecode {
return &tengo.Bytecode{
FileSet: internal.NewFileSet(),
MainFunction: &tengo.CompiledFunction{Instructions: instructions},
Constants: constants,
}
}
func expectCompile(
t *testing.T,
input string,
expected *tengo.Bytecode,
) {
actual, trace, err := traceCompile(input, nil)
var ok bool
defer func() {
if !ok {
for _, tr := range trace {
t.Log(tr)
}
}
}()
require.NoError(t, err)
equalBytecode(t, expected, actual)
ok = true
}
func expectCompileError(t *testing.T, input, expected string) {
_, trace, err := traceCompile(input, nil)
var ok bool
defer func() {
if !ok {
for _, tr := range trace {
t.Log(tr)
}
}
}()
require.Error(t, err)
require.True(t, strings.Contains(err.Error(), expected),
"expected error string: %s, got: %s", expected, err.Error())
ok = true
}
func equalBytecode(t *testing.T, expected, actual *tengo.Bytecode) {
require.Equal(t, expected.MainFunction, actual.MainFunction)
equalConstants(t, expected.Constants, actual.Constants)
}
func equalConstants(t *testing.T, expected, actual []tengo.Object) {
require.Equal(t, len(expected), len(actual))
for i := 0; i < len(expected); i++ {
require.Equal(t, expected[i], actual[i])
}
}
type compileTracer struct {
Out []string
}
func (o *compileTracer) Write(p []byte) (n int, err error) {
o.Out = append(o.Out, string(p))
return len(p), nil
}
func traceCompile(
input string,
symbols map[string]tengo.Object,
) (res *tengo.Bytecode, trace []string, err error) {
fileSet := internal.NewFileSet()
file := fileSet.AddFile("test", -1, len(input))
p := internal.NewParser(file, []byte(input), nil)
symTable := internal.NewSymbolTable()
for name := range symbols {
symTable.Define(name)
}
for idx, fn := range tengo.GetAllBuiltinFunctions() {
symTable.DefineBuiltin(idx, fn.Name)
}
tr := &compileTracer{}
c := tengo.NewCompiler(file, symTable, nil, nil, tr)
parsed, err := p.ParseFile()
if err != nil {
return
}
err = c.Compile(parsed)
res = c.Bytecode()
res.RemoveDuplicates()
{
trace = append(trace, fmt.Sprintf("Compiler Trace:\n%s",
strings.Join(tr.Out, "")))
trace = append(trace, fmt.Sprintf("Compiled Constants:\n%s",
strings.Join(res.FormatConstants(), "\n")))
trace = append(trace, fmt.Sprintf("Compiled Instructions:\n%s\n",
strings.Join(res.FormatInstructions(), "\n")))
}
if err != nil {
return
}
return
}
func objectsArray(o ...tengo.Object) []tengo.Object {
return o
}
func intObject(v int64) *tengo.Int {
return &tengo.Int{Value: v}
}
func stringObject(v string) *tengo.String {
return &tengo.String{Value: v}
}
func compiledFunction(
numLocals, numParams int,
insts ...[]byte,
) *tengo.CompiledFunction {
return &tengo.CompiledFunction{
Instructions: concatInsts(insts...),
NumLocals: numLocals,
NumParameters: numParams,
}
}