module refactor (#148)
* wip * move print and JSON functions to modules * builtin functions are not replacable now * builtin functions are added for default nil symbol table * importables: builtin modules and source modules * refactoring runtime tests * fix tests * update documentation * cleanup * clean up cli * fix REPL prints
This commit is contained in:
parent
052ae5906b
commit
61890b15cb
74 changed files with 1625 additions and 1725 deletions
5
Makefile
5
Makefile
|
@ -1,10 +1,13 @@
|
||||||
vet:
|
vet:
|
||||||
go vet ./...
|
go vet ./...
|
||||||
|
|
||||||
|
generate:
|
||||||
|
go generate ./...
|
||||||
|
|
||||||
lint:
|
lint:
|
||||||
golint -set_exit_status ./...
|
golint -set_exit_status ./...
|
||||||
|
|
||||||
test: vet lint
|
test: generate vet lint
|
||||||
go test -race -cover ./...
|
go test -race -cover ./...
|
||||||
|
|
||||||
fmt:
|
fmt:
|
||||||
|
|
78
cli/cli.go
78
cli/cli.go
|
@ -23,7 +23,7 @@ const (
|
||||||
replPrompt = ">> "
|
replPrompt = ">> "
|
||||||
)
|
)
|
||||||
|
|
||||||
//Options represent REPL options
|
// Options represent CLI options
|
||||||
type Options struct {
|
type Options struct {
|
||||||
// Compile output file
|
// Compile output file
|
||||||
CompileOutput string
|
CompileOutput string
|
||||||
|
@ -40,16 +40,11 @@ type Options struct {
|
||||||
// Version
|
// Version
|
||||||
Version string
|
Version string
|
||||||
|
|
||||||
//Builtin modules
|
// Import modules
|
||||||
BuiltinModules map[string]objects.Object
|
Modules map[string]objects.Importable
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
// Run CLI
|
||||||
bm map[string]bool
|
|
||||||
builtinModules map[string]objects.Object
|
|
||||||
)
|
|
||||||
|
|
||||||
//Run REPL
|
|
||||||
func Run(options *Options) {
|
func Run(options *Options) {
|
||||||
if options.ShowHelp {
|
if options.ShowHelp {
|
||||||
doHelp()
|
doHelp()
|
||||||
|
@ -59,15 +54,9 @@ func Run(options *Options) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
builtinModules = options.BuiltinModules
|
|
||||||
bm = make(map[string]bool, len(builtinModules))
|
|
||||||
for k := range builtinModules {
|
|
||||||
bm[k] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if options.InputFile == "" {
|
if options.InputFile == "" {
|
||||||
// REPL
|
// REPL
|
||||||
runREPL(os.Stdin, os.Stdout)
|
runREPL(options.Modules, os.Stdin, os.Stdout)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,12 +67,12 @@ func Run(options *Options) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if options.CompileOutput != "" {
|
if options.CompileOutput != "" {
|
||||||
if err := compileOnly(inputData, options.InputFile, options.CompileOutput); err != nil {
|
if err := compileOnly(options.Modules, inputData, options.InputFile, options.CompileOutput); err != nil {
|
||||||
_, _ = fmt.Fprintln(os.Stderr, err.Error())
|
_, _ = fmt.Fprintln(os.Stderr, err.Error())
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
} else if filepath.Ext(options.InputFile) == sourceFileExt {
|
} else if filepath.Ext(options.InputFile) == sourceFileExt {
|
||||||
if err := compileAndRun(inputData, options.InputFile); err != nil {
|
if err := compileAndRun(options.Modules, inputData, options.InputFile); err != nil {
|
||||||
_, _ = fmt.Fprintln(os.Stderr, err.Error())
|
_, _ = fmt.Fprintln(os.Stderr, err.Error())
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
@ -127,8 +116,8 @@ func doHelp() {
|
||||||
fmt.Println()
|
fmt.Println()
|
||||||
}
|
}
|
||||||
|
|
||||||
func compileOnly(data []byte, inputFile, outputFile string) (err error) {
|
func compileOnly(modules map[string]objects.Importable, data []byte, inputFile, outputFile string) (err error) {
|
||||||
bytecode, err := compileSrc(data, filepath.Base(inputFile))
|
bytecode, err := compileSrc(modules, data, filepath.Base(inputFile))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -159,13 +148,13 @@ func compileOnly(data []byte, inputFile, outputFile string) (err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func compileAndRun(data []byte, inputFile string) (err error) {
|
func compileAndRun(modules map[string]objects.Importable, data []byte, inputFile string) (err error) {
|
||||||
bytecode, err := compileSrc(data, filepath.Base(inputFile))
|
bytecode, err := compileSrc(modules, data, filepath.Base(inputFile))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
machine := runtime.NewVM(bytecode, nil, nil, builtinModules, -1)
|
machine := runtime.NewVM(bytecode, nil, -1)
|
||||||
|
|
||||||
err = machine.Run()
|
err = machine.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -182,7 +171,7 @@ func runCompiled(data []byte) (err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
machine := runtime.NewVM(bytecode, nil, nil, builtinModules, -1)
|
machine := runtime.NewVM(bytecode, nil, -1)
|
||||||
|
|
||||||
err = machine.Run()
|
err = machine.Run()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -192,7 +181,7 @@ func runCompiled(data []byte) (err error) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func runREPL(in io.Reader, out io.Writer) {
|
func runREPL(modules map[string]objects.Importable, in io.Reader, out io.Writer) {
|
||||||
stdin := bufio.NewScanner(in)
|
stdin := bufio.NewScanner(in)
|
||||||
|
|
||||||
fileSet := source.NewFileSet()
|
fileSet := source.NewFileSet()
|
||||||
|
@ -203,6 +192,28 @@ func runREPL(in io.Reader, out io.Writer) {
|
||||||
symbolTable.DefineBuiltin(idx, fn.Name)
|
symbolTable.DefineBuiltin(idx, fn.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// embed println function
|
||||||
|
symbol := symbolTable.Define("__repl_println__")
|
||||||
|
globals[symbol.Index] = &objects.UserFunction{
|
||||||
|
Name: "println",
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
var printArgs []interface{}
|
||||||
|
for _, arg := range args {
|
||||||
|
if _, isUndefined := arg.(*objects.Undefined); isUndefined {
|
||||||
|
printArgs = append(printArgs, "<undefined>")
|
||||||
|
} else {
|
||||||
|
s, _ := objects.ToString(arg)
|
||||||
|
printArgs = append(printArgs, s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printArgs = append(printArgs, "\n")
|
||||||
|
_, _ = fmt.Print(printArgs...)
|
||||||
|
|
||||||
|
return
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
var constants []objects.Object
|
var constants []objects.Object
|
||||||
|
|
||||||
for {
|
for {
|
||||||
|
@ -225,7 +236,7 @@ func runREPL(in io.Reader, out io.Writer) {
|
||||||
|
|
||||||
file = addPrints(file)
|
file = addPrints(file)
|
||||||
|
|
||||||
c := compiler.NewCompiler(srcFile, symbolTable, constants, bm, nil)
|
c := compiler.NewCompiler(srcFile, symbolTable, constants, modules, nil)
|
||||||
if err := c.Compile(file); err != nil {
|
if err := c.Compile(file); err != nil {
|
||||||
_, _ = fmt.Fprintln(out, err.Error())
|
_, _ = fmt.Fprintln(out, err.Error())
|
||||||
continue
|
continue
|
||||||
|
@ -233,7 +244,7 @@ func runREPL(in io.Reader, out io.Writer) {
|
||||||
|
|
||||||
bytecode := c.Bytecode()
|
bytecode := c.Bytecode()
|
||||||
|
|
||||||
machine := runtime.NewVM(bytecode, globals, nil, builtinModules, -1)
|
machine := runtime.NewVM(bytecode, globals, -1)
|
||||||
if err := machine.Run(); err != nil {
|
if err := machine.Run(); err != nil {
|
||||||
_, _ = fmt.Fprintln(out, err.Error())
|
_, _ = fmt.Fprintln(out, err.Error())
|
||||||
continue
|
continue
|
||||||
|
@ -243,7 +254,7 @@ func runREPL(in io.Reader, out io.Writer) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func compileSrc(src []byte, filename string) (*compiler.Bytecode, error) {
|
func compileSrc(modules map[string]objects.Importable, src []byte, filename string) (*compiler.Bytecode, error) {
|
||||||
fileSet := source.NewFileSet()
|
fileSet := source.NewFileSet()
|
||||||
srcFile := fileSet.AddFile(filename, -1, len(src))
|
srcFile := fileSet.AddFile(filename, -1, len(src))
|
||||||
|
|
||||||
|
@ -253,7 +264,9 @@ func compileSrc(src []byte, filename string) (*compiler.Bytecode, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
c := compiler.NewCompiler(srcFile, nil, nil, bm, nil)
|
c := compiler.NewCompiler(srcFile, nil, nil, modules, nil)
|
||||||
|
c.EnableFileImport(true)
|
||||||
|
|
||||||
if err := c.Compile(file); err != nil {
|
if err := c.Compile(file); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -266,14 +279,13 @@ func compileSrc(src []byte, filename string) (*compiler.Bytecode, error) {
|
||||||
|
|
||||||
func addPrints(file *ast.File) *ast.File {
|
func addPrints(file *ast.File) *ast.File {
|
||||||
var stmts []ast.Stmt
|
var stmts []ast.Stmt
|
||||||
|
|
||||||
for _, s := range file.Stmts {
|
for _, s := range file.Stmts {
|
||||||
switch s := s.(type) {
|
switch s := s.(type) {
|
||||||
case *ast.ExprStmt:
|
case *ast.ExprStmt:
|
||||||
stmts = append(stmts, &ast.ExprStmt{
|
stmts = append(stmts, &ast.ExprStmt{
|
||||||
Expr: &ast.CallExpr{
|
Expr: &ast.CallExpr{
|
||||||
Func: &ast.Ident{
|
Func: &ast.Ident{Name: "__repl_println__"},
|
||||||
Name: "print",
|
|
||||||
},
|
|
||||||
Args: []ast.Expr{s.Expr},
|
Args: []ast.Expr{s.Expr},
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
@ -284,7 +296,7 @@ func addPrints(file *ast.File) *ast.File {
|
||||||
stmts = append(stmts, &ast.ExprStmt{
|
stmts = append(stmts, &ast.ExprStmt{
|
||||||
Expr: &ast.CallExpr{
|
Expr: &ast.CallExpr{
|
||||||
Func: &ast.Ident{
|
Func: &ast.Ident{
|
||||||
Name: "print",
|
Name: "__repl_println__",
|
||||||
},
|
},
|
||||||
Args: s.LHS,
|
Args: s.LHS,
|
||||||
},
|
},
|
||||||
|
|
|
@ -210,7 +210,7 @@ func runVM(bytecode *compiler.Bytecode) (time.Duration, objects.Object, error) {
|
||||||
|
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
|
||||||
v := runtime.NewVM(bytecode, globals, nil, nil, -1)
|
v := runtime.NewVM(bytecode, globals, -1)
|
||||||
if err := v.Run(); err != nil {
|
if err := v.Run(); err != nil {
|
||||||
return time.Since(start), nil, err
|
return time.Since(start), nil, err
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,7 +4,6 @@ import (
|
||||||
"flag"
|
"flag"
|
||||||
|
|
||||||
"github.com/d5/tengo/cli"
|
"github.com/d5/tengo/cli"
|
||||||
"github.com/d5/tengo/objects"
|
|
||||||
"github.com/d5/tengo/stdlib"
|
"github.com/d5/tengo/stdlib"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -23,17 +22,12 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
builtinModules := make(map[string]objects.Object, len(stdlib.Modules))
|
|
||||||
for k, mod := range stdlib.Modules {
|
|
||||||
builtinModules[k] = mod
|
|
||||||
}
|
|
||||||
|
|
||||||
cli.Run(&cli.Options{
|
cli.Run(&cli.Options{
|
||||||
ShowHelp: showHelp,
|
ShowHelp: showHelp,
|
||||||
ShowVersion: showVersion,
|
ShowVersion: showVersion,
|
||||||
Version: version,
|
Version: version,
|
||||||
CompileOutput: compileOutput,
|
CompileOutput: compileOutput,
|
||||||
BuiltinModules: builtinModules,
|
Modules: stdlib.GetModules(stdlib.AllModuleNames()...),
|
||||||
InputFile: flag.Arg(0),
|
InputFile: flag.Arg(0),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,6 +16,7 @@ func (b *Bytecode) RemoveDuplicates() {
|
||||||
strings := make(map[string]int)
|
strings := make(map[string]int)
|
||||||
floats := make(map[float64]int)
|
floats := make(map[float64]int)
|
||||||
chars := make(map[rune]int)
|
chars := make(map[rune]int)
|
||||||
|
immutableMaps := make(map[string]int) // for modules
|
||||||
|
|
||||||
for curIdx, c := range b.Constants {
|
for curIdx, c := range b.Constants {
|
||||||
switch c := c.(type) {
|
switch c := c.(type) {
|
||||||
|
@ -23,7 +24,17 @@ func (b *Bytecode) RemoveDuplicates() {
|
||||||
// add to deduped list
|
// add to deduped list
|
||||||
indexMap[curIdx] = len(deduped)
|
indexMap[curIdx] = len(deduped)
|
||||||
deduped = append(deduped, c)
|
deduped = append(deduped, c)
|
||||||
continue
|
case *objects.ImmutableMap:
|
||||||
|
modName := c.Value["__module_name__"].(*objects.String).Value
|
||||||
|
newIdx, ok := immutableMaps[modName]
|
||||||
|
if modName != "" && ok {
|
||||||
|
indexMap[curIdx] = newIdx
|
||||||
|
} else {
|
||||||
|
newIdx = len(deduped)
|
||||||
|
immutableMaps[modName] = newIdx
|
||||||
|
indexMap[curIdx] = newIdx
|
||||||
|
deduped = append(deduped, c)
|
||||||
|
}
|
||||||
case *objects.Int:
|
case *objects.Int:
|
||||||
if newIdx, ok := ints[c.Value]; ok {
|
if newIdx, ok := ints[c.Value]; ok {
|
||||||
indexMap[curIdx] = newIdx
|
indexMap[curIdx] = newIdx
|
||||||
|
|
|
@ -3,7 +3,10 @@ package compiler
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
|
"io/ioutil"
|
||||||
|
"path/filepath"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/d5/tengo"
|
"github.com/d5/tengo"
|
||||||
"github.com/d5/tengo/compiler/ast"
|
"github.com/d5/tengo/compiler/ast"
|
||||||
|
@ -16,14 +19,14 @@ import (
|
||||||
type Compiler struct {
|
type Compiler struct {
|
||||||
file *source.File
|
file *source.File
|
||||||
parent *Compiler
|
parent *Compiler
|
||||||
moduleName string
|
modulePath string
|
||||||
constants []objects.Object
|
constants []objects.Object
|
||||||
symbolTable *SymbolTable
|
symbolTable *SymbolTable
|
||||||
scopes []CompilationScope
|
scopes []CompilationScope
|
||||||
scopeIndex int
|
scopeIndex int
|
||||||
moduleLoader ModuleLoader
|
importModules map[string]objects.Importable
|
||||||
builtinModules map[string]bool
|
|
||||||
compiledModules map[string]*objects.CompiledFunction
|
compiledModules map[string]*objects.CompiledFunction
|
||||||
|
allowFileImport bool
|
||||||
loops []*Loop
|
loops []*Loop
|
||||||
loopIndex int
|
loopIndex int
|
||||||
trace io.Writer
|
trace io.Writer
|
||||||
|
@ -31,12 +34,7 @@ type Compiler struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewCompiler creates a Compiler.
|
// NewCompiler creates a Compiler.
|
||||||
// User can optionally provide the symbol table if one wants to add or remove
|
func NewCompiler(file *source.File, symbolTable *SymbolTable, constants []objects.Object, importModules map[string]objects.Importable, trace io.Writer) *Compiler {
|
||||||
// some global- or builtin- scope symbols. If not (nil), Compile will create
|
|
||||||
// a new symbol table and use the default builtin functions. Likewise, standard
|
|
||||||
// modules can be explicitly provided if user wants to add or remove some modules.
|
|
||||||
// By default, Compile will use all the standard modules otherwise.
|
|
||||||
func NewCompiler(file *source.File, symbolTable *SymbolTable, constants []objects.Object, builtinModules map[string]bool, trace io.Writer) *Compiler {
|
|
||||||
mainScope := CompilationScope{
|
mainScope := CompilationScope{
|
||||||
symbolInit: make(map[string]bool),
|
symbolInit: make(map[string]bool),
|
||||||
sourceMap: make(map[int]source.Pos),
|
sourceMap: make(map[int]source.Pos),
|
||||||
|
@ -45,15 +43,16 @@ func NewCompiler(file *source.File, symbolTable *SymbolTable, constants []object
|
||||||
// symbol table
|
// symbol table
|
||||||
if symbolTable == nil {
|
if symbolTable == nil {
|
||||||
symbolTable = NewSymbolTable()
|
symbolTable = NewSymbolTable()
|
||||||
|
}
|
||||||
|
|
||||||
|
// add builtin functions to the symbol table
|
||||||
for idx, fn := range objects.Builtins {
|
for idx, fn := range objects.Builtins {
|
||||||
symbolTable.DefineBuiltin(idx, fn.Name)
|
symbolTable.DefineBuiltin(idx, fn.Name)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// builtin modules
|
// builtin modules
|
||||||
if builtinModules == nil {
|
if importModules == nil {
|
||||||
builtinModules = make(map[string]bool)
|
importModules = make(map[string]objects.Importable)
|
||||||
}
|
}
|
||||||
|
|
||||||
return &Compiler{
|
return &Compiler{
|
||||||
|
@ -64,7 +63,7 @@ func NewCompiler(file *source.File, symbolTable *SymbolTable, constants []object
|
||||||
scopeIndex: 0,
|
scopeIndex: 0,
|
||||||
loopIndex: -1,
|
loopIndex: -1,
|
||||||
trace: trace,
|
trace: trace,
|
||||||
builtinModules: builtinModules,
|
importModules: importModules,
|
||||||
compiledModules: make(map[string]*objects.CompiledFunction),
|
compiledModules: make(map[string]*objects.CompiledFunction),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -510,21 +509,53 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
c.emit(node, OpCall, len(node.Args))
|
c.emit(node, OpCall, len(node.Args))
|
||||||
|
|
||||||
case *ast.ImportExpr:
|
case *ast.ImportExpr:
|
||||||
if c.builtinModules[node.ModuleName] {
|
if mod, ok := c.importModules[node.ModuleName]; ok {
|
||||||
if len(node.ModuleName) > tengo.MaxStringLen {
|
v, err := mod.Import(node.ModuleName)
|
||||||
return c.error(node, objects.ErrStringLimit)
|
|
||||||
}
|
|
||||||
|
|
||||||
c.emit(node, OpConstant, c.addConstant(&objects.String{Value: node.ModuleName}))
|
|
||||||
c.emit(node, OpGetBuiltinModule)
|
|
||||||
} else {
|
|
||||||
userMod, err := c.compileModule(node)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(node, OpConstant, c.addConstant(userMod))
|
switch v := v.(type) {
|
||||||
|
case []byte: // module written in Tengo
|
||||||
|
compiled, err := c.compileModule(node, node.ModuleName, node.ModuleName, v)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
c.emit(node, OpConstant, c.addConstant(compiled))
|
||||||
c.emit(node, OpCall, 0)
|
c.emit(node, OpCall, 0)
|
||||||
|
case objects.Object: // builtin module
|
||||||
|
c.emit(node, OpConstant, c.addConstant(v))
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("invalid import value type: %T", v))
|
||||||
|
}
|
||||||
|
} else if c.allowFileImport {
|
||||||
|
moduleName := node.ModuleName
|
||||||
|
if !strings.HasSuffix(moduleName, ".tengo") {
|
||||||
|
moduleName += ".tengo"
|
||||||
|
}
|
||||||
|
|
||||||
|
modulePath, err := filepath.Abs(moduleName)
|
||||||
|
if err != nil {
|
||||||
|
return c.errorf(node, "module file path error: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.checkCyclicImports(node, modulePath); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
moduleSrc, err := ioutil.ReadFile(moduleName)
|
||||||
|
if err != nil {
|
||||||
|
return c.errorf(node, "module file read error: %s", err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
compiled, err := c.compileModule(node, moduleName, modulePath, moduleSrc)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
c.emit(node, OpConstant, c.addConstant(compiled))
|
||||||
|
c.emit(node, OpCall, 0)
|
||||||
|
} else {
|
||||||
|
return c.errorf(node, "module '%s' not found", node.ModuleName)
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.ExportStmt:
|
case *ast.ExportStmt:
|
||||||
|
@ -602,18 +633,16 @@ func (c *Compiler) Bytecode() *Bytecode {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetModuleLoader sets or replaces the current module loader.
|
// EnableFileImport enables or disables module loading from local files.
|
||||||
// Note that the module loader is used for user modules,
|
// Local file modules are disabled by default.
|
||||||
// not for the standard modules.
|
func (c *Compiler) EnableFileImport(enable bool) {
|
||||||
func (c *Compiler) SetModuleLoader(moduleLoader ModuleLoader) {
|
c.allowFileImport = enable
|
||||||
c.moduleLoader = moduleLoader
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Compiler) fork(file *source.File, moduleName string, symbolTable *SymbolTable) *Compiler {
|
func (c *Compiler) fork(file *source.File, modulePath string, symbolTable *SymbolTable) *Compiler {
|
||||||
child := NewCompiler(file, symbolTable, nil, c.builtinModules, c.trace)
|
child := NewCompiler(file, symbolTable, nil, c.importModules, c.trace)
|
||||||
child.moduleName = moduleName // name of the module to compile
|
child.modulePath = modulePath // module file path
|
||||||
child.parent = c // parent to set to current compiler
|
child.parent = c // parent to set to current compiler
|
||||||
child.moduleLoader = c.moduleLoader // share module loader
|
|
||||||
|
|
||||||
return child
|
return child
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@ package compiler_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestCompilerErrorReport(t *testing.T) {
|
func TestCompilerErrorReport(t *testing.T) {
|
||||||
expectError(t, `import("user1")`, "Compile Error: module file read error: open user1.tengo: no such file or directory\n\tat test:1:1")
|
expectError(t, `import("user1")`, "Compile Error: module 'user1' not found\n\tat test:1:1")
|
||||||
|
|
||||||
expectError(t, `a = 1`, "Compile Error: unresolved reference 'a'\n\tat test:1:1")
|
expectError(t, `a = 1`, "Compile Error: unresolved reference 'a'\n\tat test:1:1")
|
||||||
expectError(t, `a, b := 1, 2`, "Compile Error: tuple assignment not allowed\n\tat test:1:1")
|
expectError(t, `a, b := 1, 2`, "Compile Error: tuple assignment not allowed\n\tat test:1:1")
|
||||||
|
|
|
@ -1,72 +1,31 @@
|
||||||
package compiler
|
package compiler
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"io/ioutil"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/d5/tengo/compiler/ast"
|
"github.com/d5/tengo/compiler/ast"
|
||||||
"github.com/d5/tengo/compiler/parser"
|
"github.com/d5/tengo/compiler/parser"
|
||||||
"github.com/d5/tengo/objects"
|
"github.com/d5/tengo/objects"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (c *Compiler) compileModule(expr *ast.ImportExpr) (*objects.CompiledFunction, error) {
|
func (c *Compiler) checkCyclicImports(node ast.Node, modulePath string) error {
|
||||||
compiledModule, exists := c.loadCompiledModule(expr.ModuleName)
|
if c.modulePath == modulePath {
|
||||||
if exists {
|
return c.errorf(node, "cyclic module import: %s", modulePath)
|
||||||
return compiledModule, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
moduleName := expr.ModuleName
|
|
||||||
|
|
||||||
// read module source from loader
|
|
||||||
var moduleSrc []byte
|
|
||||||
if c.moduleLoader == nil {
|
|
||||||
// default loader: read from local file
|
|
||||||
if !strings.HasSuffix(moduleName, ".tengo") {
|
|
||||||
moduleName += ".tengo"
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := c.checkCyclicImports(expr, moduleName); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
|
||||||
moduleSrc, err = ioutil.ReadFile(moduleName)
|
|
||||||
if err != nil {
|
|
||||||
return nil, c.errorf(expr, "module file read error: %s", err.Error())
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if err := c.checkCyclicImports(expr, moduleName); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
|
||||||
moduleSrc, err = c.moduleLoader(moduleName)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compiledModule, err := c.doCompileModule(moduleName, moduleSrc)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
c.storeCompiledModule(moduleName, compiledModule)
|
|
||||||
|
|
||||||
return compiledModule, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (c *Compiler) checkCyclicImports(node ast.Node, moduleName string) error {
|
|
||||||
if c.moduleName == moduleName {
|
|
||||||
return c.errorf(node, "cyclic module import: %s", moduleName)
|
|
||||||
} else if c.parent != nil {
|
} else if c.parent != nil {
|
||||||
return c.parent.checkCyclicImports(node, moduleName)
|
return c.parent.checkCyclicImports(node, modulePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Compiler) doCompileModule(moduleName string, src []byte) (*objects.CompiledFunction, error) {
|
func (c *Compiler) compileModule(node ast.Node, moduleName, modulePath string, src []byte) (*objects.CompiledFunction, error) {
|
||||||
|
if err := c.checkCyclicImports(node, modulePath); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
compiledModule, exists := c.loadCompiledModule(modulePath)
|
||||||
|
if exists {
|
||||||
|
return compiledModule, nil
|
||||||
|
}
|
||||||
|
|
||||||
modFile := c.file.Set().AddFile(moduleName, -1, len(src))
|
modFile := c.file.Set().AddFile(moduleName, -1, len(src))
|
||||||
p := parser.NewParser(modFile, src, nil)
|
p := parser.NewParser(modFile, src, nil)
|
||||||
file, err := p.ParseFile()
|
file, err := p.ParseFile()
|
||||||
|
@ -85,7 +44,7 @@ func (c *Compiler) doCompileModule(moduleName string, src []byte) (*objects.Comp
|
||||||
symbolTable = symbolTable.Fork(false)
|
symbolTable = symbolTable.Fork(false)
|
||||||
|
|
||||||
// compile module
|
// compile module
|
||||||
moduleCompiler := c.fork(modFile, moduleName, symbolTable)
|
moduleCompiler := c.fork(modFile, modulePath, symbolTable)
|
||||||
if err := moduleCompiler.Compile(file); err != nil {
|
if err := moduleCompiler.Compile(file); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -98,23 +57,25 @@ func (c *Compiler) doCompileModule(moduleName string, src []byte) (*objects.Comp
|
||||||
compiledFunc := moduleCompiler.Bytecode().MainFunction
|
compiledFunc := moduleCompiler.Bytecode().MainFunction
|
||||||
compiledFunc.NumLocals = symbolTable.MaxSymbols()
|
compiledFunc.NumLocals = symbolTable.MaxSymbols()
|
||||||
|
|
||||||
|
c.storeCompiledModule(modulePath, compiledFunc)
|
||||||
|
|
||||||
return compiledFunc, nil
|
return compiledFunc, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Compiler) loadCompiledModule(moduleName string) (mod *objects.CompiledFunction, ok bool) {
|
func (c *Compiler) loadCompiledModule(modulePath string) (mod *objects.CompiledFunction, ok bool) {
|
||||||
if c.parent != nil {
|
if c.parent != nil {
|
||||||
return c.parent.loadCompiledModule(moduleName)
|
return c.parent.loadCompiledModule(modulePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
mod, ok = c.compiledModules[moduleName]
|
mod, ok = c.compiledModules[modulePath]
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *Compiler) storeCompiledModule(moduleName string, module *objects.CompiledFunction) {
|
func (c *Compiler) storeCompiledModule(modulePath string, module *objects.CompiledFunction) {
|
||||||
if c.parent != nil {
|
if c.parent != nil {
|
||||||
c.parent.storeCompiledModule(moduleName, module)
|
c.parent.storeCompiledModule(modulePath, module)
|
||||||
}
|
}
|
||||||
|
|
||||||
c.compiledModules[moduleName] = module
|
c.compiledModules[modulePath] = module
|
||||||
}
|
}
|
||||||
|
|
|
@ -695,7 +695,7 @@ func TestCompiler_Compile(t *testing.T) {
|
||||||
expect(t, `len([]);`,
|
expect(t, `len([]);`,
|
||||||
bytecode(
|
bytecode(
|
||||||
concat(
|
concat(
|
||||||
compiler.MakeInstruction(compiler.OpGetBuiltin, 4),
|
compiler.MakeInstruction(compiler.OpGetBuiltin, 0),
|
||||||
compiler.MakeInstruction(compiler.OpArray, 0),
|
compiler.MakeInstruction(compiler.OpArray, 0),
|
||||||
compiler.MakeInstruction(compiler.OpCall, 1),
|
compiler.MakeInstruction(compiler.OpCall, 1),
|
||||||
compiler.MakeInstruction(compiler.OpPop)),
|
compiler.MakeInstruction(compiler.OpPop)),
|
||||||
|
@ -708,7 +708,7 @@ func TestCompiler_Compile(t *testing.T) {
|
||||||
compiler.MakeInstruction(compiler.OpPop)),
|
compiler.MakeInstruction(compiler.OpPop)),
|
||||||
objectsArray(
|
objectsArray(
|
||||||
compiledFunction(0, 0,
|
compiledFunction(0, 0,
|
||||||
compiler.MakeInstruction(compiler.OpGetBuiltin, 4),
|
compiler.MakeInstruction(compiler.OpGetBuiltin, 0),
|
||||||
compiler.MakeInstruction(compiler.OpArray, 0),
|
compiler.MakeInstruction(compiler.OpArray, 0),
|
||||||
compiler.MakeInstruction(compiler.OpCall, 1),
|
compiler.MakeInstruction(compiler.OpCall, 1),
|
||||||
compiler.MakeInstruction(compiler.OpReturnValue)))))
|
compiler.MakeInstruction(compiler.OpReturnValue)))))
|
||||||
|
@ -874,7 +874,7 @@ func() {
|
||||||
intObject(1),
|
intObject(1),
|
||||||
intObject(1))))
|
intObject(1))))
|
||||||
|
|
||||||
expectError(t, `import("user1")`, "no such file or directory") // unknown module name
|
expectError(t, `import("user1")`, "module 'user1' not found") // unknown module name
|
||||||
|
|
||||||
expectError(t, `
|
expectError(t, `
|
||||||
r["x"] = {
|
r["x"] = {
|
||||||
|
|
|
@ -54,7 +54,6 @@ const (
|
||||||
OpGetLocalPtr // Get local variable as a pointer
|
OpGetLocalPtr // Get local variable as a pointer
|
||||||
OpSetSelFree // Set free variables using selectors
|
OpSetSelFree // Set free variables using selectors
|
||||||
OpGetBuiltin // Get builtin function
|
OpGetBuiltin // Get builtin function
|
||||||
OpGetBuiltinModule // Get builtin module
|
|
||||||
OpClosure // Push closure
|
OpClosure // Push closure
|
||||||
OpIteratorInit // Iterator init
|
OpIteratorInit // Iterator init
|
||||||
OpIteratorNext // Iterator next
|
OpIteratorNext // Iterator next
|
||||||
|
@ -108,7 +107,6 @@ var OpcodeNames = [...]string{
|
||||||
OpDefineLocal: "DEFL",
|
OpDefineLocal: "DEFL",
|
||||||
OpSetSelLocal: "SETSL",
|
OpSetSelLocal: "SETSL",
|
||||||
OpGetBuiltin: "BUILTIN",
|
OpGetBuiltin: "BUILTIN",
|
||||||
OpGetBuiltinModule: "BLTMOD",
|
|
||||||
OpClosure: "CLOSURE",
|
OpClosure: "CLOSURE",
|
||||||
OpGetFreePtr: "GETFP",
|
OpGetFreePtr: "GETFP",
|
||||||
OpGetFree: "GETF",
|
OpGetFree: "GETF",
|
||||||
|
@ -167,7 +165,6 @@ var OpcodeOperands = [...][]int{
|
||||||
OpDefineLocal: {1},
|
OpDefineLocal: {1},
|
||||||
OpSetSelLocal: {1, 1},
|
OpSetSelLocal: {1, 1},
|
||||||
OpGetBuiltin: {1},
|
OpGetBuiltin: {1},
|
||||||
OpGetBuiltinModule: {},
|
|
||||||
OpClosure: {2, 1},
|
OpClosure: {2, 1},
|
||||||
OpGetFreePtr: {1},
|
OpGetFreePtr: {1},
|
||||||
OpGetFree: {1},
|
OpGetFree: {1},
|
||||||
|
|
|
@ -65,6 +65,13 @@ func TestMap(t *testing.T) {
|
||||||
mapElementLit("key3", p(5, 2), p(5, 6), boolLit(true, p(5, 8))))))
|
mapElementLit("key3", p(5, 2), p(5, 6), boolLit(true, p(5, 8))))))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
expectError(t, `
|
||||||
|
{
|
||||||
|
key1: 1,
|
||||||
|
key2: "2",
|
||||||
|
key3: true,
|
||||||
|
}`) // unlike Go, trailing comma for the last element is illegal
|
||||||
|
|
||||||
expectError(t, `{ key1: 1, }`)
|
expectError(t, `{ key1: 1, }`)
|
||||||
expectError(t, `{
|
expectError(t, `{
|
||||||
key1: 1,
|
key1: 1,
|
||||||
|
|
|
@ -1,45 +1,5 @@
|
||||||
# Builtin Functions
|
# Builtin Functions
|
||||||
|
|
||||||
## print
|
|
||||||
|
|
||||||
Prints a string representation of the given variable to the standard output. No spaces are added between the operands.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
v := [1, 2, 3]
|
|
||||||
print(v) // "[1, 2, 3]"
|
|
||||||
|
|
||||||
print(1, 2, 3) // "123"
|
|
||||||
```
|
|
||||||
|
|
||||||
## println
|
|
||||||
|
|
||||||
Prints a string representation of the given variable to the standard output with a newline appended. No spaces are added between the operands.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
v := [1, 2, 3]
|
|
||||||
println(v) // "[1, 2, 3]"
|
|
||||||
|
|
||||||
println(1, 2, 3) // "123" newline
|
|
||||||
```
|
|
||||||
|
|
||||||
## printf
|
|
||||||
|
|
||||||
Prints a formatted string to the standard output. It does not append the newline character at the end. The first argument must a String object. It's same as Go's `fmt.Printf`.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
a := [1, 2, 3]
|
|
||||||
printf("foo %v", a) // "foo [1, 2, 3]"
|
|
||||||
```
|
|
||||||
|
|
||||||
## sprintf
|
|
||||||
|
|
||||||
Returns a formatted string. The first argument must be a String object. It's the same as Go's `fmt.Sprintf`.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
a := [1, 2, 3]
|
|
||||||
b := sprintp("foo %v", a) // b == "foo [1, 2, 3]"
|
|
||||||
```
|
|
||||||
|
|
||||||
## len
|
## len
|
||||||
|
|
||||||
Returns the number of elements if the given variable is array, string, map, or module map.
|
Returns the number of elements if the given variable is array, string, map, or module map.
|
||||||
|
@ -71,26 +31,6 @@ v := [1]
|
||||||
v = append(v, 2, 3) // v == [1, 2, 3]
|
v = append(v, 2, 3) // v == [1, 2, 3]
|
||||||
```
|
```
|
||||||
|
|
||||||
## to_json
|
|
||||||
|
|
||||||
Returns the JSON encoding of an object.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
print(to_json([1, 2, 3])) // [1, 2, 3]
|
|
||||||
print(to_json(4)) // 4
|
|
||||||
print(to_json("five")) // "five"
|
|
||||||
```
|
|
||||||
|
|
||||||
## from_json
|
|
||||||
|
|
||||||
Parses the JSON-encoded data and returns an object.
|
|
||||||
|
|
||||||
```golang
|
|
||||||
arr := from_json(`[1, 2, 3]`)
|
|
||||||
four := from_json(`4`)
|
|
||||||
five := from_json(`"five"`)
|
|
||||||
```
|
|
||||||
|
|
||||||
## string
|
## string
|
||||||
|
|
||||||
Tries to convert an object to string object. See [Runtime Types](https://github.com/d5/tengo/blob/master/docs/runtime-types.md) for more details on type conversion.
|
Tries to convert an object to string object. See [Runtime Types](https://github.com/d5/tengo/blob/master/docs/runtime-types.md) for more details on type conversion.
|
||||||
|
|
|
@ -119,68 +119,40 @@ Users can add and use a custom user type in Tengo code by implementing [Object](
|
||||||
|
|
||||||
To securely compile and execute _potentially_ unsafe script code, you can use the following Script functions.
|
To securely compile and execute _potentially_ unsafe script code, you can use the following Script functions.
|
||||||
|
|
||||||
#### Script.SetBuiltinFunctions(funcs []*objects.BuiltinFunction)
|
#### Script.SetImports(modules map[string]objects.Importable)
|
||||||
|
|
||||||
SetBuiltinFunctions resets all builtin functions in the compiler to the ones provided in the input parameter. Compiler will report a compile-time error if the a function not set is referenced. Passing `nil` will disable all builtin functions. All builtin functions **are included by default** unless `SetBuiltinFunctions` is called.
|
SetImports sets the import modules with corresponding names. Script **does not** include any modules by default. You can use this function to include the [Standard Library](https://github.com/d5/tengo/blob/master/docs/stdlib.md).
|
||||||
|
|
||||||
```golang
|
|
||||||
s := script.New([]byte(`print([1, 2, 3])`))
|
|
||||||
|
|
||||||
_, err := s.Run() // prints [1, 2, 3]
|
|
||||||
|
|
||||||
s.SetBuiltinFunctions(nil)
|
|
||||||
|
|
||||||
_, err := s.Run() // compile error
|
|
||||||
|
|
||||||
s.SetBuiltinFunctions([]*objects.BuiltinFunction{&objects.Builtins[0]})
|
|
||||||
|
|
||||||
_, err := s.Run() // prints [1, 2, 3]
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Script.SetBuiltinModules(modules map[string]*objects.ImmutableMap)
|
|
||||||
|
|
||||||
SetBuiltinModules adds builtin modules provided in the input parameter. This can be used to add [standard library](https://github.com/d5/tengo/blob/master/docs/stdlib.md) modules into the compiler and VM. Compiler will report a compile-time error if the code tries to import a module that hasn't been included. Passing `nil` will disable all builtin modules. No standard library modules are included by default unless `SetBuiltinModules` is called.
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
s := script.New([]byte(`math := import("math"); a := math.abs(-19.84)`))
|
s := script.New([]byte(`math := import("math"); a := math.abs(-19.84)`))
|
||||||
|
|
||||||
_, err := s.Run() // compile error
|
s.SetImports(map[string]objects.Importable{
|
||||||
|
"math": stdlib.BuiltinModules["math"],
|
||||||
s.SetBuiltinModules(stdlib.Modules)
|
})
|
||||||
|
// or
|
||||||
_, err := s.Run() // a = 19.84
|
s.SetImports(stdlib.GetModules("math"))
|
||||||
|
// or, to include all stdlib at once
|
||||||
s.SetBuiltinModules(nil)
|
s.SetImports(stdlib.GetModules(stdlib.AllModuleNames()...))
|
||||||
|
|
||||||
_, err := s.Run() // compile error
|
|
||||||
|
|
||||||
s.SetBuiltinModules(map[string]*objects.ImmutableMap{"math": stdlib.Modules["math"]})
|
|
||||||
|
|
||||||
_, err := s.Run() // a = 19.84
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Script.SetUserModuleLoader(loader compiler.ModuleLoader)
|
You can also include Tengo's written module using `objects.SourceModule` (which implements `objects.Importable`).
|
||||||
|
|
||||||
SetUserModuleLoader replaces the default user-module loader of the compiler, which tries to read the source from a local file.
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
s := script.New([]byte(`math := import("mod1"); a := math.foo()`))
|
s := script.New([]byte(`double := import("double"); a := double(20)`))
|
||||||
|
|
||||||
s.SetUserModuleLoader(func(moduleName string) ([]byte, error) {
|
s.SetImports(map[string]objects.Importable{
|
||||||
if moduleName == "mod1" {
|
"double": &objects.SourceModule{Src: []byte(`export func(x) { return x * 2 }`)},
|
||||||
return []byte(`foo := func() { return 5 }`), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, errors.New("module not found")
|
|
||||||
})
|
})
|
||||||
```
|
```
|
||||||
|
|
||||||
Note that when a script is being added to another script as a module (via `Script.AddModule`), it does not inherit the module loader from the main script.
|
|
||||||
|
|
||||||
#### Script.SetMaxAllocs(n int64)
|
#### Script.SetMaxAllocs(n int64)
|
||||||
|
|
||||||
SetMaxAllocs sets the maximum number of object allocations. Note this is a cumulative metric that tracks only the object creations. Set this to a negative number (e.g. `-1`) if you don't need to limit the number of allocations.
|
SetMaxAllocs sets the maximum number of object allocations. Note this is a cumulative metric that tracks only the object creations. Set this to a negative number (e.g. `-1`) if you don't need to limit the number of allocations.
|
||||||
|
|
||||||
|
#### Script.EnableFileImport(enable bool)
|
||||||
|
|
||||||
|
EnableFileImport enables or disables module loading from the local files. It's disabled by default.
|
||||||
|
|
||||||
#### tengo.MaxStringLen
|
#### tengo.MaxStringLen
|
||||||
|
|
||||||
|
|
12
docs/stdlib-fmt.md
Normal file
12
docs/stdlib-fmt.md
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
# Module - "fmt"
|
||||||
|
|
||||||
|
```golang
|
||||||
|
fmt := import("fmt")
|
||||||
|
```
|
||||||
|
|
||||||
|
## Functions
|
||||||
|
|
||||||
|
- `print(args...)`: Prints a string representation of the given variable to the standard output. Unlike Go's `fmt.Print` function, no spaces are added between the operands.
|
||||||
|
- `println(args...)`: Prints a string representation of the given variable to the standard output with a newline appended. Unlike Go's `fmt.Println` function, no spaces are added between the operands.
|
||||||
|
- `printf(format, args...)`: Prints a formatted string to the standard output. It does not append the newline character at the end. The first argument must a String object. It's same as Go's `fmt.Printf`.
|
||||||
|
- `sprintf(format, args...)`: Returns a formatted string. The first argument must be a String object. It's the same as Go's `fmt.Sprintf`.
|
10
docs/stdlib-json.md
Normal file
10
docs/stdlib-json.md
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
# Module - "json"
|
||||||
|
|
||||||
|
```golang
|
||||||
|
json := import("json")
|
||||||
|
```
|
||||||
|
|
||||||
|
## Functions
|
||||||
|
|
||||||
|
- `parse(v)`: Parses the JSON string and returns an object.
|
||||||
|
- `stringify(v)`: Returns the JSON string representation of the object.
|
|
@ -5,3 +5,5 @@
|
||||||
- [math](https://github.com/d5/tengo/blob/master/docs/stdlib-math.md): mathematical constants and functions
|
- [math](https://github.com/d5/tengo/blob/master/docs/stdlib-math.md): mathematical constants and functions
|
||||||
- [times](https://github.com/d5/tengo/blob/master/docs/stdlib-times.md): time-related functions
|
- [times](https://github.com/d5/tengo/blob/master/docs/stdlib-times.md): time-related functions
|
||||||
- [rand](https://github.com/d5/tengo/blob/master/docs/stdlib-rand.md): random functions
|
- [rand](https://github.com/d5/tengo/blob/master/docs/stdlib-rand.md): random functions
|
||||||
|
- [fmt](https://github.com/d5/tengo/blob/master/docs/stdlib-fmt.md): formatting functions
|
||||||
|
- [json](https://github.com/d5/tengo/blob/master/docs/stdlib-json.md): JSON functions
|
|
@ -226,7 +226,7 @@ Main script:
|
||||||
```golang
|
```golang
|
||||||
sum := import("./sum") // assuming sum.tengo file exists in the current directory
|
sum := import("./sum") // assuming sum.tengo file exists in the current directory
|
||||||
// same as 'import("./sum.tengo")' or 'import("sum")'
|
// same as 'import("./sum.tengo")' or 'import("sum")'
|
||||||
print(sum(10)) // module function
|
fmt.print(sum(10)) // module function
|
||||||
```
|
```
|
||||||
|
|
||||||
`sum.tengo` file:
|
`sum.tengo` file:
|
||||||
|
|
|
@ -1,65 +0,0 @@
|
||||||
package objects
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
|
|
||||||
"github.com/d5/tengo"
|
|
||||||
)
|
|
||||||
|
|
||||||
// to_json(v object) => bytes
|
|
||||||
func builtinToJSON(args ...Object) (Object, error) {
|
|
||||||
if len(args) != 1 {
|
|
||||||
return nil, ErrWrongNumArguments
|
|
||||||
}
|
|
||||||
|
|
||||||
v := ToInterface(args[0])
|
|
||||||
if vErr, isErr := v.(error); isErr {
|
|
||||||
v = vErr.Error()
|
|
||||||
}
|
|
||||||
|
|
||||||
res, err := json.Marshal(v)
|
|
||||||
if err != nil {
|
|
||||||
return &Error{Value: &String{Value: err.Error()}}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(res) > tengo.MaxBytesLen {
|
|
||||||
return nil, ErrBytesLimit
|
|
||||||
}
|
|
||||||
|
|
||||||
return &Bytes{Value: res}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// from_json(data string/bytes) => object
|
|
||||||
func builtinFromJSON(args ...Object) (Object, error) {
|
|
||||||
if len(args) != 1 {
|
|
||||||
return nil, ErrWrongNumArguments
|
|
||||||
}
|
|
||||||
|
|
||||||
var target interface{}
|
|
||||||
|
|
||||||
switch o := args[0].(type) {
|
|
||||||
case *Bytes:
|
|
||||||
err := json.Unmarshal(o.Value, &target)
|
|
||||||
if err != nil {
|
|
||||||
return &Error{Value: &String{Value: err.Error()}}, nil
|
|
||||||
}
|
|
||||||
case *String:
|
|
||||||
err := json.Unmarshal([]byte(o.Value), &target)
|
|
||||||
if err != nil {
|
|
||||||
return &Error{Value: &String{Value: err.Error()}}, nil
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
return nil, ErrInvalidArgumentType{
|
|
||||||
Name: "first",
|
|
||||||
Expected: "bytes/string",
|
|
||||||
Found: args[0].TypeName(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
res, err := FromInterface(target)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
return res, nil
|
|
||||||
}
|
|
16
objects/builtin_module.go
Normal file
16
objects/builtin_module.go
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
// BuiltinModule is an importable module that's written in Go.
|
||||||
|
type BuiltinModule struct {
|
||||||
|
Attrs map[string]Object
|
||||||
|
}
|
||||||
|
|
||||||
|
// Import returns an immutable map for the module.
|
||||||
|
func (m *BuiltinModule) Import(name string) (interface{}, error) {
|
||||||
|
attrs := make(map[string]Object, len(m.Attrs))
|
||||||
|
for k, v := range m.Attrs {
|
||||||
|
attrs[k] = v.Copy()
|
||||||
|
}
|
||||||
|
attrs["__module_name__"] = &String{Value: name}
|
||||||
|
return &ImmutableMap{Value: attrs}, nil
|
||||||
|
}
|
|
@ -2,23 +2,7 @@ package objects
|
||||||
|
|
||||||
// Builtins contains all default builtin functions.
|
// Builtins contains all default builtin functions.
|
||||||
// Use GetBuiltinFunctions instead of accessing Builtins directly.
|
// Use GetBuiltinFunctions instead of accessing Builtins directly.
|
||||||
var Builtins = []BuiltinFunction{
|
var Builtins = []*BuiltinFunction{
|
||||||
{
|
|
||||||
Name: "print",
|
|
||||||
Value: builtinPrint,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "println",
|
|
||||||
Value: builtinPrintln,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "printf",
|
|
||||||
Value: builtinPrintf,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "sprintf",
|
|
||||||
Value: builtinSprintf,
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
Name: "len",
|
Name: "len",
|
||||||
Value: builtinLen,
|
Value: builtinLen,
|
||||||
|
@ -119,50 +103,8 @@ var Builtins = []BuiltinFunction{
|
||||||
Name: "is_callable",
|
Name: "is_callable",
|
||||||
Value: builtinIsCallable,
|
Value: builtinIsCallable,
|
||||||
},
|
},
|
||||||
{
|
|
||||||
Name: "to_json",
|
|
||||||
Value: builtinToJSON,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
Name: "from_json",
|
|
||||||
Value: builtinFromJSON,
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
Name: "type_name",
|
Name: "type_name",
|
||||||
Value: builtinTypeName,
|
Value: builtinTypeName,
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
// AllBuiltinFunctionNames returns a list of all default builtin function names.
|
|
||||||
func AllBuiltinFunctionNames() []string {
|
|
||||||
var names []string
|
|
||||||
for _, bf := range Builtins {
|
|
||||||
names = append(names, bf.Name)
|
|
||||||
}
|
|
||||||
return names
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetBuiltinFunctions returns a slice of builtin function objects.
|
|
||||||
// GetBuiltinFunctions removes the duplicate names, and, the returned builtin functions
|
|
||||||
// are not guaranteed to be in the same order as names.
|
|
||||||
func GetBuiltinFunctions(names ...string) []*BuiltinFunction {
|
|
||||||
include := make(map[string]bool)
|
|
||||||
for _, name := range names {
|
|
||||||
include[name] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
var builtinFuncs []*BuiltinFunction
|
|
||||||
for _, bf := range Builtins {
|
|
||||||
if include[bf.Name] {
|
|
||||||
bf := bf
|
|
||||||
builtinFuncs = append(builtinFuncs, &bf)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return builtinFuncs
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetAllBuiltinFunctions returns all builtin functions.
|
|
||||||
func GetAllBuiltinFunctions() []*BuiltinFunction {
|
|
||||||
return GetBuiltinFunctions(AllBuiltinFunctionNames()...)
|
|
||||||
}
|
|
||||||
|
|
|
@ -1,65 +0,0 @@
|
||||||
package objects_test
|
|
||||||
|
|
||||||
import (
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/d5/tengo/assert"
|
|
||||||
"github.com/d5/tengo/objects"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestGetBuiltinFunctions(t *testing.T) {
|
|
||||||
testGetBuiltinFunctions(t)
|
|
||||||
testGetBuiltinFunctions(t, "print")
|
|
||||||
testGetBuiltinFunctions(t, "int", "float")
|
|
||||||
testGetBuiltinFunctions(t, "int", "float", "printf")
|
|
||||||
testGetBuiltinFunctions(t, "int", "int") // duplicate names ignored
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGetAllBuiltinFunctions(t *testing.T) {
|
|
||||||
funcs := objects.GetAllBuiltinFunctions()
|
|
||||||
if !assert.Equal(t, len(objects.Builtins), len(funcs)) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
namesM := make(map[string]bool)
|
|
||||||
for _, bf := range objects.Builtins {
|
|
||||||
namesM[bf.Name] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, bf := range funcs {
|
|
||||||
assert.True(t, namesM[bf.Name], "name: %s", bf.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAllBuiltinFunctionNames(t *testing.T) {
|
|
||||||
names := objects.AllBuiltinFunctionNames()
|
|
||||||
if !assert.Equal(t, len(objects.Builtins), len(names)) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
namesM := make(map[string]bool)
|
|
||||||
for _, name := range names {
|
|
||||||
namesM[name] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, bf := range objects.Builtins {
|
|
||||||
assert.True(t, namesM[bf.Name], "name: %s", bf.Name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func testGetBuiltinFunctions(t *testing.T, names ...string) {
|
|
||||||
// remove duplicates
|
|
||||||
namesM := make(map[string]bool)
|
|
||||||
for _, name := range names {
|
|
||||||
namesM[name] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
funcs := objects.GetBuiltinFunctions(names...)
|
|
||||||
if !assert.Equal(t, len(namesM), len(funcs)) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, bf := range funcs {
|
|
||||||
assert.True(t, namesM[bf.Name], "name: %s", bf.Name)
|
|
||||||
}
|
|
||||||
}
|
|
7
objects/importable.go
Normal file
7
objects/importable.go
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
// Importable interface represents importable module instance.
|
||||||
|
type Importable interface {
|
||||||
|
// Import should return either an Object or module source code ([]byte).
|
||||||
|
Import(name string) (interface{}, error)
|
||||||
|
}
|
11
objects/source_module.go
Normal file
11
objects/source_module.go
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
// SourceModule is an importable module that's written in Tengo.
|
||||||
|
type SourceModule struct {
|
||||||
|
Src []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
// Import returns a module source code.
|
||||||
|
func (m *SourceModule) Import(_ string) (interface{}, error) {
|
||||||
|
return m.Src, nil
|
||||||
|
}
|
|
@ -35,8 +35,6 @@ type VM struct {
|
||||||
curIPLimit int
|
curIPLimit int
|
||||||
ip int
|
ip int
|
||||||
aborting int64
|
aborting int64
|
||||||
builtinFuncs []objects.Object
|
|
||||||
builtinModules map[string]objects.Object
|
|
||||||
maxAllocs int64
|
maxAllocs int64
|
||||||
allocs int64
|
allocs int64
|
||||||
err error
|
err error
|
||||||
|
@ -44,25 +42,11 @@ type VM struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// NewVM creates a VM.
|
// NewVM creates a VM.
|
||||||
func NewVM(bytecode *compiler.Bytecode, globals []objects.Object, builtinFuncs []objects.Object, builtinModules map[string]objects.Object, maxAllocs int64) *VM {
|
func NewVM(bytecode *compiler.Bytecode, globals []objects.Object, maxAllocs int64) *VM {
|
||||||
if globals == nil {
|
if globals == nil {
|
||||||
globals = make([]objects.Object, GlobalsSize)
|
globals = make([]objects.Object, GlobalsSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
if builtinModules == nil {
|
|
||||||
builtinModules = make(map[string]objects.Object)
|
|
||||||
}
|
|
||||||
|
|
||||||
if builtinFuncs == nil {
|
|
||||||
builtinFuncs = make([]objects.Object, len(objects.Builtins))
|
|
||||||
for idx, fn := range objects.Builtins {
|
|
||||||
builtinFuncs[idx] = &objects.BuiltinFunction{
|
|
||||||
Name: fn.Name,
|
|
||||||
Value: fn.Value,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
frames := make([]Frame, MaxFrames)
|
frames := make([]Frame, MaxFrames)
|
||||||
frames[0].fn = bytecode.MainFunction
|
frames[0].fn = bytecode.MainFunction
|
||||||
frames[0].ip = -1
|
frames[0].ip = -1
|
||||||
|
@ -79,8 +63,6 @@ func NewVM(bytecode *compiler.Bytecode, globals []objects.Object, builtinFuncs [
|
||||||
curInsts: frames[0].fn.Instructions,
|
curInsts: frames[0].fn.Instructions,
|
||||||
curIPLimit: len(frames[0].fn.Instructions) - 1,
|
curIPLimit: len(frames[0].fn.Instructions) - 1,
|
||||||
ip: -1,
|
ip: -1,
|
||||||
builtinFuncs: builtinFuncs,
|
|
||||||
builtinModules: builtinModules,
|
|
||||||
maxAllocs: maxAllocs,
|
maxAllocs: maxAllocs,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -997,28 +979,7 @@ func (v *VM) run() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
v.stack[v.sp] = v.builtinFuncs[builtinIndex]
|
v.stack[v.sp] = objects.Builtins[builtinIndex]
|
||||||
v.sp++
|
|
||||||
|
|
||||||
case compiler.OpGetBuiltinModule:
|
|
||||||
val := v.stack[v.sp-1]
|
|
||||||
v.sp--
|
|
||||||
|
|
||||||
moduleName := val.(*objects.String).Value
|
|
||||||
|
|
||||||
module, ok := v.builtinModules[moduleName]
|
|
||||||
if !ok {
|
|
||||||
v.errOffset = 3
|
|
||||||
v.err = fmt.Errorf("module '%s' not found", moduleName)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if v.sp >= StackSize {
|
|
||||||
v.err = ErrStackOverflow
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
v.stack[v.sp] = module
|
|
||||||
v.sp++
|
v.sp++
|
||||||
|
|
||||||
case compiler.OpClosure:
|
case compiler.OpClosure:
|
||||||
|
|
|
@ -8,48 +8,48 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestArray(t *testing.T) {
|
func TestArray(t *testing.T) {
|
||||||
expect(t, `out = [1, 2 * 2, 3 + 3]`, ARR{1, 4, 6})
|
expect(t, `out = [1, 2 * 2, 3 + 3]`, nil, ARR{1, 4, 6})
|
||||||
|
|
||||||
// array copy-by-reference
|
// array copy-by-reference
|
||||||
expect(t, `a1 := [1, 2, 3]; a2 := a1; a1[0] = 5; out = a2`, ARR{5, 2, 3})
|
expect(t, `a1 := [1, 2, 3]; a2 := a1; a1[0] = 5; out = a2`, nil, ARR{5, 2, 3})
|
||||||
expect(t, `func () { a1 := [1, 2, 3]; a2 := a1; a1[0] = 5; out = a2 }()`, ARR{5, 2, 3})
|
expect(t, `func () { a1 := [1, 2, 3]; a2 := a1; a1[0] = 5; out = a2 }()`, nil, ARR{5, 2, 3})
|
||||||
|
|
||||||
// array index set
|
// array index set
|
||||||
expectError(t, `a1 := [1, 2, 3]; a1[3] = 5`, "index out of bounds")
|
expectError(t, `a1 := [1, 2, 3]; a1[3] = 5`, nil, "index out of bounds")
|
||||||
|
|
||||||
// index operator
|
// index operator
|
||||||
arr := ARR{1, 2, 3, 4, 5, 6}
|
arr := ARR{1, 2, 3, 4, 5, 6}
|
||||||
arrStr := `[1, 2, 3, 4, 5, 6]`
|
arrStr := `[1, 2, 3, 4, 5, 6]`
|
||||||
arrLen := 6
|
arrLen := 6
|
||||||
for idx := 0; idx < arrLen; idx++ {
|
for idx := 0; idx < arrLen; idx++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d]", arrStr, idx), arr[idx])
|
expect(t, fmt.Sprintf("out = %s[%d]", arrStr, idx), nil, arr[idx])
|
||||||
expect(t, fmt.Sprintf("out = %s[0 + %d]", arrStr, idx), arr[idx])
|
expect(t, fmt.Sprintf("out = %s[0 + %d]", arrStr, idx), nil, arr[idx])
|
||||||
expect(t, fmt.Sprintf("out = %s[1 + %d - 1]", arrStr, idx), arr[idx])
|
expect(t, fmt.Sprintf("out = %s[1 + %d - 1]", arrStr, idx), nil, arr[idx])
|
||||||
expect(t, fmt.Sprintf("idx := %d; out = %s[idx]", idx, arrStr), arr[idx])
|
expect(t, fmt.Sprintf("idx := %d; out = %s[idx]", idx, arrStr), nil, arr[idx])
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(t, fmt.Sprintf("%s[%d]", arrStr, -1), objects.UndefinedValue)
|
expect(t, fmt.Sprintf("%s[%d]", arrStr, -1), nil, objects.UndefinedValue)
|
||||||
expect(t, fmt.Sprintf("%s[%d]", arrStr, arrLen), objects.UndefinedValue)
|
expect(t, fmt.Sprintf("%s[%d]", arrStr, arrLen), nil, objects.UndefinedValue)
|
||||||
|
|
||||||
// slice operator
|
// slice operator
|
||||||
for low := 0; low < arrLen; low++ {
|
for low := 0; low < arrLen; low++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, low, low), ARR{})
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, low, low), nil, ARR{})
|
||||||
for high := low; high <= arrLen; high++ {
|
for high := low; high <= arrLen; high++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, low, high), arr[low:high])
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, low, high), nil, arr[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[0 + %d : 0 + %d]", arrStr, low, high), arr[low:high])
|
expect(t, fmt.Sprintf("out = %s[0 + %d : 0 + %d]", arrStr, low, high), nil, arr[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[1 + %d - 1 : 1 + %d - 1]", arrStr, low, high), arr[low:high])
|
expect(t, fmt.Sprintf("out = %s[1 + %d - 1 : 1 + %d - 1]", arrStr, low, high), nil, arr[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[:%d]", arrStr, high), arr[:high])
|
expect(t, fmt.Sprintf("out = %s[:%d]", arrStr, high), nil, arr[:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:]", arrStr, low), arr[low:])
|
expect(t, fmt.Sprintf("out = %s[%d:]", arrStr, low), nil, arr[low:])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(t, fmt.Sprintf("out = %s[:]", arrStr), arr)
|
expect(t, fmt.Sprintf("out = %s[:]", arrStr), nil, arr)
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:]", arrStr, -1), arr)
|
expect(t, fmt.Sprintf("out = %s[%d:]", arrStr, -1), nil, arr)
|
||||||
expect(t, fmt.Sprintf("out = %s[:%d]", arrStr, arrLen+1), arr)
|
expect(t, fmt.Sprintf("out = %s[:%d]", arrStr, arrLen+1), nil, arr)
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, 2, 2), ARR{})
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", arrStr, 2, 2), nil, ARR{})
|
||||||
|
|
||||||
expectError(t, fmt.Sprintf("%s[:%d]", arrStr, -1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[:%d]", arrStr, -1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:]", arrStr, arrLen+1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:]", arrStr, arrLen+1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:%d]", arrStr, 0, -1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:%d]", arrStr, 0, -1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:%d]", arrStr, 2, 1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:%d]", arrStr, 2, 1), nil, "invalid slice index")
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,17 +5,17 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestAssignment(t *testing.T) {
|
func TestAssignment(t *testing.T) {
|
||||||
expect(t, `a := 1; a = 2; out = a`, 2)
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
||||||
expect(t, `a := 1; a = 2; out = a`, 2)
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
||||||
expect(t, `a := 1; a = a + 4; out = a`, 5)
|
expect(t, `a := 1; a = a + 4; out = a`, nil, 5)
|
||||||
expect(t, `a := 1; f1 := func() { a = 2; return a }; out = f1()`, 2)
|
expect(t, `a := 1; f1 := func() { a = 2; return a }; out = f1()`, nil, 2)
|
||||||
expect(t, `a := 1; f1 := func() { a := 3; a = 2; return a }; out = f1()`, 2)
|
expect(t, `a := 1; f1 := func() { a := 3; a = 2; return a }; out = f1()`, nil, 2)
|
||||||
|
|
||||||
expect(t, `a := 1; out = a`, 1)
|
expect(t, `a := 1; out = a`, nil, 1)
|
||||||
expect(t, `a := 1; a = 2; out = a`, 2)
|
expect(t, `a := 1; a = 2; out = a`, nil, 2)
|
||||||
expect(t, `a := 1; func() { a = 2 }(); out = a`, 2)
|
expect(t, `a := 1; func() { a = 2 }(); out = a`, nil, 2)
|
||||||
expect(t, `a := 1; func() { a := 2 }(); out = a`, 1) // "a := 2" defines a new local variable 'a'
|
expect(t, `a := 1; func() { a := 2 }(); out = a`, nil, 1) // "a := 2" defines a new local variable 'a'
|
||||||
expect(t, `a := 1; func() { b := 2; out = b }()`, 2)
|
expect(t, `a := 1; func() { b := 2; out = b }()`, nil, 2)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
a := 2
|
a := 2
|
||||||
|
@ -24,7 +24,7 @@ out = func() {
|
||||||
}()
|
}()
|
||||||
return a
|
return a
|
||||||
}()
|
}()
|
||||||
`, 3)
|
`, nil, 3)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
|
@ -33,25 +33,25 @@ func() {
|
||||||
a := 4
|
a := 4
|
||||||
return a
|
return a
|
||||||
}()
|
}()
|
||||||
}()`, 4)
|
}()`, nil, 4)
|
||||||
|
|
||||||
expectError(t, `a := 1; a := 2`, "redeclared") // redeclared in the same scope
|
expectError(t, `a := 1; a := 2`, nil, "redeclared") // redeclared in the same scope
|
||||||
expectError(t, `func() { a := 1; a := 2 }()`, "redeclared") // redeclared in the same scope
|
expectError(t, `func() { a := 1; a := 2 }()`, nil, "redeclared") // redeclared in the same scope
|
||||||
|
|
||||||
expect(t, `a := 1; a += 2; out = a`, 3)
|
expect(t, `a := 1; a += 2; out = a`, nil, 3)
|
||||||
expect(t, `a := 1; a += 4 - 2;; out = a`, 3)
|
expect(t, `a := 1; a += 4 - 2;; out = a`, nil, 3)
|
||||||
expect(t, `a := 3; a -= 1;; out = a`, 2)
|
expect(t, `a := 3; a -= 1;; out = a`, nil, 2)
|
||||||
expect(t, `a := 3; a -= 5 - 4;; out = a`, 2)
|
expect(t, `a := 3; a -= 5 - 4;; out = a`, nil, 2)
|
||||||
expect(t, `a := 2; a *= 4;; out = a`, 8)
|
expect(t, `a := 2; a *= 4;; out = a`, nil, 8)
|
||||||
expect(t, `a := 2; a *= 1 + 3;; out = a`, 8)
|
expect(t, `a := 2; a *= 1 + 3;; out = a`, nil, 8)
|
||||||
expect(t, `a := 10; a /= 2;; out = a`, 5)
|
expect(t, `a := 10; a /= 2;; out = a`, nil, 5)
|
||||||
expect(t, `a := 10; a /= 5 - 3;; out = a`, 5)
|
expect(t, `a := 10; a /= 5 - 3;; out = a`, nil, 5)
|
||||||
|
|
||||||
// compound assignment operator does not define new variable
|
// compound assignment operator does not define new variable
|
||||||
expectError(t, `a += 4`, "unresolved reference")
|
expectError(t, `a += 4`, nil, "unresolved reference")
|
||||||
expectError(t, `a -= 4`, "unresolved reference")
|
expectError(t, `a -= 4`, nil, "unresolved reference")
|
||||||
expectError(t, `a *= 4`, "unresolved reference")
|
expectError(t, `a *= 4`, nil, "unresolved reference")
|
||||||
expectError(t, `a /= 4`, "unresolved reference")
|
expectError(t, `a /= 4`, nil, "unresolved reference")
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func() {
|
f1 := func() {
|
||||||
|
@ -64,16 +64,16 @@ f1 := func() {
|
||||||
return f2();
|
return f2();
|
||||||
};
|
};
|
||||||
|
|
||||||
out = f1();`, 3)
|
out = f1();`, nil, 3)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 1; a += 4 - 2; return a }; return f2(); }; out = f1()`, 3)
|
expect(t, `f1 := func() { f2 := func() { a := 1; a += 4 - 2; return a }; return f2(); }; out = f1()`, nil, 3)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 1; return a }; return f2(); }; out = f1()`, 2)
|
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 1; return a }; return f2(); }; out = f1()`, nil, 2)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 5 - 4; return a }; return f2(); }; out = f1()`, 2)
|
expect(t, `f1 := func() { f2 := func() { a := 3; a -= 5 - 4; return a }; return f2(); }; out = f1()`, nil, 2)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 4; return a }; return f2(); }; out = f1()`, 8)
|
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 4; return a }; return f2(); }; out = f1()`, nil, 8)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 1 + 3; return a }; return f2(); }; out = f1()`, 8)
|
expect(t, `f1 := func() { f2 := func() { a := 2; a *= 1 + 3; return a }; return f2(); }; out = f1()`, nil, 8)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 2; return a }; return f2(); }; out = f1()`, 5)
|
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 2; return a }; return f2(); }; out = f1()`, nil, 5)
|
||||||
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 5 - 3; return a }; return f2(); }; out = f1()`, 5)
|
expect(t, `f1 := func() { f2 := func() { a := 10; a /= 5 - 3; return a }; return f2(); }; out = f1()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `a := 1; f1 := func() { f2 := func() { a += 2; return a }; return f2(); }; out = f1()`, 3)
|
expect(t, `a := 1; f1 := func() { f2 := func() { a += 2; return a }; return f2(); }; out = f1()`, nil, 3)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a) {
|
f1 := func(a) {
|
||||||
|
@ -85,7 +85,7 @@ out = f1();`, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
out = f1(3)(4)
|
out = f1(3)(4)
|
||||||
`, 11)
|
`, nil, 11)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -101,7 +101,7 @@ out = f1();`, 3)
|
||||||
}()
|
}()
|
||||||
return a
|
return a
|
||||||
}()
|
}()
|
||||||
`, 3)
|
`, nil, 3)
|
||||||
|
|
||||||
// write on free variables
|
// write on free variables
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -114,7 +114,7 @@ out = f1();`, 3)
|
||||||
}()
|
}()
|
||||||
}
|
}
|
||||||
out = f1()
|
out = f1()
|
||||||
`, 8)
|
`, nil, 8)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -127,7 +127,7 @@ out = f1();`, 3)
|
||||||
}
|
}
|
||||||
return f1()
|
return f1()
|
||||||
}()()
|
}()()
|
||||||
`, 20)
|
`, nil, 20)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
it := func(seq, fn) {
|
it := func(seq, fn) {
|
||||||
|
@ -145,7 +145,7 @@ out = f1();`, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
out = foo(2)
|
out = foo(2)
|
||||||
`, 5)
|
`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
it := func(seq, fn) {
|
it := func(seq, fn) {
|
||||||
|
@ -163,7 +163,7 @@ out = f1();`, 3)
|
||||||
}
|
}
|
||||||
|
|
||||||
out = foo(2)
|
out = foo(2)
|
||||||
`, 12)
|
`, nil, 12)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -173,7 +173,7 @@ out = func() {
|
||||||
}()
|
}()
|
||||||
return a
|
return a
|
||||||
}()
|
}()
|
||||||
`, 2)
|
`, nil, 2)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f := func() {
|
f := func() {
|
||||||
|
@ -188,7 +188,7 @@ m := f()
|
||||||
m.b()
|
m.b()
|
||||||
m.c()
|
m.c()
|
||||||
out = m.d()
|
out = m.d()
|
||||||
`, 6)
|
`, nil, 6)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
each := func(s, x) { for i:=0; i<len(s); i++ { x(s[i]) } }
|
each := func(s, x) { for i:=0; i<len(s); i++ { x(s[i]) } }
|
||||||
|
@ -203,11 +203,11 @@ out = func() {
|
||||||
return a + b
|
return a + b
|
||||||
}
|
}
|
||||||
}()(20)
|
}()(20)
|
||||||
`, 136)
|
`, nil, 136)
|
||||||
|
|
||||||
// assigning different type value
|
// assigning different type value
|
||||||
expect(t, `a := 1; a = "foo"; out = a`, "foo") // global
|
expect(t, `a := 1; a = "foo"; out = a`, nil, "foo") // global
|
||||||
expect(t, `func() { a := 1; a = "foo"; out = a }()`, "foo") // local
|
expect(t, `func() { a := 1; a = "foo"; out = a }()`, nil, "foo") // local
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
a := 5
|
a := 5
|
||||||
|
@ -215,19 +215,19 @@ out = func() {
|
||||||
a = "foo"
|
a = "foo"
|
||||||
return a
|
return a
|
||||||
}()
|
}()
|
||||||
}()`, "foo") // free
|
}()`, nil, "foo") // free
|
||||||
|
|
||||||
// variables declared in if/for blocks
|
// variables declared in if/for blocks
|
||||||
expect(t, `for a:=0; a<5; a++ {}; a := "foo"; out = a`, "foo")
|
expect(t, `for a:=0; a<5; a++ {}; a := "foo"; out = a`, nil, "foo")
|
||||||
expect(t, `func() { for a:=0; a<5; a++ {}; a := "foo"; out = a }()`, "foo")
|
expect(t, `func() { for a:=0; a<5; a++ {}; a := "foo"; out = a }()`, nil, "foo")
|
||||||
|
|
||||||
// selectors
|
// selectors
|
||||||
expect(t, `a:=[1,2,3]; a[1] = 5; out = a[1]`, 5)
|
expect(t, `a:=[1,2,3]; a[1] = 5; out = a[1]`, nil, 5)
|
||||||
expect(t, `a:=[1,2,3]; a[1] += 5; out = a[1]`, 7)
|
expect(t, `a:=[1,2,3]; a[1] += 5; out = a[1]`, nil, 7)
|
||||||
expect(t, `a:={b:1,c:2}; a.b = 5; out = a.b`, 5)
|
expect(t, `a:={b:1,c:2}; a.b = 5; out = a.b`, nil, 5)
|
||||||
expect(t, `a:={b:1,c:2}; a.b += 5; out = a.b`, 6)
|
expect(t, `a:={b:1,c:2}; a.b += 5; out = a.b`, nil, 6)
|
||||||
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.b`, 3)
|
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.b`, nil, 3)
|
||||||
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.c`, 2)
|
expect(t, `a:={b:1,c:2}; a.b += a.c; out = a.c`, nil, 2)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
b: [1, 2, 3],
|
b: [1, 2, 3],
|
||||||
|
@ -239,7 +239,7 @@ a := {
|
||||||
}
|
}
|
||||||
a.c.f[1] += 2
|
a.c.f[1] += 2
|
||||||
out = a["c"]["f"][1]
|
out = a["c"]["f"][1]
|
||||||
`, 10)
|
`, nil, 10)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -252,7 +252,7 @@ a := {
|
||||||
}
|
}
|
||||||
a.c.h = "bar"
|
a.c.h = "bar"
|
||||||
out = a.c.h
|
out = a.c.h
|
||||||
`, "bar")
|
`, nil, "bar")
|
||||||
|
|
||||||
expectError(t, `
|
expectError(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -263,9 +263,5 @@ a := {
|
||||||
f: [9, 8]
|
f: [9, 8]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
a.x.e = "bar"`, "not index-assignable")
|
a.x.e = "bar"`, nil, "not index-assignable")
|
||||||
|
|
||||||
// multi-variables
|
|
||||||
//expect(t, `a, b = 1, 2; out = a + b`, 3)
|
|
||||||
//expect(t, `a, b = 1, 2; a, b = 2, 3; out = a + b`, 5)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,34 +3,34 @@ package runtime_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestBitwise(t *testing.T) {
|
func TestBitwise(t *testing.T) {
|
||||||
expect(t, `out = 1 & 1`, 1&1)
|
expect(t, `out = 1 & 1`, nil, 1)
|
||||||
expect(t, `out = 1 & 0`, 1&0)
|
expect(t, `out = 1 & 0`, nil, 0)
|
||||||
expect(t, `out = 0 & 1`, 0&1)
|
expect(t, `out = 0 & 1`, nil, 0)
|
||||||
expect(t, `out = 0 & 0`, 0&0)
|
expect(t, `out = 0 & 0`, nil, 0)
|
||||||
expect(t, `out = 1 | 1`, 1|1)
|
expect(t, `out = 1 | 1`, nil, 1)
|
||||||
expect(t, `out = 1 | 0`, 1|0)
|
expect(t, `out = 1 | 0`, nil, 1)
|
||||||
expect(t, `out = 0 | 1`, 0|1)
|
expect(t, `out = 0 | 1`, nil, 1)
|
||||||
expect(t, `out = 0 | 0`, 0|0)
|
expect(t, `out = 0 | 0`, nil, 0)
|
||||||
expect(t, `out = 1 ^ 1`, 1^1)
|
expect(t, `out = 1 ^ 1`, nil, 0)
|
||||||
expect(t, `out = 1 ^ 0`, 1^0)
|
expect(t, `out = 1 ^ 0`, nil, 1)
|
||||||
expect(t, `out = 0 ^ 1`, 0^1)
|
expect(t, `out = 0 ^ 1`, nil, 1)
|
||||||
expect(t, `out = 0 ^ 0`, 0^0)
|
expect(t, `out = 0 ^ 0`, nil, 0)
|
||||||
expect(t, `out = 1 &^ 1`, 1&^1)
|
expect(t, `out = 1 &^ 1`, nil, 0)
|
||||||
expect(t, `out = 1 &^ 0`, 1&^0)
|
expect(t, `out = 1 &^ 0`, nil, 1)
|
||||||
expect(t, `out = 0 &^ 1`, 0&^1)
|
expect(t, `out = 0 &^ 1`, nil, 0)
|
||||||
expect(t, `out = 0 &^ 0`, 0&^0)
|
expect(t, `out = 0 &^ 0`, nil, 0)
|
||||||
expect(t, `out = 1 << 2`, 1<<2)
|
expect(t, `out = 1 << 2`, nil, 4)
|
||||||
expect(t, `out = 16 >> 2`, 16>>2)
|
expect(t, `out = 16 >> 2`, nil, 4)
|
||||||
|
|
||||||
expect(t, `out = 1; out &= 1`, 1)
|
expect(t, `out = 1; out &= 1`, nil, 1)
|
||||||
expect(t, `out = 1; out |= 0`, 1)
|
expect(t, `out = 1; out |= 0`, nil, 1)
|
||||||
expect(t, `out = 1; out ^= 0`, 1)
|
expect(t, `out = 1; out ^= 0`, nil, 1)
|
||||||
expect(t, `out = 1; out &^= 0`, 1)
|
expect(t, `out = 1; out &^= 0`, nil, 1)
|
||||||
expect(t, `out = 1; out <<= 2`, 4)
|
expect(t, `out = 1; out <<= 2`, nil, 4)
|
||||||
expect(t, `out = 16; out >>= 2`, 4)
|
expect(t, `out = 16; out >>= 2`, nil, 4)
|
||||||
|
|
||||||
expect(t, `out = ^0`, ^0)
|
expect(t, `out = ^0`, nil, ^0)
|
||||||
expect(t, `out = ^1`, ^1)
|
expect(t, `out = ^1`, nil, ^1)
|
||||||
expect(t, `out = ^55`, ^55)
|
expect(t, `out = ^55`, nil, ^55)
|
||||||
expect(t, `out = ^-55`, ^-55)
|
expect(t, `out = ^-55`, nil, ^-55)
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,38 +5,38 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestBoolean(t *testing.T) {
|
func TestBoolean(t *testing.T) {
|
||||||
expect(t, `out = true`, true)
|
expect(t, `out = true`, nil, true)
|
||||||
expect(t, `out = false`, false)
|
expect(t, `out = false`, nil, false)
|
||||||
|
|
||||||
expect(t, `out = 1 < 2`, true)
|
expect(t, `out = 1 < 2`, nil, true)
|
||||||
expect(t, `out = 1 > 2`, false)
|
expect(t, `out = 1 > 2`, nil, false)
|
||||||
expect(t, `out = 1 < 1`, false)
|
expect(t, `out = 1 < 1`, nil, false)
|
||||||
expect(t, `out = 1 > 2`, false)
|
expect(t, `out = 1 > 2`, nil, false)
|
||||||
expect(t, `out = 1 == 1`, true)
|
expect(t, `out = 1 == 1`, nil, true)
|
||||||
expect(t, `out = 1 != 1`, false)
|
expect(t, `out = 1 != 1`, nil, false)
|
||||||
expect(t, `out = 1 == 2`, false)
|
expect(t, `out = 1 == 2`, nil, false)
|
||||||
expect(t, `out = 1 != 2`, true)
|
expect(t, `out = 1 != 2`, nil, true)
|
||||||
expect(t, `out = 1 <= 2`, true)
|
expect(t, `out = 1 <= 2`, nil, true)
|
||||||
expect(t, `out = 1 >= 2`, false)
|
expect(t, `out = 1 >= 2`, nil, false)
|
||||||
expect(t, `out = 1 <= 1`, true)
|
expect(t, `out = 1 <= 1`, nil, true)
|
||||||
expect(t, `out = 1 >= 2`, false)
|
expect(t, `out = 1 >= 2`, nil, false)
|
||||||
|
|
||||||
expect(t, `out = true == true`, true)
|
expect(t, `out = true == true`, nil, true)
|
||||||
expect(t, `out = false == false`, true)
|
expect(t, `out = false == false`, nil, true)
|
||||||
expect(t, `out = true == false`, false)
|
expect(t, `out = true == false`, nil, false)
|
||||||
expect(t, `out = true != false`, true)
|
expect(t, `out = true != false`, nil, true)
|
||||||
expect(t, `out = false != true`, true)
|
expect(t, `out = false != true`, nil, true)
|
||||||
expect(t, `out = (1 < 2) == true`, true)
|
expect(t, `out = (1 < 2) == true`, nil, true)
|
||||||
expect(t, `out = (1 < 2) == false`, false)
|
expect(t, `out = (1 < 2) == false`, nil, false)
|
||||||
expect(t, `out = (1 > 2) == true`, false)
|
expect(t, `out = (1 > 2) == true`, nil, false)
|
||||||
expect(t, `out = (1 > 2) == false`, true)
|
expect(t, `out = (1 > 2) == false`, nil, true)
|
||||||
|
|
||||||
expectError(t, `5 + true`, "invalid operation")
|
expectError(t, `5 + true`, nil, "invalid operation")
|
||||||
expectError(t, `5 + true; 5`, "invalid operation")
|
expectError(t, `5 + true; 5`, nil, "invalid operation")
|
||||||
expectError(t, `-true`, "invalid operation")
|
expectError(t, `-true`, nil, "invalid operation")
|
||||||
expectError(t, `true + false`, "invalid operation")
|
expectError(t, `true + false`, nil, "invalid operation")
|
||||||
expectError(t, `5; true + false; 5`, "invalid operation")
|
expectError(t, `5; true + false; 5`, nil, "invalid operation")
|
||||||
expectError(t, `if (10 > 1) { true + false; }`, "invalid operation")
|
expectError(t, `if (10 > 1) { true + false; }`, nil, "invalid operation")
|
||||||
expectError(t, `
|
expectError(t, `
|
||||||
func() {
|
func() {
|
||||||
if (10 > 1) {
|
if (10 > 1) {
|
||||||
|
@ -47,9 +47,9 @@ func() {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
`, "invalid operation")
|
`, nil, "invalid operation")
|
||||||
expectError(t, `if (true + false) { 10 }`, "invalid operation")
|
expectError(t, `if (true + false) { 10 }`, nil, "invalid operation")
|
||||||
expectError(t, `10 + (true + false)`, "invalid operation")
|
expectError(t, `10 + (true + false)`, nil, "invalid operation")
|
||||||
expectError(t, `(true + false) + 20`, "invalid operation")
|
expectError(t, `(true + false) + 20`, nil, "invalid operation")
|
||||||
expectError(t, `!(true + false)`, "invalid operation")
|
expectError(t, `!(true + false)`, nil, "invalid operation")
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,198 +8,155 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestBuiltinFunction(t *testing.T) {
|
func TestBuiltinFunction(t *testing.T) {
|
||||||
expect(t, `out = len("")`, 0)
|
expect(t, `out = len("")`, nil, 0)
|
||||||
expect(t, `out = len("four")`, 4)
|
expect(t, `out = len("four")`, nil, 4)
|
||||||
expect(t, `out = len("hello world")`, 11)
|
expect(t, `out = len("hello world")`, nil, 11)
|
||||||
expect(t, `out = len([])`, 0)
|
expect(t, `out = len([])`, nil, 0)
|
||||||
expect(t, `out = len([1, 2, 3])`, 3)
|
expect(t, `out = len([1, 2, 3])`, nil, 3)
|
||||||
expect(t, `out = len({})`, 0)
|
expect(t, `out = len({})`, nil, 0)
|
||||||
expect(t, `out = len({a:1, b:2})`, 2)
|
expect(t, `out = len({a:1, b:2})`, nil, 2)
|
||||||
expect(t, `out = len(immutable([]))`, 0)
|
expect(t, `out = len(immutable([]))`, nil, 0)
|
||||||
expect(t, `out = len(immutable([1, 2, 3]))`, 3)
|
expect(t, `out = len(immutable([1, 2, 3]))`, nil, 3)
|
||||||
expect(t, `out = len(immutable({}))`, 0)
|
expect(t, `out = len(immutable({}))`, nil, 0)
|
||||||
expect(t, `out = len(immutable({a:1, b:2}))`, 2)
|
expect(t, `out = len(immutable({a:1, b:2}))`, nil, 2)
|
||||||
expectError(t, `len(1)`, "invalid type for argument")
|
expectError(t, `len(1)`, nil, "invalid type for argument")
|
||||||
expectError(t, `len("one", "two")`, "wrong number of arguments")
|
expectError(t, `len("one", "two")`, nil, "wrong number of arguments")
|
||||||
|
|
||||||
expect(t, `out = copy(1)`, 1)
|
expect(t, `out = copy(1)`, nil, 1)
|
||||||
expectError(t, `copy(1, 2)`, "wrong number of arguments")
|
expectError(t, `copy(1, 2)`, nil, "wrong number of arguments")
|
||||||
|
|
||||||
expect(t, `out = append([1, 2, 3], 4)`, ARR{1, 2, 3, 4})
|
expect(t, `out = append([1, 2, 3], 4)`, nil, ARR{1, 2, 3, 4})
|
||||||
expect(t, `out = append([1, 2, 3], 4, 5, 6)`, ARR{1, 2, 3, 4, 5, 6})
|
expect(t, `out = append([1, 2, 3], 4, 5, 6)`, nil, ARR{1, 2, 3, 4, 5, 6})
|
||||||
expect(t, `out = append([1, 2, 3], "foo", false)`, ARR{1, 2, 3, "foo", false})
|
expect(t, `out = append([1, 2, 3], "foo", false)`, nil, ARR{1, 2, 3, "foo", false})
|
||||||
|
|
||||||
expect(t, `out = int(1)`, 1)
|
expect(t, `out = int(1)`, nil, 1)
|
||||||
expect(t, `out = int(1.8)`, 1)
|
expect(t, `out = int(1.8)`, nil, 1)
|
||||||
expect(t, `out = int("-522")`, -522)
|
expect(t, `out = int("-522")`, nil, -522)
|
||||||
expect(t, `out = int(true)`, 1)
|
expect(t, `out = int(true)`, nil, 1)
|
||||||
expect(t, `out = int(false)`, 0)
|
expect(t, `out = int(false)`, nil, 0)
|
||||||
expect(t, `out = int('8')`, 56)
|
expect(t, `out = int('8')`, nil, 56)
|
||||||
expect(t, `out = int([1])`, objects.UndefinedValue)
|
expect(t, `out = int([1])`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = int({a: 1})`, objects.UndefinedValue)
|
expect(t, `out = int({a: 1})`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = int(undefined)`, objects.UndefinedValue)
|
expect(t, `out = int(undefined)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = int("-522", 1)`, -522)
|
expect(t, `out = int("-522", 1)`, nil, -522)
|
||||||
expect(t, `out = int(undefined, 1)`, 1)
|
expect(t, `out = int(undefined, 1)`, nil, 1)
|
||||||
expect(t, `out = int(undefined, 1.8)`, 1.8)
|
expect(t, `out = int(undefined, 1.8)`, nil, 1.8)
|
||||||
expect(t, `out = int(undefined, string(1))`, "1")
|
expect(t, `out = int(undefined, string(1))`, nil, "1")
|
||||||
expect(t, `out = int(undefined, undefined)`, objects.UndefinedValue)
|
expect(t, `out = int(undefined, undefined)`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `out = string(1)`, "1")
|
expect(t, `out = string(1)`, nil, "1")
|
||||||
expect(t, `out = string(1.8)`, "1.8")
|
expect(t, `out = string(1.8)`, nil, "1.8")
|
||||||
expect(t, `out = string("-522")`, "-522")
|
expect(t, `out = string("-522")`, nil, "-522")
|
||||||
expect(t, `out = string(true)`, "true")
|
expect(t, `out = string(true)`, nil, "true")
|
||||||
expect(t, `out = string(false)`, "false")
|
expect(t, `out = string(false)`, nil, "false")
|
||||||
expect(t, `out = string('8')`, "8")
|
expect(t, `out = string('8')`, nil, "8")
|
||||||
expect(t, `out = string([1,8.1,true,3])`, "[1, 8.1, true, 3]")
|
expect(t, `out = string([1,8.1,true,3])`, nil, "[1, 8.1, true, 3]")
|
||||||
expect(t, `out = string({b: "foo"})`, `{b: "foo"}`)
|
expect(t, `out = string({b: "foo"})`, nil, `{b: "foo"}`)
|
||||||
expect(t, `out = string(undefined)`, objects.UndefinedValue) // not "undefined"
|
expect(t, `out = string(undefined)`, nil, objects.UndefinedValue) // not "undefined"
|
||||||
expect(t, `out = string(1, "-522")`, "1")
|
expect(t, `out = string(1, "-522")`, nil, "1")
|
||||||
expect(t, `out = string(undefined, "-522")`, "-522") // not "undefined"
|
expect(t, `out = string(undefined, "-522")`, nil, "-522") // not "undefined"
|
||||||
|
|
||||||
expect(t, `out = float(1)`, 1.0)
|
expect(t, `out = float(1)`, nil, 1.0)
|
||||||
expect(t, `out = float(1.8)`, 1.8)
|
expect(t, `out = float(1.8)`, nil, 1.8)
|
||||||
expect(t, `out = float("-52.2")`, -52.2)
|
expect(t, `out = float("-52.2")`, nil, -52.2)
|
||||||
expect(t, `out = float(true)`, objects.UndefinedValue)
|
expect(t, `out = float(true)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float(false)`, objects.UndefinedValue)
|
expect(t, `out = float(false)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float('8')`, objects.UndefinedValue)
|
expect(t, `out = float('8')`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float([1,8.1,true,3])`, objects.UndefinedValue)
|
expect(t, `out = float([1,8.1,true,3])`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float({a: 1, b: "foo"})`, objects.UndefinedValue)
|
expect(t, `out = float({a: 1, b: "foo"})`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float(undefined)`, objects.UndefinedValue)
|
expect(t, `out = float(undefined)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = float("-52.2", 1.8)`, -52.2)
|
expect(t, `out = float("-52.2", 1.8)`, nil, -52.2)
|
||||||
expect(t, `out = float(undefined, 1)`, 1)
|
expect(t, `out = float(undefined, 1)`, nil, 1)
|
||||||
expect(t, `out = float(undefined, 1.8)`, 1.8)
|
expect(t, `out = float(undefined, 1.8)`, nil, 1.8)
|
||||||
expect(t, `out = float(undefined, "-52.2")`, "-52.2")
|
expect(t, `out = float(undefined, "-52.2")`, nil, "-52.2")
|
||||||
expect(t, `out = float(undefined, char(56))`, '8')
|
expect(t, `out = float(undefined, char(56))`, nil, '8')
|
||||||
expect(t, `out = float(undefined, undefined)`, objects.UndefinedValue)
|
expect(t, `out = float(undefined, undefined)`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `out = char(56)`, '8')
|
expect(t, `out = char(56)`, nil, '8')
|
||||||
expect(t, `out = char(1.8)`, objects.UndefinedValue)
|
expect(t, `out = char(1.8)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char("-52.2")`, objects.UndefinedValue)
|
expect(t, `out = char("-52.2")`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char(true)`, objects.UndefinedValue)
|
expect(t, `out = char(true)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char(false)`, objects.UndefinedValue)
|
expect(t, `out = char(false)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char('8')`, '8')
|
expect(t, `out = char('8')`, nil, '8')
|
||||||
expect(t, `out = char([1,8.1,true,3])`, objects.UndefinedValue)
|
expect(t, `out = char([1,8.1,true,3])`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char({a: 1, b: "foo"})`, objects.UndefinedValue)
|
expect(t, `out = char({a: 1, b: "foo"})`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char(undefined)`, objects.UndefinedValue)
|
expect(t, `out = char(undefined)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = char(56, 'a')`, '8')
|
expect(t, `out = char(56, 'a')`, nil, '8')
|
||||||
expect(t, `out = char(undefined, '8')`, '8')
|
expect(t, `out = char(undefined, '8')`, nil, '8')
|
||||||
expect(t, `out = char(undefined, 56)`, 56)
|
expect(t, `out = char(undefined, 56)`, nil, 56)
|
||||||
expect(t, `out = char(undefined, "-52.2")`, "-52.2")
|
expect(t, `out = char(undefined, "-52.2")`, nil, "-52.2")
|
||||||
expect(t, `out = char(undefined, undefined)`, objects.UndefinedValue)
|
expect(t, `out = char(undefined, undefined)`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `out = bool(1)`, true) // non-zero integer: true
|
expect(t, `out = bool(1)`, nil, true) // non-zero integer: true
|
||||||
expect(t, `out = bool(0)`, false) // zero: true
|
expect(t, `out = bool(0)`, nil, false) // zero: true
|
||||||
expect(t, `out = bool(1.8)`, true) // all floats (except for NaN): true
|
expect(t, `out = bool(1.8)`, nil, true) // all floats (except for NaN): true
|
||||||
expect(t, `out = bool(0.0)`, true) // all floats (except for NaN): true
|
expect(t, `out = bool(0.0)`, nil, true) // all floats (except for NaN): true
|
||||||
expect(t, `out = bool("false")`, true) // non-empty string: true
|
expect(t, `out = bool("false")`, nil, true) // non-empty string: true
|
||||||
expect(t, `out = bool("")`, false) // empty string: false
|
expect(t, `out = bool("")`, nil, false) // empty string: false
|
||||||
expect(t, `out = bool(true)`, true) // true: true
|
expect(t, `out = bool(true)`, nil, true) // true: true
|
||||||
expect(t, `out = bool(false)`, false) // false: false
|
expect(t, `out = bool(false)`, nil, false) // false: false
|
||||||
expect(t, `out = bool('8')`, true) // non-zero chars: true
|
expect(t, `out = bool('8')`, nil, true) // non-zero chars: true
|
||||||
expect(t, `out = bool(char(0))`, false) // zero char: false
|
expect(t, `out = bool(char(0))`, nil, false) // zero char: false
|
||||||
expect(t, `out = bool([1])`, true) // non-empty arrays: true
|
expect(t, `out = bool([1])`, nil, true) // non-empty arrays: true
|
||||||
expect(t, `out = bool([])`, false) // empty array: false
|
expect(t, `out = bool([])`, nil, false) // empty array: false
|
||||||
expect(t, `out = bool({a: 1})`, true) // non-empty maps: true
|
expect(t, `out = bool({a: 1})`, nil, true) // non-empty maps: true
|
||||||
expect(t, `out = bool({})`, false) // empty maps: false
|
expect(t, `out = bool({})`, nil, false) // empty maps: false
|
||||||
expect(t, `out = bool(undefined)`, false) // undefined: false
|
expect(t, `out = bool(undefined)`, nil, false) // undefined: false
|
||||||
|
|
||||||
expect(t, `out = bytes(1)`, []byte{0})
|
expect(t, `out = bytes(1)`, nil, []byte{0})
|
||||||
expect(t, `out = bytes(1.8)`, objects.UndefinedValue)
|
expect(t, `out = bytes(1.8)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes("-522")`, []byte{'-', '5', '2', '2'})
|
expect(t, `out = bytes("-522")`, nil, []byte{'-', '5', '2', '2'})
|
||||||
expect(t, `out = bytes(true)`, objects.UndefinedValue)
|
expect(t, `out = bytes(true)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes(false)`, objects.UndefinedValue)
|
expect(t, `out = bytes(false)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes('8')`, objects.UndefinedValue)
|
expect(t, `out = bytes('8')`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes([1])`, objects.UndefinedValue)
|
expect(t, `out = bytes([1])`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes({a: 1})`, objects.UndefinedValue)
|
expect(t, `out = bytes({a: 1})`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes(undefined)`, objects.UndefinedValue)
|
expect(t, `out = bytes(undefined)`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = bytes("-522", ['8'])`, []byte{'-', '5', '2', '2'})
|
expect(t, `out = bytes("-522", ['8'])`, nil, []byte{'-', '5', '2', '2'})
|
||||||
expect(t, `out = bytes(undefined, "-522")`, "-522")
|
expect(t, `out = bytes(undefined, "-522")`, nil, "-522")
|
||||||
expect(t, `out = bytes(undefined, 1)`, 1)
|
expect(t, `out = bytes(undefined, 1)`, nil, 1)
|
||||||
expect(t, `out = bytes(undefined, 1.8)`, 1.8)
|
expect(t, `out = bytes(undefined, 1.8)`, nil, 1.8)
|
||||||
expect(t, `out = bytes(undefined, int("-522"))`, -522)
|
expect(t, `out = bytes(undefined, int("-522"))`, nil, -522)
|
||||||
expect(t, `out = bytes(undefined, undefined)`, objects.UndefinedValue)
|
expect(t, `out = bytes(undefined, undefined)`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `out = is_error(error(1))`, true)
|
expect(t, `out = is_error(error(1))`, nil, true)
|
||||||
expect(t, `out = is_error(1)`, false)
|
expect(t, `out = is_error(1)`, nil, false)
|
||||||
|
|
||||||
expect(t, `out = is_undefined(undefined)`, true)
|
expect(t, `out = is_undefined(undefined)`, nil, true)
|
||||||
expect(t, `out = is_undefined(error(1))`, false)
|
expect(t, `out = is_undefined(error(1))`, nil, false)
|
||||||
|
|
||||||
// to_json
|
|
||||||
expect(t, `out = to_json(5)`, []byte("5"))
|
|
||||||
expect(t, `out = to_json("foobar")`, []byte(`"foobar"`))
|
|
||||||
expect(t, `out = to_json({foo: 5})`, []byte("{\"foo\":5}"))
|
|
||||||
expect(t, `out = to_json(immutable({foo: 5}))`, []byte("{\"foo\":5}"))
|
|
||||||
expect(t, `out = to_json([1,2,3])`, []byte("[1,2,3]"))
|
|
||||||
expect(t, `out = to_json(immutable([1,2,3]))`, []byte("[1,2,3]"))
|
|
||||||
expect(t, `out = to_json({foo: "bar"})`, []byte("{\"foo\":\"bar\"}"))
|
|
||||||
expect(t, `out = to_json({foo: 1.8})`, []byte("{\"foo\":1.8}"))
|
|
||||||
expect(t, `out = to_json({foo: true})`, []byte("{\"foo\":true}"))
|
|
||||||
expect(t, `out = to_json({foo: '8'})`, []byte("{\"foo\":56}"))
|
|
||||||
expect(t, `out = to_json({foo: bytes("foo")})`, []byte("{\"foo\":\"Zm9v\"}")) // json encoding returns []byte as base64 encoded string
|
|
||||||
expect(t, `out = to_json({foo: ["bar", 1, 1.8, '8', true]})`, []byte("{\"foo\":[\"bar\",1,1.8,56,true]}"))
|
|
||||||
expect(t, `out = to_json({foo: immutable(["bar", 1, 1.8, '8', true])})`, []byte("{\"foo\":[\"bar\",1,1.8,56,true]}"))
|
|
||||||
expect(t, `out = to_json({foo: [["bar", 1], ["bar", 1]]})`, []byte("{\"foo\":[[\"bar\",1],[\"bar\",1]]}"))
|
|
||||||
expect(t, `out = to_json({foo: {string: "bar", int: 1, float: 1.8, char: '8', bool: true}})`, []byte("{\"foo\":{\"bool\":true,\"char\":56,\"float\":1.8,\"int\":1,\"string\":\"bar\"}}"))
|
|
||||||
expect(t, `out = to_json({foo: immutable({string: "bar", int: 1, float: 1.8, char: '8', bool: true})})`, []byte("{\"foo\":{\"bool\":true,\"char\":56,\"float\":1.8,\"int\":1,\"string\":\"bar\"}}"))
|
|
||||||
expect(t, `out = to_json({foo: {map1: {string: "bar"}, map2: {int: "1"}}})`, []byte("{\"foo\":{\"map1\":{\"string\":\"bar\"},\"map2\":{\"int\":\"1\"}}}"))
|
|
||||||
expect(t, `out = to_json([["bar", 1], ["bar", 1]])`, []byte("[[\"bar\",1],[\"bar\",1]]"))
|
|
||||||
expect(t, `out = to_json(error("my error"))`, []byte(`"error: \"my error\""`))
|
|
||||||
|
|
||||||
// from_json
|
|
||||||
expect(t, `out = from_json("{\"foo\":5}").foo`, 5.0)
|
|
||||||
expect(t, `out = from_json("{\"foo\":\"bar\"}").foo`, "bar")
|
|
||||||
expect(t, `out = from_json("{\"foo\":1.8}").foo`, 1.8)
|
|
||||||
expect(t, `out = from_json("{\"foo\":true}").foo`, true)
|
|
||||||
expect(t, `out = from_json("{\"foo\":[\"bar\",1,1.8,56,true]}").foo`, ARR{"bar", 1.0, 1.8, 56.0, true})
|
|
||||||
expect(t, `out = from_json("{\"foo\":[[\"bar\",1],[\"bar\",1]]}").foo[0]`, ARR{"bar", 1.0})
|
|
||||||
expect(t, `out = from_json("{\"foo\":{\"bool\":true,\"char\":56,\"float\":1.8,\"int\":1,\"string\":\"bar\"}}").foo.bool`, true)
|
|
||||||
expect(t, `out = from_json("{\"foo\":{\"map1\":{\"string\":\"bar\"},\"map2\":{\"int\":\"1\"}}}").foo.map1.string`, "bar")
|
|
||||||
expect(t, `out = from_json("5")`, 5.0)
|
|
||||||
expect(t, `out = from_json("\"foobar\"")`, "foobar")
|
|
||||||
expect(t, `out = from_json("[\"bar\",1,1.8,56,true]")`, ARR{"bar", 1.0, 1.8, 56.0, true})
|
|
||||||
|
|
||||||
// sprintf
|
|
||||||
expect(t, `out = sprintf("")`, "")
|
|
||||||
expect(t, `out = sprintf("foo")`, "foo")
|
|
||||||
expect(t, `out = sprintf("foo %d %v %s", 1, 2, "bar")`, "foo 1 2 bar")
|
|
||||||
expect(t, `out = sprintf("foo %v", [1, "bar", true])`, "foo [1 bar true]")
|
|
||||||
expect(t, `out = sprintf("foo %v %d", [1, "bar", true], 19)`, "foo [1 bar true] 19")
|
|
||||||
expectError(t, `sprintf(1)`, "invalid type for argument") // format has to be String
|
|
||||||
expectError(t, `sprintf('c')`, "invalid type for argument") // format has to be String
|
|
||||||
|
|
||||||
// type_name
|
// type_name
|
||||||
expect(t, `out = type_name(1)`, "int")
|
expect(t, `out = type_name(1)`, nil, "int")
|
||||||
expect(t, `out = type_name(1.1)`, "float")
|
expect(t, `out = type_name(1.1)`, nil, "float")
|
||||||
expect(t, `out = type_name("a")`, "string")
|
expect(t, `out = type_name("a")`, nil, "string")
|
||||||
expect(t, `out = type_name([1,2,3])`, "array")
|
expect(t, `out = type_name([1,2,3])`, nil, "array")
|
||||||
expect(t, `out = type_name({k:1})`, "map")
|
expect(t, `out = type_name({k:1})`, nil, "map")
|
||||||
expect(t, `out = type_name('a')`, "char")
|
expect(t, `out = type_name('a')`, nil, "char")
|
||||||
expect(t, `out = type_name(true)`, "bool")
|
expect(t, `out = type_name(true)`, nil, "bool")
|
||||||
expect(t, `out = type_name(false)`, "bool")
|
expect(t, `out = type_name(false)`, nil, "bool")
|
||||||
expect(t, `out = type_name(bytes( 1))`, "bytes")
|
expect(t, `out = type_name(bytes( 1))`, nil, "bytes")
|
||||||
expect(t, `out = type_name(undefined)`, "undefined")
|
expect(t, `out = type_name(undefined)`, nil, "undefined")
|
||||||
expect(t, `out = type_name(error("err"))`, "error")
|
expect(t, `out = type_name(error("err"))`, nil, "error")
|
||||||
expect(t, `out = type_name(func() {})`, "compiled-function")
|
expect(t, `out = type_name(func() {})`, nil, "compiled-function")
|
||||||
expect(t, `a := func(x) { return func() { return x } }; out = type_name(a(5))`, "closure") // closure
|
expect(t, `a := func(x) { return func() { return x } }; out = type_name(a(5))`, nil, "closure") // closure
|
||||||
|
|
||||||
// is_function
|
// is_function
|
||||||
expect(t, `out = is_function(1)`, false)
|
expect(t, `out = is_function(1)`, nil, false)
|
||||||
expect(t, `out = is_function(func() {})`, true)
|
expect(t, `out = is_function(func() {})`, nil, true)
|
||||||
expect(t, `out = is_function(func(x) { return x })`, true)
|
expect(t, `out = is_function(func(x) { return x })`, nil, true)
|
||||||
expect(t, `out = is_function(len)`, false) // builtin function
|
expect(t, `out = is_function(len)`, nil, false) // builtin function
|
||||||
expect(t, `a := func(x) { return func() { return x } }; out = is_function(a)`, true) // function
|
expect(t, `a := func(x) { return func() { return x } }; out = is_function(a)`, nil, true) // function
|
||||||
expect(t, `a := func(x) { return func() { return x } }; out = is_function(a(5))`, true) // closure
|
expect(t, `a := func(x) { return func() { return x } }; out = is_function(a(5))`, nil, true) // closure
|
||||||
expectWithSymbols(t, `out = is_function(x)`, false, SYM{"x": &StringArray{Value: []string{"foo", "bar"}}}) // user object
|
expect(t, `out = is_function(x)`, Opts().Symbol("x", &StringArray{Value: []string{"foo", "bar"}}).Skip2ndPass(), false) // user object
|
||||||
|
|
||||||
// is_callable
|
// is_callable
|
||||||
expect(t, `out = is_callable(1)`, false)
|
expect(t, `out = is_callable(1)`, nil, false)
|
||||||
expect(t, `out = is_callable(func() {})`, true)
|
expect(t, `out = is_callable(func() {})`, nil, true)
|
||||||
expect(t, `out = is_callable(func(x) { return x })`, true)
|
expect(t, `out = is_callable(func(x) { return x })`, nil, true)
|
||||||
expect(t, `out = is_callable(len)`, true) // builtin function
|
expect(t, `out = is_callable(len)`, nil, true) // builtin function
|
||||||
expect(t, `a := func(x) { return func() { return x } }; out = is_callable(a)`, true) // function
|
expect(t, `a := func(x) { return func() { return x } }; out = is_callable(a)`, nil, true) // function
|
||||||
expect(t, `a := func(x) { return func() { return x } }; out = is_callable(a(5))`, true) // closure
|
expect(t, `a := func(x) { return func() { return x } }; out = is_callable(a(5))`, nil, true) // closure
|
||||||
expectWithSymbols(t, `out = is_callable(x)`, true, SYM{"x": &StringArray{Value: []string{"foo", "bar"}}}) // user object
|
expect(t, `out = is_callable(x)`, Opts().Symbol("x", &StringArray{Value: []string{"foo", "bar"}}).Skip2ndPass(), true) // user object
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestBytesN(t *testing.T) {
|
func TestBytesN(t *testing.T) {
|
||||||
|
@ -207,11 +164,11 @@ func TestBytesN(t *testing.T) {
|
||||||
defer func() { tengo.MaxBytesLen = curMaxBytesLen }()
|
defer func() { tengo.MaxBytesLen = curMaxBytesLen }()
|
||||||
tengo.MaxBytesLen = 10
|
tengo.MaxBytesLen = 10
|
||||||
|
|
||||||
expect(t, `out = bytes(0)`, make([]byte, 0))
|
expect(t, `out = bytes(0)`, nil, make([]byte, 0))
|
||||||
expect(t, `out = bytes(10)`, make([]byte, 10))
|
expect(t, `out = bytes(10)`, nil, make([]byte, 10))
|
||||||
expectError(t, `bytes(11)`, "bytes size limit")
|
expectError(t, `bytes(11)`, nil, "bytes size limit")
|
||||||
|
|
||||||
tengo.MaxBytesLen = 1000
|
tengo.MaxBytesLen = 1000
|
||||||
expect(t, `out = bytes(1000)`, make([]byte, 1000))
|
expect(t, `out = bytes(1000)`, nil, make([]byte, 1000))
|
||||||
expectError(t, `bytes(1001)`, "bytes size limit")
|
expectError(t, `bytes(1001)`, nil, "bytes size limit")
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,12 +7,12 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestBytes(t *testing.T) {
|
func TestBytes(t *testing.T) {
|
||||||
expect(t, `out = bytes("Hello World!")`, []byte("Hello World!"))
|
expect(t, `out = bytes("Hello World!")`, nil, []byte("Hello World!"))
|
||||||
expect(t, `out = bytes("Hello") + bytes(" ") + bytes("World!")`, []byte("Hello World!"))
|
expect(t, `out = bytes("Hello") + bytes(" ") + bytes("World!")`, nil, []byte("Hello World!"))
|
||||||
|
|
||||||
// bytes[] -> int
|
// bytes[] -> int
|
||||||
expect(t, `out = bytes("abcde")[0]`, 97)
|
expect(t, `out = bytes("abcde")[0]`, nil, 97)
|
||||||
expect(t, `out = bytes("abcde")[1]`, 98)
|
expect(t, `out = bytes("abcde")[1]`, nil, 98)
|
||||||
expect(t, `out = bytes("abcde")[4]`, 101)
|
expect(t, `out = bytes("abcde")[4]`, nil, 101)
|
||||||
expect(t, `out = bytes("abcde")[10]`, objects.UndefinedValue)
|
expect(t, `out = bytes("abcde")[10]`, nil, objects.UndefinedValue)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,9 @@ package runtime_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestCall(t *testing.T) {
|
func TestCall(t *testing.T) {
|
||||||
expect(t, `a := { b: func(x) { return x + 2 } }; out = a.b(5)`, 7)
|
expect(t, `a := { b: func(x) { return x + 2 } }; out = a.b(5)`, nil, 7)
|
||||||
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a.b.c(5)`, 7)
|
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a.b.c(5)`, nil, 7)
|
||||||
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a["b"].c(5)`, 7)
|
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a["b"].c(5)`, nil, 7)
|
||||||
expectError(t, `a := 1
|
expectError(t, `a := 1
|
||||||
b := func(a, c) {
|
b := func(a, c) {
|
||||||
c(a)
|
c(a)
|
||||||
|
@ -15,5 +15,5 @@ c := func(a) {
|
||||||
a()
|
a()
|
||||||
}
|
}
|
||||||
b(a, c)
|
b(a, c)
|
||||||
`, "Runtime Error: not callable: int\n\tat test:7:4\n\tat test:3:4\n\tat test:9:1")
|
`, nil, "Runtime Error: not callable: int\n\tat test:7:4\n\tat test:3:4\n\tat test:9:1")
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,21 +3,21 @@ package runtime_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestChar(t *testing.T) {
|
func TestChar(t *testing.T) {
|
||||||
expect(t, `out = 'a'`, 'a')
|
expect(t, `out = 'a'`, nil, 'a')
|
||||||
expect(t, `out = '九'`, rune(20061))
|
expect(t, `out = '九'`, nil, rune(20061))
|
||||||
expect(t, `out = 'Æ'`, rune(198))
|
expect(t, `out = 'Æ'`, nil, rune(198))
|
||||||
|
|
||||||
expect(t, `out = '0' + '9'`, rune(105))
|
expect(t, `out = '0' + '9'`, nil, rune(105))
|
||||||
expect(t, `out = '0' + 9`, '9')
|
expect(t, `out = '0' + 9`, nil, '9')
|
||||||
expect(t, `out = '9' - 4`, '5')
|
expect(t, `out = '9' - 4`, nil, '5')
|
||||||
expect(t, `out = '0' == '0'`, true)
|
expect(t, `out = '0' == '0'`, nil, true)
|
||||||
expect(t, `out = '0' != '0'`, false)
|
expect(t, `out = '0' != '0'`, nil, false)
|
||||||
expect(t, `out = '2' < '4'`, true)
|
expect(t, `out = '2' < '4'`, nil, true)
|
||||||
expect(t, `out = '2' > '4'`, false)
|
expect(t, `out = '2' > '4'`, nil, false)
|
||||||
expect(t, `out = '2' <= '4'`, true)
|
expect(t, `out = '2' <= '4'`, nil, true)
|
||||||
expect(t, `out = '2' >= '4'`, false)
|
expect(t, `out = '2' >= '4'`, nil, false)
|
||||||
expect(t, `out = '4' < '4'`, false)
|
expect(t, `out = '4' < '4'`, nil, false)
|
||||||
expect(t, `out = '4' > '4'`, false)
|
expect(t, `out = '4' > '4'`, nil, false)
|
||||||
expect(t, `out = '4' <= '4'`, true)
|
expect(t, `out = '4' <= '4'`, nil, true)
|
||||||
expect(t, `out = '4' >= '4'`, true)
|
expect(t, `out = '4' >= '4'`, nil, true)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,25 +3,25 @@ package runtime_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestCondExpr(t *testing.T) {
|
func TestCondExpr(t *testing.T) {
|
||||||
expect(t, `out = true ? 5 : 10`, 5)
|
expect(t, `out = true ? 5 : 10`, nil, 5)
|
||||||
expect(t, `out = false ? 5 : 10`, 10)
|
expect(t, `out = false ? 5 : 10`, nil, 10)
|
||||||
expect(t, `out = (1 == 1) ? 2 + 3 : 12 - 2`, 5)
|
expect(t, `out = (1 == 1) ? 2 + 3 : 12 - 2`, nil, 5)
|
||||||
expect(t, `out = (1 != 1) ? 2 + 3 : 12 - 2`, 10)
|
expect(t, `out = (1 != 1) ? 2 + 3 : 12 - 2`, nil, 10)
|
||||||
expect(t, `out = (1 == 1) ? true ? 10 - 8 : 1 + 3 : 12 - 2`, 2)
|
expect(t, `out = (1 == 1) ? true ? 10 - 8 : 1 + 3 : 12 - 2`, nil, 2)
|
||||||
expect(t, `out = (1 == 1) ? false ? 10 - 8 : 1 + 3 : 12 - 2`, 4)
|
expect(t, `out = (1 == 1) ? false ? 10 - 8 : 1 + 3 : 12 - 2`, nil, 4)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
f1 := func() { out += 10 }
|
f1 := func() { out += 10 }
|
||||||
f2 := func() { out = -out }
|
f2 := func() { out = -out }
|
||||||
true ? f1() : f2()
|
true ? f1() : f2()
|
||||||
`, 10)
|
`, nil, 10)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 5
|
out = 5
|
||||||
f1 := func() { out += 10 }
|
f1 := func() { out += 10 }
|
||||||
f2 := func() { out = -out }
|
f2 := func() { out = -out }
|
||||||
false ? f1() : f2()
|
false ? f1() : f2()
|
||||||
`, -5)
|
`, nil, -5)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a) { return a + 2 }
|
f1 := func(a) { return a + 2 }
|
||||||
f2 := func(a) { return a - 2 }
|
f2 := func(a) { return a - 2 }
|
||||||
|
@ -33,13 +33,13 @@ f := func(c) {
|
||||||
}
|
}
|
||||||
|
|
||||||
out = [f(0), f(1), f(2)]
|
out = [f(0), f(1), f(2)]
|
||||||
`, ARR{2, 11, -2})
|
`, nil, ARR{2, 11, -2})
|
||||||
|
|
||||||
expect(t, `f := func(a) { return -a }; out = f(true ? 5 : 3)`, -5)
|
expect(t, `f := func(a) { return -a }; out = f(true ? 5 : 3)`, nil, -5)
|
||||||
expect(t, `out = [false?5:10, true?1:2]`, ARR{10, 1})
|
expect(t, `out = [false?5:10, true?1:2]`, nil, ARR{10, 1})
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 1 > 2 ?
|
out = 1 > 2 ?
|
||||||
1 + 2 + 3 :
|
1 + 2 + 3 :
|
||||||
10 - 5`, 5)
|
10 - 5`, nil, 5)
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,8 +43,8 @@ func TestEquality(t *testing.T) {
|
||||||
func testEquality(t *testing.T, lhs, rhs string, expected bool) {
|
func testEquality(t *testing.T, lhs, rhs string, expected bool) {
|
||||||
// 1. equality is commutative
|
// 1. equality is commutative
|
||||||
// 2. equality and inequality must be always opposite
|
// 2. equality and inequality must be always opposite
|
||||||
expect(t, fmt.Sprintf("out = %s == %s", lhs, rhs), expected)
|
expect(t, fmt.Sprintf("out = %s == %s", lhs, rhs), nil, expected)
|
||||||
expect(t, fmt.Sprintf("out = %s == %s", rhs, lhs), expected)
|
expect(t, fmt.Sprintf("out = %s == %s", rhs, lhs), nil, expected)
|
||||||
expect(t, fmt.Sprintf("out = %s != %s", lhs, rhs), !expected)
|
expect(t, fmt.Sprintf("out = %s != %s", lhs, rhs), nil, !expected)
|
||||||
expect(t, fmt.Sprintf("out = %s != %s", rhs, lhs), !expected)
|
expect(t, fmt.Sprintf("out = %s != %s", rhs, lhs), nil, !expected)
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,16 +5,16 @@ import "testing"
|
||||||
func TestVMErrorInfo(t *testing.T) {
|
func TestVMErrorInfo(t *testing.T) {
|
||||||
expectError(t, `a := 5
|
expectError(t, `a := 5
|
||||||
a + "boo"`,
|
a + "boo"`,
|
||||||
"Runtime Error: invalid operation: int + string\n\tat test:2:1")
|
nil, "Runtime Error: invalid operation: int + string\n\tat test:2:1")
|
||||||
|
|
||||||
expectError(t, `a := 5
|
expectError(t, `a := 5
|
||||||
b := a(5)`,
|
b := a(5)`,
|
||||||
"Runtime Error: not callable: int\n\tat test:2:6")
|
nil, "Runtime Error: not callable: int\n\tat test:2:6")
|
||||||
|
|
||||||
expectError(t, `a := 5
|
expectError(t, `a := 5
|
||||||
b := {}
|
b := {}
|
||||||
b.x.y = 10`,
|
b.x.y = 10`,
|
||||||
"Runtime Error: not index-assignable: undefined\n\tat test:3:1")
|
nil, "Runtime Error: not index-assignable: undefined\n\tat test:3:1")
|
||||||
|
|
||||||
expectError(t, `
|
expectError(t, `
|
||||||
a := func() {
|
a := func() {
|
||||||
|
@ -22,27 +22,28 @@ a := func() {
|
||||||
b += "foo"
|
b += "foo"
|
||||||
}
|
}
|
||||||
a()`,
|
a()`,
|
||||||
"Runtime Error: invalid operation: int + string\n\tat test:4:2")
|
nil, "Runtime Error: invalid operation: int + string\n\tat test:4:2")
|
||||||
|
|
||||||
expectErrorWithUserModules(t, `a := 5
|
expectError(t, `a := 5
|
||||||
a + import("mod1")`, map[string]string{
|
a + import("mod1")`, Opts().Module(
|
||||||
"mod1": `export "foo"`,
|
"mod1", `export "foo"`,
|
||||||
}, ": invalid operation: int + string\n\tat test:2:2")
|
), ": invalid operation: int + string\n\tat test:2:1")
|
||||||
|
|
||||||
expectErrorWithUserModules(t, `a := import("mod1")()`, map[string]string{
|
expectError(t, `a := import("mod1")()`,
|
||||||
"mod1": `
|
Opts().Module(
|
||||||
|
"mod1", `
|
||||||
export func() {
|
export func() {
|
||||||
b := 5
|
b := 5
|
||||||
return b + "foo"
|
return b + "foo"
|
||||||
}`,
|
}`), "Runtime Error: invalid operation: int + string\n\tat mod1:4:9")
|
||||||
}, "Runtime Error: invalid operation: int + string\n\tat mod1:4:9")
|
|
||||||
|
|
||||||
expectErrorWithUserModules(t, `a := import("mod1")()`, map[string]string{
|
expectError(t, `a := import("mod1")()`,
|
||||||
"mod1": `export import("mod2")()`,
|
Opts().Module(
|
||||||
"mod2": `
|
"mod1", `export import("mod2")()`).
|
||||||
|
Module(
|
||||||
|
"mod2", `
|
||||||
export func() {
|
export func() {
|
||||||
b := 5
|
b := 5
|
||||||
return b + "foo"
|
return b + "foo"
|
||||||
}`,
|
}`), "Runtime Error: invalid operation: int + string\n\tat mod2:4:9")
|
||||||
}, "Runtime Error: invalid operation: int + string\n\tat mod2:4:9")
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,17 +5,17 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestError(t *testing.T) {
|
func TestError(t *testing.T) {
|
||||||
expect(t, `out = error(1)`, errorObject(1))
|
expect(t, `out = error(1)`, nil, errorObject(1))
|
||||||
expect(t, `out = error(1).value`, 1)
|
expect(t, `out = error(1).value`, nil, 1)
|
||||||
expect(t, `out = error("some error")`, errorObject("some error"))
|
expect(t, `out = error("some error")`, nil, errorObject("some error"))
|
||||||
expect(t, `out = error("some" + " error")`, errorObject("some error"))
|
expect(t, `out = error("some" + " error")`, nil, errorObject("some error"))
|
||||||
expect(t, `out = func() { return error(5) }()`, errorObject(5))
|
expect(t, `out = func() { return error(5) }()`, nil, errorObject(5))
|
||||||
expect(t, `out = error(error("foo"))`, errorObject(errorObject("foo")))
|
expect(t, `out = error(error("foo"))`, nil, errorObject(errorObject("foo")))
|
||||||
expect(t, `out = error("some error")`, errorObject("some error"))
|
expect(t, `out = error("some error")`, nil, errorObject("some error"))
|
||||||
expect(t, `out = error("some error").value`, "some error")
|
expect(t, `out = error("some error").value`, nil, "some error")
|
||||||
expect(t, `out = error("some error")["value"]`, "some error")
|
expect(t, `out = error("some error")["value"]`, nil, "some error")
|
||||||
|
|
||||||
expectError(t, `error("error").err`, "invalid index on error")
|
expectError(t, `error("error").err`, nil, "invalid index on error")
|
||||||
expectError(t, `error("error").value_`, "invalid index on error")
|
expectError(t, `error("error").value_`, nil, "invalid index on error")
|
||||||
expectError(t, `error([1,2,3])[1]`, "invalid index on error")
|
expectError(t, `error([1,2,3])[1]`, nil, "invalid index on error")
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,11 +5,11 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestFloat(t *testing.T) {
|
func TestFloat(t *testing.T) {
|
||||||
expect(t, `out = 0.0`, 0.0)
|
expect(t, `out = 0.0`, nil, 0.0)
|
||||||
expect(t, `out = -10.3`, -10.3)
|
expect(t, `out = -10.3`, nil, -10.3)
|
||||||
expect(t, `out = 3.2 + 2.0 * -4.0`, -4.8)
|
expect(t, `out = 3.2 + 2.0 * -4.0`, nil, -4.8)
|
||||||
expect(t, `out = 4 + 2.3`, 6.3)
|
expect(t, `out = 4 + 2.3`, nil, 6.3)
|
||||||
expect(t, `out = 2.3 + 4`, 6.3)
|
expect(t, `out = 2.3 + 4`, nil, 6.3)
|
||||||
expect(t, `out = +5.0`, 5.0)
|
expect(t, `out = +5.0`, nil, 5.0)
|
||||||
expect(t, `out = -5.0 + +5.0`, 0.0)
|
expect(t, `out = -5.0 + +5.0`, nil, 0.0)
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,20 +6,20 @@ import (
|
||||||
|
|
||||||
func TestForIn(t *testing.T) {
|
func TestForIn(t *testing.T) {
|
||||||
// array
|
// array
|
||||||
expect(t, `out = 0; for x in [1, 2, 3] { out += x }`, 6) // value
|
expect(t, `out = 0; for x in [1, 2, 3] { out += x }`, nil, 6) // value
|
||||||
expect(t, `out = 0; for i, x in [1, 2, 3] { out += i + x }`, 9) // index, value
|
expect(t, `out = 0; for i, x in [1, 2, 3] { out += i + x }`, nil, 9) // index, value
|
||||||
expect(t, `out = 0; func() { for i, x in [1, 2, 3] { out += i + x } }()`, 9) // index, value
|
expect(t, `out = 0; func() { for i, x in [1, 2, 3] { out += i + x } }()`, nil, 9) // index, value
|
||||||
expect(t, `out = 0; for i, _ in [1, 2, 3] { out += i }`, 3) // index, _
|
expect(t, `out = 0; for i, _ in [1, 2, 3] { out += i }`, nil, 3) // index, _
|
||||||
expect(t, `out = 0; func() { for i, _ in [1, 2, 3] { out += i } }()`, 3) // index, _
|
expect(t, `out = 0; func() { for i, _ in [1, 2, 3] { out += i } }()`, nil, 3) // index, _
|
||||||
|
|
||||||
// map
|
// map
|
||||||
expect(t, `out = 0; for v in {a:2,b:3,c:4} { out += v }`, 9) // value
|
expect(t, `out = 0; for v in {a:2,b:3,c:4} { out += v }`, nil, 9) // value
|
||||||
expect(t, `out = ""; for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } }`, "b") // key, value
|
expect(t, `out = ""; for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } }`, nil, "b") // key, value
|
||||||
expect(t, `out = ""; for k, _ in {a:2} { out += k }`, "a") // key, _
|
expect(t, `out = ""; for k, _ in {a:2} { out += k }`, nil, "a") // key, _
|
||||||
expect(t, `out = 0; for _, v in {a:2,b:3,c:4} { out += v }`, 9) // _, value
|
expect(t, `out = 0; for _, v in {a:2,b:3,c:4} { out += v }`, nil, 9) // _, value
|
||||||
expect(t, `out = ""; func() { for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } } }()`, "b") // key, value
|
expect(t, `out = ""; func() { for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } } }()`, nil, "b") // key, value
|
||||||
|
|
||||||
// string
|
// string
|
||||||
expect(t, `out = ""; for c in "abcde" { out += c }`, "abcde")
|
expect(t, `out = ""; for c in "abcde" { out += c }`, nil, "abcde")
|
||||||
expect(t, `out = ""; for i, c in "abcde" { if i == 2 { continue }; out += c }`, "abde")
|
expect(t, `out = ""; for i, c in "abcde" { if i == 2 { continue }; out += c }`, nil, "abde")
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ func TestFor(t *testing.T) {
|
||||||
if out == 5 {
|
if out == 5 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}`, 5)
|
}`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -21,7 +21,7 @@ func TestFor(t *testing.T) {
|
||||||
if out == 5 {
|
if out == 5 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}`, 5)
|
}`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -31,7 +31,7 @@ func TestFor(t *testing.T) {
|
||||||
if a == 3 { continue }
|
if a == 3 { continue }
|
||||||
if a == 5 { break }
|
if a == 5 { break }
|
||||||
out += a
|
out += a
|
||||||
}`, 7) // 1 + 2 + 4
|
}`, nil, 7) // 1 + 2 + 4
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -41,7 +41,7 @@ func TestFor(t *testing.T) {
|
||||||
if a == 3 { continue }
|
if a == 3 { continue }
|
||||||
out += a
|
out += a
|
||||||
if a == 5 { break }
|
if a == 5 { break }
|
||||||
}`, 12) // 1 + 2 + 4 + 5
|
}`, nil, 12) // 1 + 2 + 4 + 5
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -50,7 +50,7 @@ func TestFor(t *testing.T) {
|
||||||
if out == 5 {
|
if out == 5 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}`, 5)
|
}`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := 0
|
a := 0
|
||||||
|
@ -60,7 +60,7 @@ func TestFor(t *testing.T) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out = a`, 5)
|
out = a`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -70,7 +70,7 @@ func TestFor(t *testing.T) {
|
||||||
if a == 3 { continue }
|
if a == 3 { continue }
|
||||||
if a == 5 { break }
|
if a == 5 { break }
|
||||||
out += a
|
out += a
|
||||||
}`, 7) // 1 + 2 + 4
|
}`, nil, 7) // 1 + 2 + 4
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -80,7 +80,7 @@ func TestFor(t *testing.T) {
|
||||||
if a == 3 { continue }
|
if a == 3 { continue }
|
||||||
out += a
|
out += a
|
||||||
if a == 5 { break }
|
if a == 5 { break }
|
||||||
}`, 12) // 1 + 2 + 4 + 5
|
}`, nil, 12) // 1 + 2 + 4 + 5
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -91,13 +91,13 @@ func TestFor(t *testing.T) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
for a:=1; a<=10; a++ {
|
for a:=1; a<=10; a++ {
|
||||||
out += a
|
out += a
|
||||||
}`, 55)
|
}`, nil, 55)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -105,7 +105,7 @@ func TestFor(t *testing.T) {
|
||||||
for b:=3; b<=6; b++ {
|
for b:=3; b<=6; b++ {
|
||||||
out += b
|
out += b
|
||||||
}
|
}
|
||||||
}`, 54)
|
}`, nil, 54)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -116,7 +116,7 @@ func TestFor(t *testing.T) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -127,7 +127,7 @@ func TestFor(t *testing.T) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -139,7 +139,7 @@ func TestFor(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return a
|
return a
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -151,7 +151,7 @@ func TestFor(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return a
|
return a
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -165,7 +165,7 @@ func TestFor(t *testing.T) {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
return a
|
return a
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -179,7 +179,7 @@ func TestFor(t *testing.T) {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
return a
|
return a
|
||||||
}()`, 5)
|
}()`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -188,7 +188,7 @@ func TestFor(t *testing.T) {
|
||||||
sum += a
|
sum += a
|
||||||
}
|
}
|
||||||
return sum
|
return sum
|
||||||
}()`, 55)
|
}()`, nil, 55)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
|
@ -199,7 +199,7 @@ func TestFor(t *testing.T) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return sum
|
return sum
|
||||||
}()`, 48) // (3+4+5) * 4
|
}()`, nil, 48) // (3+4+5) * 4
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := 1
|
a := 1
|
||||||
|
@ -208,7 +208,7 @@ func TestFor(t *testing.T) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out = a`, 5)
|
out = a`, nil, 5)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -220,7 +220,7 @@ func TestFor(t *testing.T) {
|
||||||
if a == 5 {
|
if a == 5 {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
}`, 12) // 1 + 2 + 4 + 5
|
}`, nil, 12) // 1 + 2 + 4 + 5
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = 0
|
out = 0
|
||||||
|
@ -234,5 +234,5 @@ func TestFor(t *testing.T) {
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
a++
|
a++
|
||||||
}`, 12) // 1 + 2 + 4 + 5
|
}`, nil, 12) // 1 + 2 + 4 + 5
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,16 +8,16 @@ import (
|
||||||
|
|
||||||
func TestFunction(t *testing.T) {
|
func TestFunction(t *testing.T) {
|
||||||
// function with no "return" statement returns "invalid" value.
|
// function with no "return" statement returns "invalid" value.
|
||||||
expect(t, `f1 := func() {}; out = f1();`, objects.UndefinedValue)
|
expect(t, `f1 := func() {}; out = f1();`, nil, objects.UndefinedValue)
|
||||||
expect(t, `f1 := func() {}; f2 := func() { return f1(); }; f1(); out = f2();`, 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);`, objects.UndefinedValue)
|
expect(t, `f := func(x) { x; }; out = f(5);`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `f := func(x) { return x; }; out = f(5);`, 5)
|
expect(t, `f := func(x) { return x; }; out = f(5);`, nil, 5)
|
||||||
expect(t, `f := func(x) { return x * 2; }; out = f(5);`, 10)
|
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);`, 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));`, 20)
|
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)`, 5)
|
expect(t, `out = func(x) { return x; }(5)`, nil, 5)
|
||||||
expect(t, `x := 10; f := func(x) { return x; }; f(5); out = x;`, 10)
|
expect(t, `x := 10; f := func(x) { return x; }; f(5); out = x;`, nil, 10)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f2 := func(a) {
|
f2 := func(a) {
|
||||||
|
@ -29,7 +29,7 @@ func TestFunction(t *testing.T) {
|
||||||
};
|
};
|
||||||
|
|
||||||
out = f2(10);
|
out = f2(10);
|
||||||
`, 60)
|
`, nil, 60)
|
||||||
|
|
||||||
// closures
|
// closures
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -39,7 +39,7 @@ func TestFunction(t *testing.T) {
|
||||||
|
|
||||||
add2 := newAdder(2);
|
add2 := newAdder(2);
|
||||||
out = add2(5);
|
out = add2(5);
|
||||||
`, 7)
|
`, nil, 7)
|
||||||
|
|
||||||
// function as a argument
|
// function as a argument
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -48,17 +48,17 @@ func TestFunction(t *testing.T) {
|
||||||
applyFunc := func(a, b, f) { return f(a, b) };
|
applyFunc := func(a, b, f) { return f(a, b) };
|
||||||
|
|
||||||
out = applyFunc(applyFunc(2, 2, add), 3, sub);
|
out = applyFunc(applyFunc(2, 2, add), 3, sub);
|
||||||
`, 1)
|
`, nil, 1)
|
||||||
|
|
||||||
expect(t, `f1 := func() { return 5 + 10; }; out = f1();`, 15)
|
expect(t, `f1 := func() { return 5 + 10; }; out = f1();`, nil, 15)
|
||||||
expect(t, `f1 := func() { return 1 }; f2 := func() { return 2 }; out = f1() + f2()`, 3)
|
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()`, 6)
|
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();`, 99)
|
expect(t, `f1 := func() { return 99; 100 }; out = f1();`, nil, 99)
|
||||||
expect(t, `f1 := func() { return 99; return 100 }; out = f1();`, 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()();`, 33)
|
expect(t, `f1 := func() { return 33; }; f2 := func() { return f1 }; out = f2()();`, nil, 33)
|
||||||
expect(t, `one := func() { one = 1; return one }; out = one()`, 1)
|
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()`, 3)
|
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()`, 10)
|
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, `
|
expect(t, `
|
||||||
foo1 := func() {
|
foo1 := func() {
|
||||||
foo := 50
|
foo := 50
|
||||||
|
@ -68,7 +68,7 @@ func TestFunction(t *testing.T) {
|
||||||
foo := 100
|
foo := 100
|
||||||
return foo
|
return foo
|
||||||
}
|
}
|
||||||
out = foo1() + foo2()`, 150)
|
out = foo1() + foo2()`, nil, 150)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
g := 50;
|
g := 50;
|
||||||
minusOne := func() {
|
minusOne := func() {
|
||||||
|
@ -80,35 +80,35 @@ func TestFunction(t *testing.T) {
|
||||||
return g - n;
|
return g - n;
|
||||||
};
|
};
|
||||||
out = minusOne() + minusTwo()
|
out = minusOne() + minusTwo()
|
||||||
`, 97)
|
`, nil, 97)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func() {
|
f1 := func() {
|
||||||
f2 := func() { return 1; }
|
f2 := func() { return 1; }
|
||||||
return f2
|
return f2
|
||||||
};
|
};
|
||||||
out = f1()()
|
out = f1()()
|
||||||
`, 1)
|
`, nil, 1)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a) { return a; };
|
f1 := func(a) { return a; };
|
||||||
out = f1(4)`, 4)
|
out = f1(4)`, nil, 4)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a, b) { return a + b; };
|
f1 := func(a, b) { return a + b; };
|
||||||
out = f1(1, 2)`, 3)
|
out = f1(1, 2)`, nil, 3)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
sum := func(a, b) {
|
sum := func(a, b) {
|
||||||
c := a + b;
|
c := a + b;
|
||||||
return c;
|
return c;
|
||||||
};
|
};
|
||||||
out = sum(1, 2);`, 3)
|
out = sum(1, 2);`, nil, 3)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
sum := func(a, b) {
|
sum := func(a, b) {
|
||||||
c := a + b;
|
c := a + b;
|
||||||
return c;
|
return c;
|
||||||
};
|
};
|
||||||
out = sum(1, 2) + sum(3, 4);`, 10)
|
out = sum(1, 2) + sum(3, 4);`, nil, 10)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
sum := func(a, b) {
|
sum := func(a, b) {
|
||||||
|
@ -118,7 +118,7 @@ func TestFunction(t *testing.T) {
|
||||||
outer := func() {
|
outer := func() {
|
||||||
return sum(1, 2) + sum(3, 4)
|
return sum(1, 2) + sum(3, 4)
|
||||||
};
|
};
|
||||||
out = outer();`, 10)
|
out = outer();`, nil, 10)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
g := 10;
|
g := 10;
|
||||||
|
@ -133,11 +133,11 @@ func TestFunction(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
out = outer() + g
|
out = outer() + g
|
||||||
`, 50)
|
`, nil, 50)
|
||||||
|
|
||||||
expectError(t, `func() { return 1; }(1)`, "wrong number of arguments")
|
expectError(t, `func() { return 1; }(1)`, nil, "wrong number of arguments")
|
||||||
expectError(t, `func(a) { return a; }()`, "wrong number of arguments")
|
expectError(t, `func(a) { return a; }()`, nil, "wrong number of arguments")
|
||||||
expectError(t, `func(a, b) { return a + b; }(1)`, "wrong number of arguments")
|
expectError(t, `func(a, b) { return a + b; }(1)`, nil, "wrong number of arguments")
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a) {
|
f1 := func(a) {
|
||||||
|
@ -145,7 +145,7 @@ func TestFunction(t *testing.T) {
|
||||||
};
|
};
|
||||||
f2 := f1(99);
|
f2 := f1(99);
|
||||||
out = f2()
|
out = f2()
|
||||||
`, 99)
|
`, nil, 99)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a, b) {
|
f1 := func(a, b) {
|
||||||
|
@ -154,7 +154,7 @@ func TestFunction(t *testing.T) {
|
||||||
|
|
||||||
f2 := f1(1, 2);
|
f2 := f1(1, 2);
|
||||||
out = f2(8);
|
out = f2(8);
|
||||||
`, 11)
|
`, nil, 11)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a, b) {
|
f1 := func(a, b) {
|
||||||
c := a + b;
|
c := a + b;
|
||||||
|
@ -162,7 +162,7 @@ func TestFunction(t *testing.T) {
|
||||||
};
|
};
|
||||||
f2 := f1(1, 2);
|
f2 := f1(1, 2);
|
||||||
out = f2(8);
|
out = f2(8);
|
||||||
`, 11)
|
`, nil, 11)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a, b) {
|
f1 := func(a, b) {
|
||||||
c := a + b;
|
c := a + b;
|
||||||
|
@ -174,7 +174,7 @@ func TestFunction(t *testing.T) {
|
||||||
f2 := f1(1, 2);
|
f2 := f1(1, 2);
|
||||||
f3 := f2(3);
|
f3 := f2(3);
|
||||||
out = f3(8);
|
out = f3(8);
|
||||||
`, 14)
|
`, nil, 14)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := 1;
|
a := 1;
|
||||||
f1 := func(b) {
|
f1 := func(b) {
|
||||||
|
@ -185,7 +185,7 @@ func TestFunction(t *testing.T) {
|
||||||
f2 := f1(2);
|
f2 := f1(2);
|
||||||
f3 := f2(3);
|
f3 := f2(3);
|
||||||
out = f3(8);
|
out = f3(8);
|
||||||
`, 14)
|
`, nil, 14)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func(a, b) {
|
f1 := func(a, b) {
|
||||||
one := func() { return a; };
|
one := func() { return a; };
|
||||||
|
@ -194,7 +194,7 @@ func TestFunction(t *testing.T) {
|
||||||
};
|
};
|
||||||
f2 := f1(9, 90);
|
f2 := f1(9, 90);
|
||||||
out = f2();
|
out = f2();
|
||||||
`, 99)
|
`, nil, 99)
|
||||||
|
|
||||||
// global function recursion
|
// global function recursion
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -207,7 +207,7 @@ func TestFunction(t *testing.T) {
|
||||||
return fib(x-1) + fib(x-2)
|
return fib(x-1) + fib(x-2)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out = fib(15)`, 610)
|
out = fib(15)`, nil, 610)
|
||||||
|
|
||||||
// local function recursion
|
// local function recursion
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -216,9 +216,9 @@ out = func() {
|
||||||
return x == 0 ? 0 : x + sum(x-1)
|
return x == 0 ? 0 : x + sum(x-1)
|
||||||
}
|
}
|
||||||
return sum(5)
|
return sum(5)
|
||||||
}()`, 15)
|
}()`, nil, 15)
|
||||||
|
|
||||||
expectError(t, `return 5`, "return not allowed outside function")
|
expectError(t, `return 5`, nil, "return not allowed outside function")
|
||||||
|
|
||||||
// closure and block scopes
|
// closure and block scopes
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -230,7 +230,7 @@ func() {
|
||||||
out = a + 5
|
out = a + 5
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}()`, 15)
|
}()`, nil, 15)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
a := 10
|
a := 10
|
||||||
|
@ -240,7 +240,7 @@ func() {
|
||||||
out = a + b()
|
out = a + b()
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}()`, 15)
|
}()`, nil, 15)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
a := 10
|
a := 10
|
||||||
|
@ -252,5 +252,5 @@ func() {
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
}()
|
}()
|
||||||
}()`, 15)
|
}()`, nil, 15)
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,29 +7,29 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestIf(t *testing.T) {
|
func TestIf(t *testing.T) {
|
||||||
expect(t, `if (true) { out = 10 }`, 10)
|
expect(t, `if (true) { out = 10 }`, nil, 10)
|
||||||
expect(t, `if (false) { out = 10 }`, objects.UndefinedValue)
|
expect(t, `if (false) { out = 10 }`, nil, objects.UndefinedValue)
|
||||||
expect(t, `if (false) { out = 10 } else { out = 20 }`, 20)
|
expect(t, `if (false) { out = 10 } else { out = 20 }`, nil, 20)
|
||||||
expect(t, `if (1) { out = 10 }`, 10)
|
expect(t, `if (1) { out = 10 }`, nil, 10)
|
||||||
expect(t, `if (0) { out = 10 } else { out = 20 }`, 20)
|
expect(t, `if (0) { out = 10 } else { out = 20 }`, nil, 20)
|
||||||
expect(t, `if (1 < 2) { out = 10 }`, 10)
|
expect(t, `if (1 < 2) { out = 10 }`, nil, 10)
|
||||||
expect(t, `if (1 > 2) { out = 10 }`, objects.UndefinedValue)
|
expect(t, `if (1 > 2) { out = 10 }`, nil, objects.UndefinedValue)
|
||||||
expect(t, `if (1 < 2) { out = 10 } else { out = 20 }`, 10)
|
expect(t, `if (1 < 2) { out = 10 } else { out = 20 }`, nil, 10)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else { out = 20 }`, 20)
|
expect(t, `if (1 > 2) { out = 10 } else { out = 20 }`, nil, 20)
|
||||||
|
|
||||||
expect(t, `if (1 < 2) { out = 10 } else if (1 > 2) { out = 20 } else { out = 30 }`, 10)
|
expect(t, `if (1 < 2) { out = 10 } else if (1 > 2) { out = 20 } else { out = 30 }`, nil, 10)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { out = 20 } else { out = 30 }`, 20)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { out = 20 } else { out = 30 }`, nil, 20)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { out = 30 }`, 30)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { out = 30 }`, nil, 30)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else if (1 < 2) { out = 30 } else { out = 40 }`, 30)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else if (1 < 2) { out = 30 } else { out = 40 }`, nil, 30)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { out = 20; out = 21; out = 22 } else { out = 30 }`, 22)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { out = 20; out = 21; out = 22 } else { out = 30 }`, nil, 22)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { out = 30; out = 31; out = 32}`, 32)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { out = 30; out = 31; out = 32}`, nil, 32)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { if (1 == 2) { out = 21 } else { out = 22 } } else { out = 30 }`, 22)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { if (1 == 2) { out = 21 } else { out = 22 } } else { out = 30 }`, nil, 22)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { if (1 == 2) { out = 21 } else if (2 == 3) { out = 22 } else { out = 23 } } else { out = 30 }`, 23)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 < 2) { if (1 == 2) { out = 21 } else if (2 == 3) { out = 22 } else { out = 23 } } else { out = 30 }`, nil, 23)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { if (1 == 2) { out = 21 } else if (2 == 3) { out = 22 } else { out = 23 } } else { out = 30 }`, 30)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { if (1 == 2) { out = 21 } else if (2 == 3) { out = 22 } else { out = 23 } } else { out = 30 }`, nil, 30)
|
||||||
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { if (1 == 2) { out = 31 } else if (2 == 3) { out = 32 } else { out = 33 } }`, 33)
|
expect(t, `if (1 > 2) { out = 10 } else if (1 == 2) { out = 20 } else { if (1 == 2) { out = 31 } else if (2 == 3) { out = 32 } else { out = 33 } }`, nil, 33)
|
||||||
|
|
||||||
expect(t, `if a:=0; a<1 { out = 10 }`, 10)
|
expect(t, `if a:=0; a<1 { out = 10 }`, nil, 10)
|
||||||
expect(t, `a:=0; if a++; a==1 { out = 10 }`, 10)
|
expect(t, `a:=0; if a++; a==1 { out = 10 }`, nil, 10)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
a := 1
|
a := 1
|
||||||
|
@ -37,7 +37,7 @@ func() {
|
||||||
out = a
|
out = a
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
`, 2)
|
`, nil, 2)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
a := 1
|
a := 1
|
||||||
|
@ -47,7 +47,7 @@ func() {
|
||||||
out = 20
|
out = 20
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
`, 20)
|
`, nil, 20)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
a := 1
|
a := 1
|
||||||
|
@ -60,5 +60,5 @@ func() {
|
||||||
|
|
||||||
out = a
|
out = a
|
||||||
}()
|
}()
|
||||||
`, 3)
|
`, nil, 3)
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,46 +9,46 @@ import (
|
||||||
func TestImmutable(t *testing.T) {
|
func TestImmutable(t *testing.T) {
|
||||||
// primitive types are already immutable values
|
// primitive types are already immutable values
|
||||||
// immutable expression has no effects.
|
// immutable expression has no effects.
|
||||||
expect(t, `a := immutable(1); out = a`, 1)
|
expect(t, `a := immutable(1); out = a`, nil, 1)
|
||||||
expect(t, `a := 5; b := immutable(a); out = b`, 5)
|
expect(t, `a := 5; b := immutable(a); out = b`, nil, 5)
|
||||||
expect(t, `a := immutable(1); a = 5; out = a`, 5)
|
expect(t, `a := immutable(1); a = 5; out = a`, nil, 5)
|
||||||
|
|
||||||
// array
|
// array
|
||||||
expectError(t, `a := immutable([1, 2, 3]); a[1] = 5`, "not index-assignable")
|
expectError(t, `a := immutable([1, 2, 3]); a[1] = 5`, nil, "not index-assignable")
|
||||||
expectError(t, `a := immutable(["foo", [1,2,3]]); a[1] = "bar"`, "not index-assignable")
|
expectError(t, `a := immutable(["foo", [1,2,3]]); a[1] = "bar"`, nil, "not index-assignable")
|
||||||
expect(t, `a := immutable(["foo", [1,2,3]]); a[1][1] = "bar"; out = a`, IARR{"foo", ARR{1, "bar", 3}})
|
expect(t, `a := immutable(["foo", [1,2,3]]); a[1][1] = "bar"; out = a`, nil, IARR{"foo", ARR{1, "bar", 3}})
|
||||||
expectError(t, `a := immutable(["foo", immutable([1,2,3])]); a[1][1] = "bar"`, "not index-assignable")
|
expectError(t, `a := immutable(["foo", immutable([1,2,3])]); a[1][1] = "bar"`, nil, "not index-assignable")
|
||||||
expectError(t, `a := ["foo", immutable([1,2,3])]; a[1][1] = "bar"`, "not index-assignable")
|
expectError(t, `a := ["foo", immutable([1,2,3])]; a[1][1] = "bar"`, nil, "not index-assignable")
|
||||||
expect(t, `a := immutable([1,2,3]); b := copy(a); b[1] = 5; out = b`, ARR{1, 5, 3})
|
expect(t, `a := immutable([1,2,3]); b := copy(a); b[1] = 5; out = b`, nil, ARR{1, 5, 3})
|
||||||
expect(t, `a := immutable([1,2,3]); b := copy(a); b[1] = 5; out = a`, IARR{1, 2, 3})
|
expect(t, `a := immutable([1,2,3]); b := copy(a); b[1] = 5; out = a`, nil, IARR{1, 2, 3})
|
||||||
expect(t, `out = immutable([1,2,3]) == [1,2,3]`, true)
|
expect(t, `out = immutable([1,2,3]) == [1,2,3]`, nil, true)
|
||||||
expect(t, `out = immutable([1,2,3]) == immutable([1,2,3])`, true)
|
expect(t, `out = immutable([1,2,3]) == immutable([1,2,3])`, nil, true)
|
||||||
expect(t, `out = [1,2,3] == immutable([1,2,3])`, true)
|
expect(t, `out = [1,2,3] == immutable([1,2,3])`, nil, true)
|
||||||
expect(t, `out = immutable([1,2,3]) == [1,2]`, false)
|
expect(t, `out = immutable([1,2,3]) == [1,2]`, nil, false)
|
||||||
expect(t, `out = immutable([1,2,3]) == immutable([1,2])`, false)
|
expect(t, `out = immutable([1,2,3]) == immutable([1,2])`, nil, false)
|
||||||
expect(t, `out = [1,2,3] == immutable([1,2])`, false)
|
expect(t, `out = [1,2,3] == immutable([1,2])`, nil, false)
|
||||||
expect(t, `out = immutable([1, 2, 3, 4])[1]`, 2)
|
expect(t, `out = immutable([1, 2, 3, 4])[1]`, nil, 2)
|
||||||
expect(t, `out = immutable([1, 2, 3, 4])[1:3]`, ARR{2, 3})
|
expect(t, `out = immutable([1, 2, 3, 4])[1:3]`, nil, ARR{2, 3})
|
||||||
expect(t, `a := immutable([1,2,3]); a = 5; out = a`, 5)
|
expect(t, `a := immutable([1,2,3]); a = 5; out = a`, nil, 5)
|
||||||
expect(t, `a := immutable([1, 2, 3]); out = a[5]`, objects.UndefinedValue)
|
expect(t, `a := immutable([1, 2, 3]); out = a[5]`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
// map
|
// map
|
||||||
expectError(t, `a := immutable({b: 1, c: 2}); a.b = 5`, "not index-assignable")
|
expectError(t, `a := immutable({b: 1, c: 2}); a.b = 5`, nil, "not index-assignable")
|
||||||
expectError(t, `a := immutable({b: 1, c: 2}); a["b"] = "bar"`, "not index-assignable")
|
expectError(t, `a := immutable({b: 1, c: 2}); a["b"] = "bar"`, nil, "not index-assignable")
|
||||||
expect(t, `a := immutable({b: 1, c: [1,2,3]}); a.c[1] = "bar"; out = a`, IMAP{"b": 1, "c": ARR{1, "bar", 3}})
|
expect(t, `a := immutable({b: 1, c: [1,2,3]}); a.c[1] = "bar"; out = a`, nil, IMAP{"b": 1, "c": ARR{1, "bar", 3}})
|
||||||
expectError(t, `a := immutable({b: 1, c: immutable([1,2,3])}); a.c[1] = "bar"`, "not index-assignable")
|
expectError(t, `a := immutable({b: 1, c: immutable([1,2,3])}); a.c[1] = "bar"`, nil, "not index-assignable")
|
||||||
expectError(t, `a := {b: 1, c: immutable([1,2,3])}; a.c[1] = "bar"`, "not index-assignable")
|
expectError(t, `a := {b: 1, c: immutable([1,2,3])}; a.c[1] = "bar"`, nil, "not index-assignable")
|
||||||
expect(t, `out = immutable({a:1,b:2}) == {a:1,b:2}`, true)
|
expect(t, `out = immutable({a:1,b:2}) == {a:1,b:2}`, nil, true)
|
||||||
expect(t, `out = immutable({a:1,b:2}) == immutable({a:1,b:2})`, true)
|
expect(t, `out = immutable({a:1,b:2}) == immutable({a:1,b:2})`, nil, true)
|
||||||
expect(t, `out = {a:1,b:2} == immutable({a:1,b:2})`, true)
|
expect(t, `out = {a:1,b:2} == immutable({a:1,b:2})`, nil, true)
|
||||||
expect(t, `out = immutable({a:1,b:2}) == {a:1,b:3}`, false)
|
expect(t, `out = immutable({a:1,b:2}) == {a:1,b:3}`, nil, false)
|
||||||
expect(t, `out = immutable({a:1,b:2}) == immutable({a:1,b:3})`, false)
|
expect(t, `out = immutable({a:1,b:2}) == immutable({a:1,b:3})`, nil, false)
|
||||||
expect(t, `out = {a:1,b:2} == immutable({a:1,b:3})`, false)
|
expect(t, `out = {a:1,b:2} == immutable({a:1,b:3})`, nil, false)
|
||||||
expect(t, `out = immutable({a:1,b:2}).b`, 2)
|
expect(t, `out = immutable({a:1,b:2}).b`, nil, 2)
|
||||||
expect(t, `out = immutable({a:1,b:2})["b"]`, 2)
|
expect(t, `out = immutable({a:1,b:2})["b"]`, nil, 2)
|
||||||
expect(t, `a := immutable({a:1,b:2}); a = 5; out = 5`, 5)
|
expect(t, `a := immutable({a:1,b:2}); a = 5; out = 5`, nil, 5)
|
||||||
expect(t, `a := immutable({a:1,b:2}); out = a.c`, objects.UndefinedValue)
|
expect(t, `a := immutable({a:1,b:2}); out = a.c`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `a := immutable({b: 5, c: "foo"}); out = a.b`, 5)
|
expect(t, `a := immutable({b: 5, c: "foo"}); out = a.b`, nil, 5)
|
||||||
expectError(t, `a := immutable({b: 5, c: "foo"}); a.b = 10`, "not index-assignable")
|
expectError(t, `a := immutable({b: 5, c: "foo"}); a.b = 10`, nil, "not index-assignable")
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,18 +5,17 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestIncDec(t *testing.T) {
|
func TestIncDec(t *testing.T) {
|
||||||
expect(t, `out = 0; out++`, 1)
|
expect(t, `out = 0; out++`, nil, 1)
|
||||||
expect(t, `out = 0; out--`, -1)
|
expect(t, `out = 0; out--`, nil, -1)
|
||||||
expect(t, `a := 0; a++; out = a`, 1)
|
expect(t, `a := 0; a++; out = a`, nil, 1)
|
||||||
expect(t, `a := 0; a++; a--; out = a`, 0)
|
expect(t, `a := 0; a++; a--; out = a`, nil, 0)
|
||||||
|
|
||||||
// this seems strange but it works because 'a += b' is
|
// this seems strange but it works because 'a += b' is
|
||||||
// translated into 'a = a + b' and string type takes other types for + operator.
|
// translated into 'a = a + b' and string type takes other types for + operator.
|
||||||
expect(t, `a := "foo"; a++; out = a`, "foo1")
|
expect(t, `a := "foo"; a++; out = a`, nil, "foo1")
|
||||||
expectError(t, `a := "foo"; a--`, "invalid operation")
|
expectError(t, `a := "foo"; a--`, nil, "invalid operation")
|
||||||
|
|
||||||
expectError(t, `a++`, "unresolved reference") // not declared
|
expectError(t, `a++`, nil, "unresolved reference") // not declared
|
||||||
expectError(t, `a--`, "unresolved reference") // not declared
|
expectError(t, `a--`, nil, "unresolved reference") // not declared
|
||||||
//expectError(t, `a := 0; b := a++`) // inc-dec is statement not expression <- parser error
|
expectError(t, `4++`, nil, "unresolved reference")
|
||||||
expectError(t, `4++`, "unresolved reference")
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -225,44 +225,44 @@ func (o *StringArray) Call(args ...objects.Object) (ret objects.Object, err erro
|
||||||
|
|
||||||
func TestIndexable(t *testing.T) {
|
func TestIndexable(t *testing.T) {
|
||||||
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
||||||
expectWithSymbols(t, `out = dict["a"]`, "foo", SYM{"dict": dict()})
|
expect(t, `out = dict["a"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "foo")
|
||||||
expectWithSymbols(t, `out = dict["B"]`, "bar", SYM{"dict": dict()})
|
expect(t, `out = dict["B"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "bar")
|
||||||
expectWithSymbols(t, `out = dict["x"]`, objects.UndefinedValue, SYM{"dict": dict()})
|
expect(t, `out = dict["x"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), objects.UndefinedValue)
|
||||||
expectErrorWithSymbols(t, `dict[0]`, SYM{"dict": dict()}, "invalid index type")
|
expectError(t, `dict[0]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "invalid index type")
|
||||||
|
|
||||||
strCir := func() *StringCircle { return &StringCircle{Value: []string{"one", "two", "three"}} }
|
strCir := func() *StringCircle { return &StringCircle{Value: []string{"one", "two", "three"}} }
|
||||||
expectWithSymbols(t, `out = cir[0]`, "one", SYM{"cir": strCir()})
|
expect(t, `out = cir[0]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "one")
|
||||||
expectWithSymbols(t, `out = cir[1]`, "two", SYM{"cir": strCir()})
|
expect(t, `out = cir[1]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "two")
|
||||||
expectWithSymbols(t, `out = cir[-1]`, "three", SYM{"cir": strCir()})
|
expect(t, `out = cir[-1]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "three")
|
||||||
expectWithSymbols(t, `out = cir[-2]`, "two", SYM{"cir": strCir()})
|
expect(t, `out = cir[-2]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "two")
|
||||||
expectWithSymbols(t, `out = cir[3]`, "one", SYM{"cir": strCir()})
|
expect(t, `out = cir[3]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "one")
|
||||||
expectErrorWithSymbols(t, `cir["a"]`, SYM{"cir": strCir()}, "invalid index type")
|
expectError(t, `cir["a"]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "invalid index type")
|
||||||
|
|
||||||
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
||||||
expectWithSymbols(t, `out = arr["one"]`, 0, SYM{"arr": strArr()})
|
expect(t, `out = arr["one"]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), 0)
|
||||||
expectWithSymbols(t, `out = arr["three"]`, 2, SYM{"arr": strArr()})
|
expect(t, `out = arr["three"]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), 2)
|
||||||
expectWithSymbols(t, `out = arr["four"]`, objects.UndefinedValue, SYM{"arr": strArr()})
|
expect(t, `out = arr["four"]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), objects.UndefinedValue)
|
||||||
expectWithSymbols(t, `out = arr[0]`, "one", SYM{"arr": strArr()})
|
expect(t, `out = arr[0]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "one")
|
||||||
expectWithSymbols(t, `out = arr[1]`, "two", SYM{"arr": strArr()})
|
expect(t, `out = arr[1]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "two")
|
||||||
expectErrorWithSymbols(t, `arr[-1]`, SYM{"arr": strArr()}, "index out of bounds")
|
expectError(t, `arr[-1]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "index out of bounds")
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestIndexAssignable(t *testing.T) {
|
func TestIndexAssignable(t *testing.T) {
|
||||||
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
||||||
expectWithSymbols(t, `dict["a"] = "1984"; out = dict["a"]`, "1984", SYM{"dict": dict()})
|
expect(t, `dict["a"] = "1984"; out = dict["a"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "1984")
|
||||||
expectWithSymbols(t, `dict["c"] = "1984"; out = dict["c"]`, "1984", SYM{"dict": dict()})
|
expect(t, `dict["c"] = "1984"; out = dict["c"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "1984")
|
||||||
expectWithSymbols(t, `dict["c"] = 1984; out = dict["C"]`, "1984", SYM{"dict": dict()})
|
expect(t, `dict["c"] = 1984; out = dict["C"]`, Opts().Symbol("dict", dict()).Skip2ndPass(), "1984")
|
||||||
expectErrorWithSymbols(t, `dict[0] = "1984"`, SYM{"dict": dict()}, "invalid index type")
|
expectError(t, `dict[0] = "1984"`, Opts().Symbol("dict", dict()).Skip2ndPass(), "invalid index type")
|
||||||
|
|
||||||
strCir := func() *StringCircle { return &StringCircle{Value: []string{"one", "two", "three"}} }
|
strCir := func() *StringCircle { return &StringCircle{Value: []string{"one", "two", "three"}} }
|
||||||
expectWithSymbols(t, `cir[0] = "ONE"; out = cir[0]`, "ONE", SYM{"cir": strCir()})
|
expect(t, `cir[0] = "ONE"; out = cir[0]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "ONE")
|
||||||
expectWithSymbols(t, `cir[1] = "TWO"; out = cir[1]`, "TWO", SYM{"cir": strCir()})
|
expect(t, `cir[1] = "TWO"; out = cir[1]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "TWO")
|
||||||
expectWithSymbols(t, `cir[-1] = "THREE"; out = cir[2]`, "THREE", SYM{"cir": strCir()})
|
expect(t, `cir[-1] = "THREE"; out = cir[2]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "THREE")
|
||||||
expectWithSymbols(t, `cir[0] = "ONE"; out = cir[3]`, "ONE", SYM{"cir": strCir()})
|
expect(t, `cir[0] = "ONE"; out = cir[3]`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "ONE")
|
||||||
expectErrorWithSymbols(t, `cir["a"] = "ONE"`, SYM{"cir": strCir()}, "invalid index type")
|
expectError(t, `cir["a"] = "ONE"`, Opts().Symbol("cir", strCir()).Skip2ndPass(), "invalid index type")
|
||||||
|
|
||||||
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
||||||
expectWithSymbols(t, `arr[0] = "ONE"; out = arr[0]`, "ONE", SYM{"arr": strArr()})
|
expect(t, `arr[0] = "ONE"; out = arr[0]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "ONE")
|
||||||
expectWithSymbols(t, `arr[1] = "TWO"; out = arr[1]`, "TWO", SYM{"arr": strArr()})
|
expect(t, `arr[1] = "TWO"; out = arr[1]`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "TWO")
|
||||||
expectErrorWithSymbols(t, `arr["one"] = "ONE"`, SYM{"arr": strArr()}, "invalid index type")
|
expectError(t, `arr["one"] = "ONE"`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "invalid index type")
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,26 +5,26 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestInteger(t *testing.T) {
|
func TestInteger(t *testing.T) {
|
||||||
expect(t, `out = 5`, 5)
|
expect(t, `out = 5`, nil, 5)
|
||||||
expect(t, `out = 10`, 10)
|
expect(t, `out = 10`, nil, 10)
|
||||||
expect(t, `out = -5`, -5)
|
expect(t, `out = -5`, nil, -5)
|
||||||
expect(t, `out = -10`, -10)
|
expect(t, `out = -10`, nil, -10)
|
||||||
expect(t, `out = 5 + 5 + 5 + 5 - 10`, 10)
|
expect(t, `out = 5 + 5 + 5 + 5 - 10`, nil, 10)
|
||||||
expect(t, `out = 2 * 2 * 2 * 2 * 2`, 32)
|
expect(t, `out = 2 * 2 * 2 * 2 * 2`, nil, 32)
|
||||||
expect(t, `out = -50 + 100 + -50`, 0)
|
expect(t, `out = -50 + 100 + -50`, nil, 0)
|
||||||
expect(t, `out = 5 * 2 + 10`, 20)
|
expect(t, `out = 5 * 2 + 10`, nil, 20)
|
||||||
expect(t, `out = 5 + 2 * 10`, 25)
|
expect(t, `out = 5 + 2 * 10`, nil, 25)
|
||||||
expect(t, `out = 20 + 2 * -10`, 0)
|
expect(t, `out = 20 + 2 * -10`, nil, 0)
|
||||||
expect(t, `out = 50 / 2 * 2 + 10`, 60)
|
expect(t, `out = 50 / 2 * 2 + 10`, nil, 60)
|
||||||
expect(t, `out = 2 * (5 + 10)`, 30)
|
expect(t, `out = 2 * (5 + 10)`, nil, 30)
|
||||||
expect(t, `out = 3 * 3 * 3 + 10`, 37)
|
expect(t, `out = 3 * 3 * 3 + 10`, nil, 37)
|
||||||
expect(t, `out = 3 * (3 * 3) + 10`, 37)
|
expect(t, `out = 3 * (3 * 3) + 10`, nil, 37)
|
||||||
expect(t, `out = (5 + 10 * 2 + 15 /3) * 2 + -10`, 50)
|
expect(t, `out = (5 + 10 * 2 + 15 /3) * 2 + -10`, nil, 50)
|
||||||
expect(t, `out = 5 % 3`, 2)
|
expect(t, `out = 5 % 3`, nil, 2)
|
||||||
expect(t, `out = 5 % 3 + 4`, 6)
|
expect(t, `out = 5 % 3 + 4`, nil, 6)
|
||||||
expect(t, `out = +5`, 5)
|
expect(t, `out = +5`, nil, 5)
|
||||||
expect(t, `out = +5 + -5`, 0)
|
expect(t, `out = +5 + -5`, nil, 0)
|
||||||
|
|
||||||
expect(t, `out = 9 + '0'`, '9')
|
expect(t, `out = 9 + '0'`, nil, '9')
|
||||||
expect(t, `out = '9' - 5`, '4')
|
expect(t, `out = '9' - 5`, nil, '4')
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,7 +38,7 @@ func (o *StringArray) Iterate() objects.Iterator {
|
||||||
func TestIterable(t *testing.T) {
|
func TestIterable(t *testing.T) {
|
||||||
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
strArr := func() *StringArray { return &StringArray{Value: []string{"one", "two", "three"}} }
|
||||||
|
|
||||||
expectWithSymbols(t, `for i, s in arr { out += i }`, 3, SYM{"arr": strArr()})
|
expect(t, `for i, s in arr { out += i }`, Opts().Symbol("arr", strArr()).Skip2ndPass(), 3)
|
||||||
expectWithSymbols(t, `for i, s in arr { out += s }`, "onetwothree", SYM{"arr": strArr()})
|
expect(t, `for i, s in arr { out += s }`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "onetwothree")
|
||||||
expectWithSymbols(t, `for i, s in arr { out += s + i }`, "one0two1three2", SYM{"arr": strArr()})
|
expect(t, `for i, s in arr { out += s + i }`, Opts().Symbol("arr", strArr()).Skip2ndPass(), "one0two1three2")
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,39 +3,39 @@ package runtime_test
|
||||||
import "testing"
|
import "testing"
|
||||||
|
|
||||||
func TestLogical(t *testing.T) {
|
func TestLogical(t *testing.T) {
|
||||||
expect(t, `out = true && true`, true)
|
expect(t, `out = true && true`, nil, true)
|
||||||
expect(t, `out = true && false`, false)
|
expect(t, `out = true && false`, nil, false)
|
||||||
expect(t, `out = false && true`, false)
|
expect(t, `out = false && true`, nil, false)
|
||||||
expect(t, `out = false && false`, false)
|
expect(t, `out = false && false`, nil, false)
|
||||||
expect(t, `out = !true && true`, false)
|
expect(t, `out = !true && true`, nil, false)
|
||||||
expect(t, `out = !true && false`, false)
|
expect(t, `out = !true && false`, nil, false)
|
||||||
expect(t, `out = !false && true`, true)
|
expect(t, `out = !false && true`, nil, true)
|
||||||
expect(t, `out = !false && false`, false)
|
expect(t, `out = !false && false`, nil, false)
|
||||||
|
|
||||||
expect(t, `out = true || true`, true)
|
expect(t, `out = true || true`, nil, true)
|
||||||
expect(t, `out = true || false`, true)
|
expect(t, `out = true || false`, nil, true)
|
||||||
expect(t, `out = false || true`, true)
|
expect(t, `out = false || true`, nil, true)
|
||||||
expect(t, `out = false || false`, false)
|
expect(t, `out = false || false`, nil, false)
|
||||||
expect(t, `out = !true || true`, true)
|
expect(t, `out = !true || true`, nil, true)
|
||||||
expect(t, `out = !true || false`, false)
|
expect(t, `out = !true || false`, nil, false)
|
||||||
expect(t, `out = !false || true`, true)
|
expect(t, `out = !false || true`, nil, true)
|
||||||
expect(t, `out = !false || false`, true)
|
expect(t, `out = !false || false`, nil, true)
|
||||||
|
|
||||||
expect(t, `out = 1 && 2`, 2)
|
expect(t, `out = 1 && 2`, nil, 2)
|
||||||
expect(t, `out = 1 || 2`, 1)
|
expect(t, `out = 1 || 2`, nil, 1)
|
||||||
expect(t, `out = 1 && 0`, 0)
|
expect(t, `out = 1 && 0`, nil, 0)
|
||||||
expect(t, `out = 1 || 0`, 1)
|
expect(t, `out = 1 || 0`, nil, 1)
|
||||||
expect(t, `out = 1 && (0 || 2)`, 2)
|
expect(t, `out = 1 && (0 || 2)`, nil, 2)
|
||||||
expect(t, `out = 0 || (0 || 2)`, 2)
|
expect(t, `out = 0 || (0 || 2)`, nil, 2)
|
||||||
expect(t, `out = 0 || (0 && 2)`, 0)
|
expect(t, `out = 0 || (0 && 2)`, nil, 0)
|
||||||
expect(t, `out = 0 || (2 && 0)`, 0)
|
expect(t, `out = 0 || (2 && 0)`, nil, 0)
|
||||||
|
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; t() && f()`, 7)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; t() && f()`, nil, 7)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; f() && t()`, 7)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; f() && t()`, nil, 7)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; f() || t()`, 3)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; f() || t()`, nil, 3)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; t() || f()`, 3)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; t() || f()`, nil, 3)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !t() && f()`, 3)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !t() && f()`, nil, 3)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !f() && t()`, 3)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !f() && t()`, nil, 3)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !f() || t()`, 7)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !f() || t()`, nil, 7)
|
||||||
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !t() || f()`, 7)
|
expect(t, `t:=func() {out = 3; return true}; f:=func() {out = 7; return false}; !t() || f()`, nil, 7)
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ out = {
|
||||||
one: 10 - 9,
|
one: 10 - 9,
|
||||||
two: 1 + 1,
|
two: 1 + 1,
|
||||||
three: 6 / 2
|
three: 6 / 2
|
||||||
}`, MAP{
|
}`, nil, MAP{
|
||||||
"one": 1,
|
"one": 1,
|
||||||
"two": 2,
|
"two": 2,
|
||||||
"three": 3,
|
"three": 3,
|
||||||
|
@ -23,16 +23,16 @@ out = {
|
||||||
"one": 10 - 9,
|
"one": 10 - 9,
|
||||||
"two": 1 + 1,
|
"two": 1 + 1,
|
||||||
"three": 6 / 2
|
"three": 6 / 2
|
||||||
}`, MAP{
|
}`, nil, MAP{
|
||||||
"one": 1,
|
"one": 1,
|
||||||
"two": 2,
|
"two": 2,
|
||||||
"three": 3,
|
"three": 3,
|
||||||
})
|
})
|
||||||
|
|
||||||
expect(t, `out = {foo: 5}["foo"]`, 5)
|
expect(t, `out = {foo: 5}["foo"]`, nil, 5)
|
||||||
expect(t, `out = {foo: 5}["bar"]`, objects.UndefinedValue)
|
expect(t, `out = {foo: 5}["bar"]`, nil, objects.UndefinedValue)
|
||||||
expect(t, `key := "foo"; out = {foo: 5}[key]`, 5)
|
expect(t, `key := "foo"; out = {foo: 5}[key]`, nil, 5)
|
||||||
expect(t, `out = {}["foo"]`, objects.UndefinedValue)
|
expect(t, `out = {}["foo"]`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
m := {
|
m := {
|
||||||
|
@ -41,11 +41,11 @@ m := {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out = m["foo"](2) + m["foo"](3)
|
out = m["foo"](2) + m["foo"](3)
|
||||||
`, 10)
|
`, nil, 10)
|
||||||
|
|
||||||
// map assignment is copy-by-reference
|
// map assignment is copy-by-reference
|
||||||
expect(t, `m1 := {k1: 1, k2: "foo"}; m2 := m1; m1.k1 = 5; out = m2.k1`, 5)
|
expect(t, `m1 := {k1: 1, k2: "foo"}; m2 := m1; m1.k1 = 5; out = m2.k1`, nil, 5)
|
||||||
expect(t, `m1 := {k1: 1, k2: "foo"}; m2 := m1; m2.k1 = 3; out = m1.k1`, 3)
|
expect(t, `m1 := {k1: 1, k2: "foo"}; m2 := m1; m2.k1 = 3; out = m1.k1`, nil, 3)
|
||||||
expect(t, `func() { m1 := {k1: 1, k2: "foo"}; m2 := m1; m1.k1 = 5; out = m2.k1 }()`, 5)
|
expect(t, `func() { m1 := {k1: 1, k2: "foo"}; m2 := m1; m1.k1 = 5; out = m2.k1 }()`, nil, 5)
|
||||||
expect(t, `func() { m1 := {k1: 1, k2: "foo"}; m2 := m1; m2.k1 = 3; out = m1.k1 }()`, 3)
|
expect(t, `func() { m1 := {k1: 1, k2: "foo"}; m2 := m1; m2.k1 = 3; out = m1.k1 }()`, nil, 3)
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,89 +9,66 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestBuiltin(t *testing.T) {
|
func TestBuiltin(t *testing.T) {
|
||||||
|
m := Opts().Module("math",
|
||||||
mathModule := map[string]objects.Object{
|
&objects.BuiltinModule{
|
||||||
"math": &objects.ImmutableMap{Value: map[string]objects.Object{
|
Attrs: map[string]objects.Object{
|
||||||
"abs": &objects.UserFunction{Name: "abs", Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
"abs": &objects.UserFunction{
|
||||||
|
Name: "abs",
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
v, _ := objects.ToFloat64(args[0])
|
v, _ := objects.ToFloat64(args[0])
|
||||||
return &objects.Float{Value: math.Abs(v)}, nil
|
return &objects.Float{Value: math.Abs(v)}, nil
|
||||||
}},
|
},
|
||||||
}},
|
},
|
||||||
}
|
},
|
||||||
|
})
|
||||||
|
|
||||||
// builtin
|
// builtin
|
||||||
expectWithBuiltinModules(t, `math := import("math"); out = math.abs(1)`, 1.0, mathModule)
|
expect(t, `math := import("math"); out = math.abs(1)`, m, 1.0)
|
||||||
expectWithBuiltinModules(t, `math := import("math"); out = math.abs(-1)`, 1.0, mathModule)
|
expect(t, `math := import("math"); out = math.abs(-1)`, m, 1.0)
|
||||||
expectWithBuiltinModules(t, `math := import("math"); out = math.abs(1.0)`, 1.0, mathModule)
|
expect(t, `math := import("math"); out = math.abs(1.0)`, m, 1.0)
|
||||||
expectWithBuiltinModules(t, `math := import("math"); out = math.abs(-1.0)`, 1.0, mathModule)
|
expect(t, `math := import("math"); out = math.abs(-1.0)`, m, 1.0)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestUserModules(t *testing.T) {
|
func TestUserModules(t *testing.T) {
|
||||||
// user modules
|
|
||||||
|
|
||||||
// export none
|
// export none
|
||||||
expectWithUserModules(t, `out = import("mod1")`, objects.UndefinedValue, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `fn := func() { return 5.0 }; a := 2`), objects.UndefinedValue)
|
||||||
"mod1": `fn := func() { return 5.0 }; a := 2`,
|
|
||||||
})
|
|
||||||
|
|
||||||
// export values
|
// export values
|
||||||
expectWithUserModules(t, `out = import("mod1")`, 5, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `export 5`), 5)
|
||||||
"mod1": `export 5`,
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `export "foo"`), "foo")
|
||||||
})
|
|
||||||
expectWithUserModules(t, `out = import("mod1")`, "foo", map[string]string{
|
|
||||||
"mod1": `export "foo"`,
|
|
||||||
})
|
|
||||||
|
|
||||||
// export compound types
|
// export compound types
|
||||||
expectWithUserModules(t, `out = import("mod1")`, IARR{1, 2, 3}, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `export [1, 2, 3]`), IARR{1, 2, 3})
|
||||||
"mod1": `export [1, 2, 3]`,
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `export {a: 1, b: 2}`), IMAP{"a": 1, "b": 2})
|
||||||
})
|
|
||||||
expectWithUserModules(t, `out = import("mod1")`, IMAP{"a": 1, "b": 2}, map[string]string{
|
|
||||||
"mod1": `export {a: 1, b: 2}`,
|
|
||||||
})
|
|
||||||
// export value is immutable
|
// export value is immutable
|
||||||
expectErrorWithUserModules(t, `m1 := import("mod1"); m1.a = 5`, map[string]string{
|
expectError(t, `m1 := import("mod1"); m1.a = 5`, Opts().Module("mod1", `export {a: 1, b: 2}`), "not index-assignable")
|
||||||
"mod1": `export {a: 1, b: 2}`,
|
expectError(t, `m1 := import("mod1"); m1[1] = 5`, Opts().Module("mod1", `export [1, 2, 3]`), "not index-assignable")
|
||||||
}, "not index-assignable")
|
|
||||||
expectErrorWithUserModules(t, `m1 := import("mod1"); m1[1] = 5`, map[string]string{
|
|
||||||
"mod1": `export [1, 2, 3]`,
|
|
||||||
}, "not index-assignable")
|
|
||||||
|
|
||||||
// code after export statement will not be executed
|
// code after export statement will not be executed
|
||||||
expectWithUserModules(t, `out = import("mod1")`, 10, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `a := 10; export a; a = 20`), 10)
|
||||||
"mod1": `a := 10; export a; a = 20`,
|
expect(t, `out = import("mod1")`, Opts().Module("mod1", `a := 10; export a; a = 20; export a`), 10)
|
||||||
})
|
|
||||||
expectWithUserModules(t, `out = import("mod1")`, 10, map[string]string{
|
|
||||||
"mod1": `a := 10; export a; a = 20; export a`,
|
|
||||||
})
|
|
||||||
|
|
||||||
// export function
|
// export function
|
||||||
expectWithUserModules(t, `out = import("mod1")()`, 5.0, map[string]string{
|
expect(t, `out = import("mod1")()`, Opts().Module("mod1", `export func() { return 5.0 }`), 5.0)
|
||||||
"mod1": `export func() { return 5.0 }`,
|
|
||||||
})
|
|
||||||
// export function that reads module-global variable
|
// export function that reads module-global variable
|
||||||
expectWithUserModules(t, `out = import("mod1")()`, 6.5, map[string]string{
|
expect(t, `out = import("mod1")()`, Opts().Module("mod1", `a := 1.5; export func() { return a + 5.0 }`), 6.5)
|
||||||
"mod1": `a := 1.5; export func() { return a + 5.0 }`,
|
|
||||||
})
|
|
||||||
// export function that read local variable
|
// export function that read local variable
|
||||||
expectWithUserModules(t, `out = import("mod1")()`, 6.5, map[string]string{
|
expect(t, `out = import("mod1")()`, Opts().Module("mod1", `export func() { a := 1.5; return a + 5.0 }`), 6.5)
|
||||||
"mod1": `export func() { a := 1.5; return a + 5.0 }`,
|
|
||||||
})
|
|
||||||
// export function that read free variables
|
// export function that read free variables
|
||||||
expectWithUserModules(t, `out = import("mod1")()`, 6.5, map[string]string{
|
expect(t, `out = import("mod1")()`, Opts().Module("mod1", `export func() { a := 1.5; return func() { return a + 5.0 }() }`), 6.5)
|
||||||
"mod1": `export func() { a := 1.5; return func() { return a + 5.0 }() }`,
|
|
||||||
})
|
|
||||||
|
|
||||||
// recursive function in module
|
// recursive function in module
|
||||||
expectWithUserModules(t, `out = import("mod1")`, 15, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module(
|
||||||
"mod1": `
|
"mod1", `
|
||||||
a := func(x) {
|
a := func(x) {
|
||||||
return x == 0 ? 0 : x + a(x-1)
|
return x == 0 ? 0 : x + a(x-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
export a(5)
|
export a(5)
|
||||||
`})
|
`), 15)
|
||||||
expectWithUserModules(t, `out = import("mod1")`, 15, map[string]string{
|
expect(t, `out = import("mod1")`, Opts().Module(
|
||||||
"mod1": `
|
"mod1", `
|
||||||
export func() {
|
export func() {
|
||||||
a := func(x) {
|
a := func(x) {
|
||||||
return x == 0 ? 0 : x + a(x-1)
|
return x == 0 ? 0 : x + a(x-1)
|
||||||
|
@ -99,117 +76,114 @@ export func() {
|
||||||
|
|
||||||
return a(5)
|
return a(5)
|
||||||
}()
|
}()
|
||||||
`})
|
`), 15)
|
||||||
|
|
||||||
// (main) -> mod1 -> mod2
|
// (main) -> mod1 -> mod2
|
||||||
expectWithUserModules(t, `out = import("mod1")()`, 5.0, map[string]string{
|
expect(t, `out = import("mod1")()`,
|
||||||
"mod1": `export import("mod2")`,
|
Opts().Module("mod1", `export import("mod2")`).
|
||||||
"mod2": `export func() { return 5.0 }`,
|
Module("mod2", `export func() { return 5.0 }`),
|
||||||
})
|
5.0)
|
||||||
// (main) -> mod1 -> mod2
|
// (main) -> mod1 -> mod2
|
||||||
// -> mod2
|
// -> mod2
|
||||||
expectWithUserModules(t, `import("mod1"); out = import("mod2")()`, 5.0, map[string]string{
|
expect(t, `import("mod1"); out = import("mod2")()`,
|
||||||
"mod1": `export import("mod2")`,
|
Opts().Module("mod1", `export import("mod2")`).
|
||||||
"mod2": `export func() { return 5.0 }`,
|
Module("mod2", `export func() { return 5.0 }`),
|
||||||
})
|
5.0)
|
||||||
// (main) -> mod1 -> mod2 -> mod3
|
// (main) -> mod1 -> mod2 -> mod3
|
||||||
// -> mod2 -> mod3
|
// -> mod2 -> mod3
|
||||||
expectWithUserModules(t, `import("mod1"); out = import("mod2")()`, 5.0, map[string]string{
|
expect(t, `import("mod1"); out = import("mod2")()`,
|
||||||
"mod1": `export import("mod2")`,
|
Opts().Module("mod1", `export import("mod2")`).
|
||||||
"mod2": `export import("mod3")`,
|
Module("mod2", `export import("mod3")`).
|
||||||
"mod3": `export func() { return 5.0 }`,
|
Module("mod3", `export func() { return 5.0 }`),
|
||||||
})
|
5.0)
|
||||||
|
|
||||||
// cyclic imports
|
// cyclic imports
|
||||||
// (main) -> mod1 -> mod2 -> mod1
|
// (main) -> mod1 -> mod2 -> mod1
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
expectError(t, `import("mod1")`,
|
||||||
"mod1": `import("mod2")`,
|
Opts().Module("mod1", `import("mod2")`).
|
||||||
"mod2": `import("mod1")`,
|
Module("mod2", `import("mod1")`),
|
||||||
}, "Compile Error: cyclic module import: mod1\n\tat mod2:1:1")
|
"Compile Error: cyclic module import: mod1\n\tat mod2:1:1")
|
||||||
// (main) -> mod1 -> mod2 -> mod3 -> mod1
|
// (main) -> mod1 -> mod2 -> mod3 -> mod1
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
expectError(t, `import("mod1")`,
|
||||||
"mod1": `import("mod2")`,
|
Opts().Module("mod1", `import("mod2")`).
|
||||||
"mod2": `import("mod3")`,
|
Module("mod2", `import("mod3")`).
|
||||||
"mod3": `import("mod1")`,
|
Module("mod3", `import("mod1")`),
|
||||||
}, "Compile Error: cyclic module import: mod1\n\tat mod3:1:1")
|
"Compile Error: cyclic module import: mod1\n\tat mod3:1:1")
|
||||||
// (main) -> mod1 -> mod2 -> mod3 -> mod2
|
// (main) -> mod1 -> mod2 -> mod3 -> mod2
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
expectError(t, `import("mod1")`,
|
||||||
"mod1": `import("mod2")`,
|
Opts().Module("mod1", `import("mod2")`).
|
||||||
"mod2": `import("mod3")`,
|
Module("mod2", `import("mod3")`).
|
||||||
"mod3": `import("mod2")`,
|
Module("mod3", `import("mod2")`),
|
||||||
}, "Compile Error: cyclic module import: mod2\n\tat mod3:1:1")
|
"Compile Error: cyclic module import: mod2\n\tat mod3:1:1")
|
||||||
|
|
||||||
// unknown modules
|
// unknown modules
|
||||||
expectErrorWithUserModules(t, `import("mod0")`, map[string]string{
|
expectError(t, `import("mod0")`, Opts().Module("mod1", `a := 5`), "module 'mod0' not found")
|
||||||
"mod1": `a := 5`,
|
expectError(t, `import("mod1")`, Opts().Module("mod1", `import("mod2")`), "module 'mod2' not found")
|
||||||
}, "module 'mod0' not found")
|
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
|
||||||
"mod1": `import("mod2")`,
|
|
||||||
}, "module 'mod2' not found")
|
|
||||||
|
|
||||||
// module is immutable but its variables is not necessarily immutable.
|
// module is immutable but its variables is not necessarily immutable.
|
||||||
expectWithUserModules(t, `m1 := import("mod1"); m1.a.b = 5; out = m1.a.b`, 5, map[string]string{
|
expect(t, `m1 := import("mod1"); m1.a.b = 5; out = m1.a.b`,
|
||||||
"mod1": `export {a: {b: 3}}`,
|
Opts().Module("mod1", `export {a: {b: 3}}`),
|
||||||
})
|
5)
|
||||||
|
|
||||||
// make sure module has same builtin functions
|
// make sure module has same builtin functions
|
||||||
expectWithUserModules(t, `out = import("mod1")`, "int", map[string]string{
|
expect(t, `out = import("mod1")`,
|
||||||
"mod1": `export func() { return type_name(0) }()`,
|
Opts().Module("mod1", `export func() { return type_name(0) }()`),
|
||||||
})
|
"int")
|
||||||
|
|
||||||
// 'export' statement is ignored outside module
|
// 'export' statement is ignored outside module
|
||||||
expectNoMod(t, `a := 5; export func() { a = 10 }(); out = a`, 5)
|
expect(t, `a := 5; export func() { a = 10 }(); out = a`, Opts().Skip2ndPass(), 5)
|
||||||
|
|
||||||
// 'export' must be in the top-level
|
// 'export' must be in the top-level
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
expectError(t, `import("mod1")`,
|
||||||
"mod1": `func() { export 5 }()`,
|
Opts().Module("mod1", `func() { export 5 }()`),
|
||||||
}, "Compile Error: export not allowed inside function\n\tat mod1:1:10")
|
"Compile Error: export not allowed inside function\n\tat mod1:1:10")
|
||||||
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
expectError(t, `import("mod1")`,
|
||||||
"mod1": `func() { func() { export 5 }() }()`,
|
Opts().Module("mod1", `func() { func() { export 5 }() }()`),
|
||||||
}, "Compile Error: export not allowed inside function\n\tat mod1:1:19")
|
"Compile Error: export not allowed inside function\n\tat mod1:1:19")
|
||||||
|
|
||||||
// module cannot access outer scope
|
// module cannot access outer scope
|
||||||
expectErrorWithUserModules(t, `a := 5; import("mod1")`, map[string]string{
|
expectError(t, `a := 5; import("mod1")`,
|
||||||
"mod1": `export a`,
|
Opts().Module("mod1", `export a`),
|
||||||
}, "Compile Error: unresolved reference 'a'\n\tat mod1:1:8")
|
"Compile Error: unresolved reference 'a'\n\tat mod1:1:8")
|
||||||
|
|
||||||
// runtime error within modules
|
// runtime error within modules
|
||||||
expectErrorWithUserModules(t, `
|
expectError(t, `
|
||||||
a := 1;
|
a := 1;
|
||||||
b := import("mod1");
|
b := import("mod1");
|
||||||
b(a)`,
|
b(a)`,
|
||||||
map[string]string{"mod1": `
|
Opts().Module("mod1", `
|
||||||
export func(a) {
|
export func(a) {
|
||||||
a()
|
a()
|
||||||
}
|
}
|
||||||
`,
|
`), "Runtime Error: not callable: int\n\tat mod1:3:4\n\tat test:4:1")
|
||||||
}, "Runtime Error: not callable: int\n\tat mod1:3:4\n\tat test:4:1")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestModuleBlockScopes(t *testing.T) {
|
func TestModuleBlockScopes(t *testing.T) {
|
||||||
randModule := map[string]objects.Object{
|
m := Opts().Module("rand",
|
||||||
"rand": &objects.ImmutableMap{Value: map[string]objects.Object{
|
&objects.BuiltinModule{
|
||||||
"intn": &objects.UserFunction{Name: "abs", Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
Attrs: map[string]objects.Object{
|
||||||
|
"intn": &objects.UserFunction{
|
||||||
|
Name: "abs",
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
v, _ := objects.ToInt64(args[0])
|
v, _ := objects.ToInt64(args[0])
|
||||||
return &objects.Int{Value: rand.Int63n(v)}, nil
|
return &objects.Int{Value: rand.Int63n(v)}, nil
|
||||||
}},
|
},
|
||||||
}},
|
},
|
||||||
}
|
},
|
||||||
|
})
|
||||||
|
|
||||||
// block scopes in module
|
// block scopes in module
|
||||||
expectWithUserAndBuiltinModules(t, `out = import("mod1")()`, 1, map[string]string{
|
expect(t, `out = import("mod1")()`, m.Module(
|
||||||
"mod1": `
|
"mod1", `
|
||||||
rand := import("rand")
|
rand := import("rand")
|
||||||
foo := func() { return 1 }
|
foo := func() { return 1 }
|
||||||
export func() {
|
export func() {
|
||||||
rand.intn(3)
|
rand.intn(3)
|
||||||
return foo()
|
return foo()
|
||||||
}
|
}`), 1)
|
||||||
`,
|
|
||||||
}, randModule)
|
|
||||||
|
|
||||||
expectWithUserAndBuiltinModules(t, `out = import("mod1")()`, 10, map[string]string{
|
expect(t, `out = import("mod1")()`, m.Module(
|
||||||
"mod1": `
|
"mod1", `
|
||||||
rand := import("rand")
|
rand := import("rand")
|
||||||
foo := func() { return 1 }
|
foo := func() { return 1 }
|
||||||
export func() {
|
export func() {
|
||||||
|
@ -217,11 +191,10 @@ export func() {
|
||||||
if foo() {}
|
if foo() {}
|
||||||
return 10
|
return 10
|
||||||
}
|
}
|
||||||
`,
|
`), 10)
|
||||||
}, randModule)
|
|
||||||
|
|
||||||
expectWithUserAndBuiltinModules(t, `out = import("mod1")()`, 10, map[string]string{
|
expect(t, `out = import("mod1")()`, m.Module(
|
||||||
"mod1": `
|
"mod1", `
|
||||||
rand := import("rand")
|
rand := import("rand")
|
||||||
foo := func() { return 1 }
|
foo := func() { return 1 }
|
||||||
export func() {
|
export func() {
|
||||||
|
@ -229,6 +202,5 @@ export func() {
|
||||||
if true { foo() }
|
if true { foo() }
|
||||||
return 10
|
return 10
|
||||||
}
|
}
|
||||||
`,
|
`), 10)
|
||||||
}, randModule)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,11 +5,11 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestBangOperator(t *testing.T) {
|
func TestBangOperator(t *testing.T) {
|
||||||
expect(t, `out = !true`, false)
|
expect(t, `out = !true`, nil, false)
|
||||||
expect(t, `out = !false`, true)
|
expect(t, `out = !false`, nil, true)
|
||||||
expect(t, `out = !0`, true)
|
expect(t, `out = !0`, nil, true)
|
||||||
expect(t, `out = !5`, false)
|
expect(t, `out = !5`, nil, false)
|
||||||
expect(t, `out = !!true`, true)
|
expect(t, `out = !!true`, nil, true)
|
||||||
expect(t, `out = !!false`, false)
|
expect(t, `out = !!false`, nil, false)
|
||||||
expect(t, `out = !!5`, true)
|
expect(t, `out = !!5`, nil, true)
|
||||||
}
|
}
|
||||||
|
|
|
@ -37,13 +37,13 @@ f()
|
||||||
}
|
}
|
||||||
|
|
||||||
func testAllocsLimit(t *testing.T, src string, limit int64) {
|
func testAllocsLimit(t *testing.T, src string, limit int64) {
|
||||||
expectAllocsLimit(t, src, -1, objects.UndefinedValue) // no limit
|
expect(t, src, Opts().Skip2ndPass(), objects.UndefinedValue) // no limit
|
||||||
expectAllocsLimit(t, src, limit, objects.UndefinedValue)
|
expect(t, src, Opts().MaxAllocs(limit).Skip2ndPass(), objects.UndefinedValue)
|
||||||
expectAllocsLimit(t, src, limit+1, objects.UndefinedValue)
|
expect(t, src, Opts().MaxAllocs(limit+1).Skip2ndPass(), objects.UndefinedValue)
|
||||||
if limit > 1 {
|
if limit > 1 {
|
||||||
expectErrorAllocsLimit(t, src, limit-1, "allocation limit exceeded")
|
expectError(t, src, Opts().MaxAllocs(limit-1).Skip2ndPass(), "allocation limit exceeded")
|
||||||
}
|
}
|
||||||
if limit > 2 {
|
if limit > 2 {
|
||||||
expectErrorAllocsLimit(t, src, limit-2, "allocation limit exceeded")
|
expectError(t, src, Opts().MaxAllocs(limit-2).Skip2ndPass(), "allocation limit exceeded")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,10 +5,10 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestReturn(t *testing.T) {
|
func TestReturn(t *testing.T) {
|
||||||
expect(t, `out = func() { return 10; }()`, 10)
|
expect(t, `out = func() { return 10; }()`, nil, 10)
|
||||||
expect(t, `out = func() { return 10; return 9; }()`, 10)
|
expect(t, `out = func() { return 10; return 9; }()`, nil, 10)
|
||||||
expect(t, `out = func() { return 2 * 5; return 9 }()`, 10)
|
expect(t, `out = func() { return 2 * 5; return 9 }()`, nil, 10)
|
||||||
expect(t, `out = func() { 9; return 2 * 5; return 9 }()`, 10)
|
expect(t, `out = func() { 9; return 2 * 5; return 9 }()`, nil, 10)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
out = func() {
|
out = func() {
|
||||||
if (10 > 1) {
|
if (10 > 1) {
|
||||||
|
@ -18,7 +18,7 @@ func TestReturn(t *testing.T) {
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}()`, 10)
|
}()`, nil, 10)
|
||||||
|
|
||||||
expect(t, `f1 := func() { return 2 * 5; }; out = f1()`, 10)
|
expect(t, `f1 := func() { return 2 * 5; }; out = f1()`, nil, 10)
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,9 +7,9 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestSelector(t *testing.T) {
|
func TestSelector(t *testing.T) {
|
||||||
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k1`, 5)
|
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k1`, nil, 5)
|
||||||
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k2`, "foo")
|
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k2`, nil, "foo")
|
||||||
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k3`, objects.UndefinedValue)
|
expect(t, `a := {k1: 5, k2: "foo"}; out = a.k3`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -19,7 +19,7 @@ a := {
|
||||||
},
|
},
|
||||||
c: "foo bar"
|
c: "foo bar"
|
||||||
}
|
}
|
||||||
out = a.b.c`, 4)
|
out = a.b.c`, nil, 4)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -29,7 +29,7 @@ a := {
|
||||||
},
|
},
|
||||||
c: "foo bar"
|
c: "foo bar"
|
||||||
}
|
}
|
||||||
b := a.x.c`, objects.UndefinedValue)
|
b := a.x.c`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -39,25 +39,25 @@ a := {
|
||||||
},
|
},
|
||||||
c: "foo bar"
|
c: "foo bar"
|
||||||
}
|
}
|
||||||
b := a.x.y`, objects.UndefinedValue)
|
b := a.x.y`, nil, objects.UndefinedValue)
|
||||||
|
|
||||||
expect(t, `a := {b: 1, c: "foo"}; a.b = 2; out = a.b`, 2)
|
expect(t, `a := {b: 1, c: "foo"}; a.b = 2; out = a.b`, nil, 2)
|
||||||
expect(t, `a := {b: 1, c: "foo"}; a.c = 2; out = a.c`, 2) // type not checked on sub-field
|
expect(t, `a := {b: 1, c: "foo"}; a.c = 2; out = a.c`, nil, 2) // type not checked on sub-field
|
||||||
expect(t, `a := {b: {c: 1}}; a.b.c = 2; out = a.b.c`, 2)
|
expect(t, `a := {b: {c: 1}}; a.b.c = 2; out = a.b.c`, nil, 2)
|
||||||
expect(t, `a := {b: 1}; a.c = 2; out = a`, MAP{"b": 1, "c": 2})
|
expect(t, `a := {b: 1}; a.c = 2; out = a`, nil, MAP{"b": 1, "c": 2})
|
||||||
expect(t, `a := {b: {c: 1}}; a.b.d = 2; out = a`, MAP{"b": MAP{"c": 1, "d": 2}})
|
expect(t, `a := {b: {c: 1}}; a.b.d = 2; out = a`, nil, MAP{"b": MAP{"c": 1, "d": 2}})
|
||||||
|
|
||||||
expect(t, `func() { a := {b: 1, c: "foo"}; a.b = 2; out = a.b }()`, 2)
|
expect(t, `func() { a := {b: 1, c: "foo"}; a.b = 2; out = a.b }()`, nil, 2)
|
||||||
expect(t, `func() { a := {b: 1, c: "foo"}; a.c = 2; out = a.c }()`, 2) // type not checked on sub-field
|
expect(t, `func() { a := {b: 1, c: "foo"}; a.c = 2; out = a.c }()`, nil, 2) // type not checked on sub-field
|
||||||
expect(t, `func() { a := {b: {c: 1}}; a.b.c = 2; out = a.b.c }()`, 2)
|
expect(t, `func() { a := {b: {c: 1}}; a.b.c = 2; out = a.b.c }()`, nil, 2)
|
||||||
expect(t, `func() { a := {b: 1}; a.c = 2; out = a }()`, MAP{"b": 1, "c": 2})
|
expect(t, `func() { a := {b: 1}; a.c = 2; out = a }()`, nil, MAP{"b": 1, "c": 2})
|
||||||
expect(t, `func() { a := {b: {c: 1}}; a.b.d = 2; out = a }()`, MAP{"b": MAP{"c": 1, "d": 2}})
|
expect(t, `func() { a := {b: {c: 1}}; a.b.d = 2; out = a }()`, nil, MAP{"b": MAP{"c": 1, "d": 2}})
|
||||||
|
|
||||||
expect(t, `func() { a := {b: 1, c: "foo"}; func() { a.b = 2 }(); out = a.b }()`, 2)
|
expect(t, `func() { a := {b: 1, c: "foo"}; func() { a.b = 2 }(); out = a.b }()`, nil, 2)
|
||||||
expect(t, `func() { a := {b: 1, c: "foo"}; func() { a.c = 2 }(); out = a.c }()`, 2) // type not checked on sub-field
|
expect(t, `func() { a := {b: 1, c: "foo"}; func() { a.c = 2 }(); out = a.c }()`, nil, 2) // type not checked on sub-field
|
||||||
expect(t, `func() { a := {b: {c: 1}}; func() { a.b.c = 2 }(); out = a.b.c }()`, 2)
|
expect(t, `func() { a := {b: {c: 1}}; func() { a.b.c = 2 }(); out = a.b.c }()`, nil, 2)
|
||||||
expect(t, `func() { a := {b: 1}; func() { a.c = 2 }(); out = a }()`, MAP{"b": 1, "c": 2})
|
expect(t, `func() { a := {b: 1}; func() { a.c = 2 }(); out = a }()`, nil, MAP{"b": 1, "c": 2})
|
||||||
expect(t, `func() { a := {b: {c: 1}}; func() { a.b.d = 2 }(); out = a }()`, MAP{"b": MAP{"c": 1, "d": 2}})
|
expect(t, `func() { a := {b: {c: 1}}; func() { a.b.d = 2 }(); out = a }()`, nil, MAP{"b": MAP{"c": 1, "d": 2}})
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
a := {
|
a := {
|
||||||
|
@ -69,7 +69,7 @@ a := {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
out = [a.b[2], a.c.d, a.c.e, a.c.f[1]]
|
out = [a.b[2], a.c.d, a.c.e, a.c.f[1]]
|
||||||
`, ARR{3, 8, "foo", 8})
|
`, nil, ARR{3, 8, "foo", 8})
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
func() {
|
func() {
|
||||||
|
@ -79,12 +79,12 @@ func() {
|
||||||
b = 7 // make sure a[1] has a COPY of value of 'b'
|
b = 7 // make sure a[1] has a COPY of value of 'b'
|
||||||
out = a[1]
|
out = a[1]
|
||||||
}()
|
}()
|
||||||
`, 9)
|
`, nil, 9)
|
||||||
|
|
||||||
expectError(t, `a := {b: {c: 1}}; a.d.c = 2`, "not index-assignable")
|
expectError(t, `a := {b: {c: 1}}; a.d.c = 2`, nil, "not index-assignable")
|
||||||
expectError(t, `a := [1, 2, 3]; a.b = 2`, "invalid index type")
|
expectError(t, `a := [1, 2, 3]; a.b = 2`, nil, "invalid index type")
|
||||||
expectError(t, `a := "foo"; a.b = 2`, "not index-assignable")
|
expectError(t, `a := "foo"; a.b = 2`, nil, "not index-assignable")
|
||||||
expectError(t, `func() { a := {b: {c: 1}}; a.d.c = 2 }()`, "not index-assignable")
|
expectError(t, `func() { a := {b: {c: 1}}; a.d.c = 2 }()`, nil, "not index-assignable")
|
||||||
expectError(t, `func() { a := [1, 2, 3]; a.b = 2 }()`, "invalid index type")
|
expectError(t, `func() { a := [1, 2, 3]; a.b = 2 }()`, nil, "invalid index type")
|
||||||
expectError(t, `func() { a := "foo"; a.b = 2 }()`, "not index-assignable")
|
expectError(t, `func() { a := "foo"; a.b = 2 }()`, nil, "not index-assignable")
|
||||||
}
|
}
|
||||||
|
|
13
runtime/vm_source_modules_test.go
Normal file
13
runtime/vm_source_modules_test.go
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
package runtime_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/stdlib"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestSourceModules(t *testing.T) {
|
||||||
|
expect(t, `enum := import("enum"); out = enum.any([1,2,3], func(i, v) { return v == 2 })`,
|
||||||
|
Opts().Module("enum", stdlib.SourceModules["enum"]),
|
||||||
|
true)
|
||||||
|
}
|
|
@ -8,67 +8,66 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestString(t *testing.T) {
|
func TestString(t *testing.T) {
|
||||||
expect(t, `out = "Hello World!"`, "Hello World!")
|
expect(t, `out = "Hello World!"`, nil, "Hello World!")
|
||||||
expect(t, `out = "Hello" + " " + "World!"`, "Hello World!")
|
expect(t, `out = "Hello" + " " + "World!"`, nil, "Hello World!")
|
||||||
|
|
||||||
expect(t, `out = "Hello" == "Hello"`, true)
|
expect(t, `out = "Hello" == "Hello"`, nil, true)
|
||||||
expect(t, `out = "Hello" == "World"`, false)
|
expect(t, `out = "Hello" == "World"`, nil, false)
|
||||||
expect(t, `out = "Hello" != "Hello"`, false)
|
expect(t, `out = "Hello" != "Hello"`, nil, false)
|
||||||
expect(t, `out = "Hello" != "World"`, true)
|
expect(t, `out = "Hello" != "World"`, nil, true)
|
||||||
|
|
||||||
// index operator
|
// index operator
|
||||||
str := "abcdef"
|
str := "abcdef"
|
||||||
strStr := `"abcdef"`
|
strStr := `"abcdef"`
|
||||||
strLen := 6
|
strLen := 6
|
||||||
for idx := 0; idx < strLen; idx++ {
|
for idx := 0; idx < strLen; idx++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d]", strStr, idx), str[idx])
|
expect(t, fmt.Sprintf("out = %s[%d]", strStr, idx), nil, str[idx])
|
||||||
expect(t, fmt.Sprintf("out = %s[0 + %d]", strStr, idx), str[idx])
|
expect(t, fmt.Sprintf("out = %s[0 + %d]", strStr, idx), nil, str[idx])
|
||||||
expect(t, fmt.Sprintf("out = %s[1 + %d - 1]", strStr, idx), str[idx])
|
expect(t, fmt.Sprintf("out = %s[1 + %d - 1]", strStr, idx), nil, str[idx])
|
||||||
expect(t, fmt.Sprintf("idx := %d; out = %s[idx]", idx, strStr), str[idx])
|
expect(t, fmt.Sprintf("idx := %d; out = %s[idx]", idx, strStr), nil, str[idx])
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(t, fmt.Sprintf("%s[%d]", strStr, -1), objects.UndefinedValue)
|
expect(t, fmt.Sprintf("%s[%d]", strStr, -1), nil, objects.UndefinedValue)
|
||||||
expect(t, fmt.Sprintf("%s[%d]", strStr, strLen), objects.UndefinedValue)
|
expect(t, fmt.Sprintf("%s[%d]", strStr, strLen), nil, objects.UndefinedValue)
|
||||||
|
|
||||||
// slice operator
|
// slice operator
|
||||||
for low := 0; low <= strLen; low++ {
|
for low := 0; low <= strLen; low++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, low, low), "")
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, low, low), nil, "")
|
||||||
for high := low; high <= strLen; high++ {
|
for high := low; high <= strLen; high++ {
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, low, high), str[low:high])
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, low, high), nil, str[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[0 + %d : 0 + %d]", strStr, low, high), str[low:high])
|
expect(t, fmt.Sprintf("out = %s[0 + %d : 0 + %d]", strStr, low, high), nil, str[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[1 + %d - 1 : 1 + %d - 1]", strStr, low, high), str[low:high])
|
expect(t, fmt.Sprintf("out = %s[1 + %d - 1 : 1 + %d - 1]", strStr, low, high), nil, str[low:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[:%d]", strStr, high), str[:high])
|
expect(t, fmt.Sprintf("out = %s[:%d]", strStr, high), nil, str[:high])
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:]", strStr, low), str[low:])
|
expect(t, fmt.Sprintf("out = %s[%d:]", strStr, low), nil, str[low:])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
expect(t, fmt.Sprintf("out = %s[:]", strStr), str[:])
|
expect(t, fmt.Sprintf("out = %s[:]", strStr), nil, str[:])
|
||||||
expect(t, fmt.Sprintf("out = %s[:]", strStr), str)
|
expect(t, fmt.Sprintf("out = %s[:]", strStr), nil, str)
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:]", strStr, -1), str)
|
expect(t, fmt.Sprintf("out = %s[%d:]", strStr, -1), nil, str)
|
||||||
expect(t, fmt.Sprintf("out = %s[:%d]", strStr, strLen+1), str)
|
expect(t, fmt.Sprintf("out = %s[:%d]", strStr, strLen+1), nil, str)
|
||||||
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, 2, 2), "")
|
expect(t, fmt.Sprintf("out = %s[%d:%d]", strStr, 2, 2), nil, "")
|
||||||
|
|
||||||
expectError(t, fmt.Sprintf("%s[:%d]", strStr, -1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[:%d]", strStr, -1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:]", strStr, strLen+1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:]", strStr, strLen+1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 0, -1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 0, -1), nil, "invalid slice index")
|
||||||
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 2, 1), "invalid slice index")
|
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 2, 1), nil, "invalid slice index")
|
||||||
|
|
||||||
// string concatenation with other types
|
// string concatenation with other types
|
||||||
expect(t, `out = "foo" + 1`, "foo1")
|
expect(t, `out = "foo" + 1`, nil, "foo1")
|
||||||
// Float.String() returns the smallest number of digits
|
// Float.String() returns the smallest number of digits
|
||||||
// necessary such that ParseFloat will return f exactly.
|
// necessary such that ParseFloat will return f exactly.
|
||||||
expect(t, `out = "foo" + 1.0`, "foo1") // <- note '1' instead of '1.0'
|
expect(t, `out = "foo" + 1.0`, nil, "foo1") // <- note '1' instead of '1.0'
|
||||||
expect(t, `out = "foo" + 1.5`, "foo1.5")
|
expect(t, `out = "foo" + 1.5`, nil, "foo1.5")
|
||||||
expect(t, `out = "foo" + true`, "footrue")
|
expect(t, `out = "foo" + true`, nil, "footrue")
|
||||||
expect(t, `out = "foo" + 'X'`, "fooX")
|
expect(t, `out = "foo" + 'X'`, nil, "fooX")
|
||||||
expect(t, `out = "foo" + error(5)`, "fooerror: 5")
|
expect(t, `out = "foo" + error(5)`, nil, "fooerror: 5")
|
||||||
expect(t, `out = "foo" + undefined`, "foo<undefined>")
|
expect(t, `out = "foo" + undefined`, nil, "foo<undefined>")
|
||||||
expect(t, `out = "foo" + [1,2,3]`, "foo[1, 2, 3]")
|
expect(t, `out = "foo" + [1,2,3]`, nil, "foo[1, 2, 3]")
|
||||||
//expect(t, `out = "foo" + {a: 1, b: 2}`, "foo{a: 1, b: 2}") // TODO: commented because order of key is not consistent
|
|
||||||
// also works with "+=" operator
|
// also works with "+=" operator
|
||||||
expect(t, `out = "foo"; out += 1.5`, "foo1.5")
|
expect(t, `out = "foo"; out += 1.5`, nil, "foo1.5")
|
||||||
// string concats works only when string is LHS
|
// string concats works only when string is LHS
|
||||||
expectError(t, `1 + "foo"`, "invalid operation")
|
expectError(t, `1 + "foo"`, nil, "invalid operation")
|
||||||
|
|
||||||
expectError(t, `"foo" - "bar"`, "invalid operation")
|
expectError(t, `"foo" - "bar"`, nil, "invalid operation")
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ func TestTailCall(t *testing.T) {
|
||||||
}
|
}
|
||||||
return fac(n-1, n*a)
|
return fac(n-1, n*a)
|
||||||
}
|
}
|
||||||
out = fac(5, 1)`, 120)
|
out = fac(5, 1)`, nil, 120)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
fac := func(n, a) {
|
fac := func(n, a) {
|
||||||
|
@ -20,7 +20,7 @@ func TestTailCall(t *testing.T) {
|
||||||
x := {foo: fac} // indirection for test
|
x := {foo: fac} // indirection for test
|
||||||
return x.foo(n-1, n*a)
|
return x.foo(n-1, n*a)
|
||||||
}
|
}
|
||||||
out = fac(5, 1)`, 120)
|
out = fac(5, 1)`, nil, 120)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
fib := func(x, s) {
|
fib := func(x, s) {
|
||||||
|
@ -31,7 +31,7 @@ func TestTailCall(t *testing.T) {
|
||||||
}
|
}
|
||||||
return fib(x-1, fib(x-2, s))
|
return fib(x-1, fib(x-2, s))
|
||||||
}
|
}
|
||||||
out = fib(15, 0)`, 610)
|
out = fib(15, 0)`, nil, 610)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
fib := func(n, a, b) {
|
fib := func(n, a, b) {
|
||||||
|
@ -42,7 +42,7 @@ func TestTailCall(t *testing.T) {
|
||||||
}
|
}
|
||||||
return fib(n-1, b, a + b)
|
return fib(n-1, b, a + b)
|
||||||
}
|
}
|
||||||
out = fib(15, 0, 1)`, 610)
|
out = fib(15, 0, 1)`, nil, 610)
|
||||||
|
|
||||||
// global variable and no return value
|
// global variable and no return value
|
||||||
expect(t, `
|
expect(t, `
|
||||||
|
@ -54,7 +54,7 @@ func TestTailCall(t *testing.T) {
|
||||||
out += a
|
out += a
|
||||||
foo(a-1)
|
foo(a-1)
|
||||||
}
|
}
|
||||||
foo(10)`, 55)
|
foo(10)`, nil, 55)
|
||||||
|
|
||||||
expect(t, `
|
expect(t, `
|
||||||
f1 := func() {
|
f1 := func() {
|
||||||
|
@ -65,7 +65,7 @@ func TestTailCall(t *testing.T) {
|
||||||
}
|
}
|
||||||
return f2(5, 0)
|
return f2(5, 0)
|
||||||
}
|
}
|
||||||
out = f1()`, 15)
|
out = f1()`, nil, 15)
|
||||||
|
|
||||||
// tail-call replacing loop
|
// tail-call replacing loop
|
||||||
// without tail-call optimization, this code will cause stack overflow
|
// without tail-call optimization, this code will cause stack overflow
|
||||||
|
@ -78,7 +78,7 @@ iter := func(n, max) {
|
||||||
return iter(n+1, max)
|
return iter(n+1, max)
|
||||||
}
|
}
|
||||||
out = iter(0, 9999)
|
out = iter(0, 9999)
|
||||||
`, 9999)
|
`, nil, 9999)
|
||||||
expect(t, `
|
expect(t, `
|
||||||
c := 0
|
c := 0
|
||||||
iter := func(n, max) {
|
iter := func(n, max) {
|
||||||
|
@ -91,7 +91,7 @@ iter := func(n, max) {
|
||||||
}
|
}
|
||||||
iter(0, 9999)
|
iter(0, 9999)
|
||||||
out = c
|
out = c
|
||||||
`, 9999)
|
`, nil, 9999)
|
||||||
}
|
}
|
||||||
|
|
||||||
// tail call with free vars
|
// tail call with free vars
|
||||||
|
@ -107,5 +107,5 @@ func() {
|
||||||
return f2(n-1, n+s)
|
return f2(n-1, n+s)
|
||||||
}
|
}
|
||||||
out = f2(5, 0)
|
out = f2(5, 0)
|
||||||
}()`, 25)
|
}()`, nil, 25)
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,53 +22,81 @@ type IARR []interface{}
|
||||||
type IMAP map[string]interface{}
|
type IMAP map[string]interface{}
|
||||||
type MAP = map[string]interface{}
|
type MAP = map[string]interface{}
|
||||||
type ARR = []interface{}
|
type ARR = []interface{}
|
||||||
type SYM = map[string]objects.Object
|
|
||||||
|
|
||||||
func expect(t *testing.T, input string, expected interface{}) {
|
type testopts struct {
|
||||||
runVM(t, input, expected, nil, nil, nil, -1, false)
|
modules map[string]objects.Importable
|
||||||
|
symbols map[string]objects.Object
|
||||||
|
maxAllocs int64
|
||||||
|
skip2ndPass bool
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectAllocsLimit(t *testing.T, input string, maxAllocs int64, expected interface{}) {
|
func Opts() *testopts {
|
||||||
runVM(t, input, expected, nil, nil, nil, maxAllocs, true)
|
return &testopts{
|
||||||
|
modules: make(map[string]objects.Importable),
|
||||||
|
symbols: make(map[string]objects.Object),
|
||||||
|
maxAllocs: -1,
|
||||||
|
skip2ndPass: false,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectNoMod(t *testing.T, input string, expected interface{}) {
|
func (o *testopts) copy() *testopts {
|
||||||
runVM(t, input, expected, nil, nil, nil, -1, true)
|
c := &testopts{
|
||||||
|
modules: make(map[string]objects.Importable),
|
||||||
|
symbols: make(map[string]objects.Object),
|
||||||
|
maxAllocs: o.maxAllocs,
|
||||||
|
skip2ndPass: o.skip2ndPass,
|
||||||
|
}
|
||||||
|
for k, v := range o.modules {
|
||||||
|
c.modules[k] = v
|
||||||
|
}
|
||||||
|
for k, v := range o.symbols {
|
||||||
|
c.symbols[k] = v
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectWithSymbols(t *testing.T, input string, expected interface{}, symbols map[string]objects.Object) {
|
func (o *testopts) Module(name string, mod interface{}) *testopts {
|
||||||
runVM(t, input, expected, symbols, nil, nil, -1, true)
|
c := o.copy()
|
||||||
|
switch mod := mod.(type) {
|
||||||
|
case objects.Importable:
|
||||||
|
c.modules[name] = mod
|
||||||
|
case string:
|
||||||
|
c.modules[name] = &objects.SourceModule{Src: []byte(mod)}
|
||||||
|
case []byte:
|
||||||
|
c.modules[name] = &objects.SourceModule{Src: mod}
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("invalid module type: %T", mod))
|
||||||
|
}
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectWithUserModules(t *testing.T, input string, expected interface{}, userModules map[string]string) {
|
func (o *testopts) Symbol(name string, value objects.Object) *testopts {
|
||||||
runVM(t, input, expected, nil, userModules, nil, -1, false)
|
c := o.copy()
|
||||||
|
c.symbols[name] = value
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectWithBuiltinModules(t *testing.T, input string, expected interface{}, builtinModules map[string]objects.Object) {
|
func (o *testopts) MaxAllocs(limit int64) *testopts {
|
||||||
runVM(t, input, expected, nil, nil, builtinModules, -1, false)
|
c := o.copy()
|
||||||
|
c.maxAllocs = limit
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectWithUserAndBuiltinModules(t *testing.T, input string, expected interface{}, userModules map[string]string, builtinModules map[string]objects.Object) {
|
func (o *testopts) Skip2ndPass() *testopts {
|
||||||
runVM(t, input, expected, nil, userModules, builtinModules, -1, false)
|
c := o.copy()
|
||||||
|
c.skip2ndPass = true
|
||||||
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectError(t *testing.T, input, expected string) {
|
func expect(t *testing.T, input string, opts *testopts, expected interface{}) {
|
||||||
runVMError(t, input, nil, nil, nil, -1, expected)
|
if opts == nil {
|
||||||
|
opts = Opts()
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectErrorAllocsLimit(t *testing.T, input string, maxAllocs int64, expected string) {
|
symbols := opts.symbols
|
||||||
runVMError(t, input, nil, nil, nil, maxAllocs, expected)
|
modules := opts.modules
|
||||||
}
|
maxAllocs := opts.maxAllocs
|
||||||
|
|
||||||
func expectErrorWithUserModules(t *testing.T, input string, userModules map[string]string, expected string) {
|
|
||||||
runVMError(t, input, nil, userModules, nil, -1, expected)
|
|
||||||
}
|
|
||||||
|
|
||||||
func expectErrorWithSymbols(t *testing.T, input string, symbols map[string]objects.Object, expected string) {
|
|
||||||
runVMError(t, input, symbols, nil, nil, -1, expected)
|
|
||||||
}
|
|
||||||
|
|
||||||
func runVM(t *testing.T, input string, expected interface{}, symbols map[string]objects.Object, userModules map[string]string, builtinModules map[string]objects.Object, maxAllocs int64, skipModuleTest bool) {
|
|
||||||
expectedObj := toObject(expected)
|
expectedObj := toObject(expected)
|
||||||
|
|
||||||
if symbols == nil {
|
if symbols == nil {
|
||||||
|
@ -85,7 +113,7 @@ func runVM(t *testing.T, input string, expected interface{}, symbols map[string]
|
||||||
}
|
}
|
||||||
|
|
||||||
// compiler/VM
|
// compiler/VM
|
||||||
res, trace, err := traceCompileRun(file, symbols, userModules, builtinModules, maxAllocs)
|
res, trace, err := traceCompileRun(file, symbols, modules, maxAllocs)
|
||||||
if !assert.NoError(t, err) ||
|
if !assert.NoError(t, err) ||
|
||||||
!assert.Equal(t, expectedObj, res[testOut]) {
|
!assert.Equal(t, expectedObj, res[testOut]) {
|
||||||
t.Log("\n" + strings.Join(trace, "\n"))
|
t.Log("\n" + strings.Join(trace, "\n"))
|
||||||
|
@ -93,7 +121,7 @@ func runVM(t *testing.T, input string, expected interface{}, symbols map[string]
|
||||||
}
|
}
|
||||||
|
|
||||||
// second pass: run the code as import module
|
// second pass: run the code as import module
|
||||||
if !skipModuleTest {
|
if !opts.skip2ndPass {
|
||||||
file := parse(t, `out = import("__code__")`)
|
file := parse(t, `out = import("__code__")`)
|
||||||
if file == nil {
|
if file == nil {
|
||||||
return
|
return
|
||||||
|
@ -107,12 +135,11 @@ func runVM(t *testing.T, input string, expected interface{}, symbols map[string]
|
||||||
expectedObj = &objects.ImmutableMap{Value: eo.Value}
|
expectedObj = &objects.ImmutableMap{Value: eo.Value}
|
||||||
}
|
}
|
||||||
|
|
||||||
if userModules == nil {
|
modules["__code__"] = &objects.SourceModule{
|
||||||
userModules = make(map[string]string)
|
Src: []byte(fmt.Sprintf("out := undefined; %s; export out", input)),
|
||||||
}
|
}
|
||||||
userModules["__code__"] = fmt.Sprintf("out := undefined; %s; export out", input)
|
|
||||||
|
|
||||||
res, trace, err := traceCompileRun(file, symbols, userModules, builtinModules, maxAllocs)
|
res, trace, err := traceCompileRun(file, symbols, modules, maxAllocs)
|
||||||
if !assert.NoError(t, err) ||
|
if !assert.NoError(t, err) ||
|
||||||
!assert.Equal(t, expectedObj, res[testOut]) {
|
!assert.Equal(t, expectedObj, res[testOut]) {
|
||||||
t.Log("\n" + strings.Join(trace, "\n"))
|
t.Log("\n" + strings.Join(trace, "\n"))
|
||||||
|
@ -120,7 +147,15 @@ func runVM(t *testing.T, input string, expected interface{}, symbols map[string]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func runVMError(t *testing.T, input string, symbols map[string]objects.Object, userModules map[string]string, builtinModules map[string]objects.Object, maxAllocs int64, expected string) {
|
func expectError(t *testing.T, input string, opts *testopts, expected string) {
|
||||||
|
if opts == nil {
|
||||||
|
opts = Opts()
|
||||||
|
}
|
||||||
|
|
||||||
|
symbols := opts.symbols
|
||||||
|
modules := opts.modules
|
||||||
|
maxAllocs := opts.maxAllocs
|
||||||
|
|
||||||
expected = strings.TrimSpace(expected)
|
expected = strings.TrimSpace(expected)
|
||||||
if expected == "" {
|
if expected == "" {
|
||||||
panic("expected must not be empty")
|
panic("expected must not be empty")
|
||||||
|
@ -133,7 +168,7 @@ func runVMError(t *testing.T, input string, symbols map[string]objects.Object, u
|
||||||
}
|
}
|
||||||
|
|
||||||
// compiler/VM
|
// compiler/VM
|
||||||
_, trace, err := traceCompileRun(program, symbols, userModules, builtinModules, maxAllocs)
|
_, trace, err := traceCompileRun(program, symbols, modules, maxAllocs)
|
||||||
if !assert.Error(t, err) ||
|
if !assert.Error(t, err) ||
|
||||||
!assert.True(t, strings.Contains(err.Error(), expected), "expected error string: %s, got: %s", expected, err.Error()) {
|
!assert.True(t, strings.Contains(err.Error(), expected), "expected error string: %s, got: %s", expected, err.Error()) {
|
||||||
t.Log("\n" + strings.Join(trace, "\n"))
|
t.Log("\n" + strings.Join(trace, "\n"))
|
||||||
|
@ -149,7 +184,7 @@ func (o *tracer) Write(p []byte) (n int, err error) {
|
||||||
return len(p), nil
|
return len(p), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func traceCompileRun(file *ast.File, symbols map[string]objects.Object, userModules map[string]string, builtinModules map[string]objects.Object, maxAllocs int64) (res map[string]objects.Object, trace []string, err error) {
|
func traceCompileRun(file *ast.File, symbols map[string]objects.Object, modules map[string]objects.Importable, maxAllocs int64) (res map[string]objects.Object, trace []string, err error) {
|
||||||
var v *runtime.VM
|
var v *runtime.VM
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
|
@ -185,20 +220,8 @@ func traceCompileRun(file *ast.File, symbols map[string]objects.Object, userModu
|
||||||
symTable.DefineBuiltin(idx, fn.Name)
|
symTable.DefineBuiltin(idx, fn.Name)
|
||||||
}
|
}
|
||||||
|
|
||||||
bm := make(map[string]bool)
|
|
||||||
for k := range builtinModules {
|
|
||||||
bm[k] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
tr := &tracer{}
|
tr := &tracer{}
|
||||||
c := compiler.NewCompiler(file.InputFile, symTable, nil, bm, tr)
|
c := compiler.NewCompiler(file.InputFile, symTable, nil, modules, tr)
|
||||||
c.SetModuleLoader(func(moduleName string) ([]byte, error) {
|
|
||||||
if src, ok := userModules[moduleName]; ok {
|
|
||||||
return []byte(src), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, fmt.Errorf("module '%s' not found", moduleName)
|
|
||||||
})
|
|
||||||
err = c.Compile(file)
|
err = c.Compile(file)
|
||||||
trace = append(trace, fmt.Sprintf("\n[Compiler Trace]\n\n%s", strings.Join(tr.Out, "")))
|
trace = append(trace, fmt.Sprintf("\n[Compiler Trace]\n\n%s", strings.Join(tr.Out, "")))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -210,7 +233,7 @@ func traceCompileRun(file *ast.File, symbols map[string]objects.Object, userModu
|
||||||
trace = append(trace, fmt.Sprintf("\n[Compiled Constants]\n\n%s", strings.Join(bytecode.FormatConstants(), "\n")))
|
trace = append(trace, fmt.Sprintf("\n[Compiled Constants]\n\n%s", strings.Join(bytecode.FormatConstants(), "\n")))
|
||||||
trace = append(trace, fmt.Sprintf("\n[Compiled Instructions]\n\n%s\n", strings.Join(bytecode.FormatInstructions(), "\n")))
|
trace = append(trace, fmt.Sprintf("\n[Compiled Instructions]\n\n%s\n", strings.Join(bytecode.FormatInstructions(), "\n")))
|
||||||
|
|
||||||
v = runtime.NewVM(bytecode, globals, nil, builtinModules, maxAllocs)
|
v = runtime.NewVM(bytecode, globals, maxAllocs)
|
||||||
|
|
||||||
err = v.Run()
|
err = v.Run()
|
||||||
{
|
{
|
||||||
|
|
|
@ -7,15 +7,15 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestUndefined(t *testing.T) {
|
func TestUndefined(t *testing.T) {
|
||||||
expect(t, `out = undefined`, objects.UndefinedValue)
|
expect(t, `out = undefined`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = undefined.a`, objects.UndefinedValue)
|
expect(t, `out = undefined.a`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = undefined[1]`, objects.UndefinedValue)
|
expect(t, `out = undefined[1]`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = undefined.a.b`, objects.UndefinedValue)
|
expect(t, `out = undefined.a.b`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = undefined[1][2]`, objects.UndefinedValue)
|
expect(t, `out = undefined[1][2]`, nil, objects.UndefinedValue)
|
||||||
expect(t, `out = undefined ? 1 : 2`, 2)
|
expect(t, `out = undefined ? 1 : 2`, nil, 2)
|
||||||
expect(t, `out = undefined == undefined`, true)
|
expect(t, `out = undefined == undefined`, nil, true)
|
||||||
expect(t, `out = undefined == 1`, false)
|
expect(t, `out = undefined == 1`, nil, false)
|
||||||
expect(t, `out = 1 == undefined`, false)
|
expect(t, `out = 1 == undefined`, nil, false)
|
||||||
expect(t, `out = undefined == float([])`, true)
|
expect(t, `out = undefined == float([])`, nil, true)
|
||||||
expect(t, `out = float([]) == undefined`, true)
|
expect(t, `out = float([]) == undefined`, nil, true)
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,8 +16,6 @@ type Compiled struct {
|
||||||
globalIndexes map[string]int // global symbol name to index
|
globalIndexes map[string]int // global symbol name to index
|
||||||
bytecode *compiler.Bytecode
|
bytecode *compiler.Bytecode
|
||||||
globals []objects.Object
|
globals []objects.Object
|
||||||
builtinFunctions []objects.Object
|
|
||||||
builtinModules map[string]objects.Object
|
|
||||||
maxAllocs int64
|
maxAllocs int64
|
||||||
lock sync.RWMutex
|
lock sync.RWMutex
|
||||||
}
|
}
|
||||||
|
@ -27,7 +25,7 @@ func (c *Compiled) Run() error {
|
||||||
c.lock.Lock()
|
c.lock.Lock()
|
||||||
defer c.lock.Unlock()
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
v := runtime.NewVM(c.bytecode, c.globals, c.builtinFunctions, c.builtinModules, c.maxAllocs)
|
v := runtime.NewVM(c.bytecode, c.globals, c.maxAllocs)
|
||||||
|
|
||||||
return v.Run()
|
return v.Run()
|
||||||
}
|
}
|
||||||
|
@ -37,7 +35,7 @@ func (c *Compiled) RunContext(ctx context.Context) (err error) {
|
||||||
c.lock.Lock()
|
c.lock.Lock()
|
||||||
defer c.lock.Unlock()
|
defer c.lock.Unlock()
|
||||||
|
|
||||||
v := runtime.NewVM(c.bytecode, c.globals, c.builtinFunctions, c.builtinModules, c.maxAllocs)
|
v := runtime.NewVM(c.bytecode, c.globals, c.maxAllocs)
|
||||||
|
|
||||||
ch := make(chan error, 1)
|
ch := make(chan error, 1)
|
||||||
|
|
||||||
|
@ -66,8 +64,6 @@ func (c *Compiled) Clone() *Compiled {
|
||||||
globalIndexes: c.globalIndexes,
|
globalIndexes: c.globalIndexes,
|
||||||
bytecode: c.bytecode,
|
bytecode: c.bytecode,
|
||||||
globals: make([]objects.Object, len(c.globals)),
|
globals: make([]objects.Object, len(c.globals)),
|
||||||
builtinFunctions: c.builtinFunctions,
|
|
||||||
builtinModules: c.builtinModules,
|
|
||||||
maxAllocs: c.maxAllocs,
|
maxAllocs: c.maxAllocs,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -14,12 +14,11 @@ import (
|
||||||
// Script can simplify compilation and execution of embedded scripts.
|
// Script can simplify compilation and execution of embedded scripts.
|
||||||
type Script struct {
|
type Script struct {
|
||||||
variables map[string]*Variable
|
variables map[string]*Variable
|
||||||
builtinFuncs []objects.Object
|
importModules map[string]objects.Importable
|
||||||
builtinModules map[string]objects.Object
|
|
||||||
userModuleLoader compiler.ModuleLoader
|
|
||||||
input []byte
|
input []byte
|
||||||
maxAllocs int64
|
maxAllocs int64
|
||||||
maxConstObjects int
|
maxConstObjects int
|
||||||
|
enableFileImport bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// New creates a Script instance with an input script.
|
// New creates a Script instance with an input script.
|
||||||
|
@ -59,33 +58,9 @@ func (s *Script) Remove(name string) bool {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetBuiltinFunctions allows to define builtin functions.
|
// SetImports sets import modules.
|
||||||
func (s *Script) SetBuiltinFunctions(funcs []*objects.BuiltinFunction) {
|
func (s *Script) SetImports(modules map[string]objects.Importable) {
|
||||||
if funcs != nil {
|
s.importModules = modules
|
||||||
s.builtinFuncs = make([]objects.Object, len(funcs))
|
|
||||||
for idx, fn := range funcs {
|
|
||||||
s.builtinFuncs[idx] = fn
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
s.builtinFuncs = []objects.Object{}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetBuiltinModules allows to define builtin modules.
|
|
||||||
func (s *Script) SetBuiltinModules(modules map[string]*objects.ImmutableMap) {
|
|
||||||
if modules != nil {
|
|
||||||
s.builtinModules = make(map[string]objects.Object, len(modules))
|
|
||||||
for k, mod := range modules {
|
|
||||||
s.builtinModules[k] = mod
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
s.builtinModules = map[string]objects.Object{}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// SetUserModuleLoader sets the user module loader for the compiler.
|
|
||||||
func (s *Script) SetUserModuleLoader(loader compiler.ModuleLoader) {
|
|
||||||
s.userModuleLoader = loader
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetMaxAllocs sets the maximum number of objects allocations during the run time.
|
// SetMaxAllocs sets the maximum number of objects allocations during the run time.
|
||||||
|
@ -99,9 +74,15 @@ func (s *Script) SetMaxConstObjects(n int) {
|
||||||
s.maxConstObjects = n
|
s.maxConstObjects = n
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EnableFileImport enables or disables module loading from local files.
|
||||||
|
// Local file modules are disabled by default.
|
||||||
|
func (s *Script) EnableFileImport(enable bool) {
|
||||||
|
s.enableFileImport = enable
|
||||||
|
}
|
||||||
|
|
||||||
// Compile compiles the script with all the defined variables, and, returns Compiled object.
|
// Compile compiles the script with all the defined variables, and, returns Compiled object.
|
||||||
func (s *Script) Compile() (*Compiled, error) {
|
func (s *Script) Compile() (*Compiled, error) {
|
||||||
symbolTable, builtinModules, globals, err := s.prepCompile()
|
symbolTable, globals, err := s.prepCompile()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -115,12 +96,8 @@ func (s *Script) Compile() (*Compiled, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
c := compiler.NewCompiler(srcFile, symbolTable, nil, builtinModules, nil)
|
c := compiler.NewCompiler(srcFile, symbolTable, nil, s.importModules, nil)
|
||||||
|
c.EnableFileImport(s.enableFileImport)
|
||||||
if s.userModuleLoader != nil {
|
|
||||||
c.SetModuleLoader(s.userModuleLoader)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := c.Compile(file); err != nil {
|
if err := c.Compile(file); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -153,8 +130,6 @@ func (s *Script) Compile() (*Compiled, error) {
|
||||||
globalIndexes: globalIndexes,
|
globalIndexes: globalIndexes,
|
||||||
bytecode: bytecode,
|
bytecode: bytecode,
|
||||||
globals: globals,
|
globals: globals,
|
||||||
builtinFunctions: s.builtinFuncs,
|
|
||||||
builtinModules: s.builtinModules,
|
|
||||||
maxAllocs: s.maxAllocs,
|
maxAllocs: s.maxAllocs,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
@ -184,36 +159,15 @@ func (s *Script) RunContext(ctx context.Context) (compiled *Compiled, err error)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Script) prepCompile() (symbolTable *compiler.SymbolTable, builtinModules map[string]bool, globals []objects.Object, err error) {
|
func (s *Script) prepCompile() (symbolTable *compiler.SymbolTable, globals []objects.Object, err error) {
|
||||||
var names []string
|
var names []string
|
||||||
for name := range s.variables {
|
for name := range s.variables {
|
||||||
names = append(names, name)
|
names = append(names, name)
|
||||||
}
|
}
|
||||||
|
|
||||||
symbolTable = compiler.NewSymbolTable()
|
symbolTable = compiler.NewSymbolTable()
|
||||||
|
|
||||||
if s.builtinFuncs == nil {
|
|
||||||
s.builtinFuncs = make([]objects.Object, len(objects.Builtins))
|
|
||||||
for idx, fn := range objects.Builtins {
|
for idx, fn := range objects.Builtins {
|
||||||
s.builtinFuncs[idx] = &objects.BuiltinFunction{
|
symbolTable.DefineBuiltin(idx, fn.Name)
|
||||||
Name: fn.Name,
|
|
||||||
Value: fn.Value,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if s.builtinModules == nil {
|
|
||||||
s.builtinModules = make(map[string]objects.Object)
|
|
||||||
}
|
|
||||||
|
|
||||||
for idx, fn := range s.builtinFuncs {
|
|
||||||
f := fn.(*objects.BuiltinFunction)
|
|
||||||
symbolTable.DefineBuiltin(idx, f.Name)
|
|
||||||
}
|
|
||||||
|
|
||||||
builtinModules = make(map[string]bool)
|
|
||||||
for name := range s.builtinModules {
|
|
||||||
builtinModules[name] = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
globals = make([]objects.Object, runtime.GlobalsSize)
|
globals = make([]objects.Object, runtime.GlobalsSize)
|
||||||
|
|
|
@ -33,7 +33,7 @@ b += c
|
||||||
a += b * 2
|
a += b * 2
|
||||||
|
|
||||||
arr := [a, b, c]
|
arr := [a, b, c]
|
||||||
arrstr := stringify(arr)
|
arrstr := string(arr)
|
||||||
map := {a: a, b: b, c: c}
|
map := {a: a, b: b, c: c}
|
||||||
|
|
||||||
d := a + b + c
|
d := a + b + c
|
||||||
|
@ -45,8 +45,8 @@ for i:=1; i<=d; i++ {
|
||||||
|
|
||||||
e := mod1.double(s)
|
e := mod1.double(s)
|
||||||
`)
|
`)
|
||||||
mod1 := &objects.ImmutableMap{
|
mod1 := &objects.BuiltinModule{
|
||||||
Value: map[string]objects.Object{
|
Attrs: map[string]objects.Object{
|
||||||
"double": &objects.UserFunction{
|
"double": &objects.UserFunction{
|
||||||
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
arg0, _ := objects.ToInt64(args[0])
|
arg0, _ := objects.ToInt64(args[0])
|
||||||
|
@ -61,18 +61,9 @@ e := mod1.double(s)
|
||||||
_ = scr.Add("a", 0)
|
_ = scr.Add("a", 0)
|
||||||
_ = scr.Add("b", 0)
|
_ = scr.Add("b", 0)
|
||||||
_ = scr.Add("c", 0)
|
_ = scr.Add("c", 0)
|
||||||
scr.SetBuiltinModules(map[string]*objects.ImmutableMap{
|
scr.SetImports(map[string]objects.Importable{
|
||||||
"mod1": mod1,
|
"mod1": mod1,
|
||||||
})
|
})
|
||||||
scr.SetBuiltinFunctions([]*objects.BuiltinFunction{
|
|
||||||
{
|
|
||||||
Name: "stringify",
|
|
||||||
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
|
||||||
ret = &objects.String{Value: args[0].String()}
|
|
||||||
return
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
compiled, err := scr.Compile()
|
compiled, err := scr.Compile()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
package script_test
|
package script_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"strings"
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
@ -10,57 +9,40 @@ import (
|
||||||
"github.com/d5/tengo/script"
|
"github.com/d5/tengo/script"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestScript_SetUserModuleLoader(t *testing.T) {
|
func TestScriptSourceModule(t *testing.T) {
|
||||||
// script1 imports "mod1"
|
// script1 imports "mod1"
|
||||||
scr := script.New([]byte(`out := import("mod")`))
|
scr := script.New([]byte(`out := import("mod")`))
|
||||||
scr.SetUserModuleLoader(func(name string) ([]byte, error) {
|
scr.SetImports(map[string]objects.Importable{
|
||||||
return []byte(`export 5`), nil
|
"mod": &objects.SourceModule{Src: []byte(`export 5`)},
|
||||||
})
|
})
|
||||||
c, err := scr.Run()
|
c, err := scr.Run()
|
||||||
assert.Equal(t, int64(5), c.Get("out").Value())
|
assert.Equal(t, int64(5), c.Get("out").Value())
|
||||||
|
|
||||||
// executing module function
|
// executing module function
|
||||||
scr = script.New([]byte(`fn := import("mod"); out := fn()`))
|
scr = script.New([]byte(`fn := import("mod"); out := fn()`))
|
||||||
scr.SetUserModuleLoader(func(name string) ([]byte, error) {
|
scr.SetImports(map[string]objects.Importable{
|
||||||
return []byte(`a := 3; export func() { return a + 5 }`), nil
|
"mod": &objects.SourceModule{Src: []byte(`a := 3; export func() { return a + 5 }`)},
|
||||||
})
|
})
|
||||||
c, err = scr.Run()
|
c, err = scr.Run()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, int64(8), c.Get("out").Value())
|
assert.Equal(t, int64(8), c.Get("out").Value())
|
||||||
|
|
||||||
// disabled builtin function
|
|
||||||
scr = script.New([]byte(`out := import("mod")`))
|
scr = script.New([]byte(`out := import("mod")`))
|
||||||
scr.SetUserModuleLoader(func(name string) ([]byte, error) {
|
scr.SetImports(map[string]objects.Importable{
|
||||||
return []byte(`export len([1, 2, 3])`), nil
|
"text": &objects.BuiltinModule{
|
||||||
})
|
Attrs: map[string]objects.Object{
|
||||||
c, err = scr.Run()
|
|
||||||
assert.NoError(t, err)
|
|
||||||
assert.Equal(t, int64(3), c.Get("out").Value())
|
|
||||||
scr.SetBuiltinFunctions(nil)
|
|
||||||
_, err = scr.Run()
|
|
||||||
assert.Error(t, err)
|
|
||||||
|
|
||||||
scr = script.New([]byte(`out := import("mod")`))
|
|
||||||
scr.SetBuiltinModules(map[string]*objects.ImmutableMap{
|
|
||||||
"text": objectPtr(&objects.ImmutableMap{
|
|
||||||
Value: map[string]objects.Object{
|
|
||||||
"title": &objects.UserFunction{Name: "title", Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
"title": &objects.UserFunction{Name: "title", Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
s, _ := objects.ToString(args[0])
|
s, _ := objects.ToString(args[0])
|
||||||
return &objects.String{Value: strings.Title(s)}, nil
|
return &objects.String{Value: strings.Title(s)}, nil
|
||||||
}},
|
}},
|
||||||
},
|
},
|
||||||
}),
|
},
|
||||||
})
|
"mod": &objects.SourceModule{Src: []byte(`text := import("text"); export text.title("foo")`)},
|
||||||
scr.SetUserModuleLoader(func(name string) ([]byte, error) {
|
|
||||||
if name == "mod" {
|
|
||||||
return []byte(`text := import("text"); export text.title("foo")`), nil
|
|
||||||
}
|
|
||||||
return nil, errors.New("module not found")
|
|
||||||
})
|
})
|
||||||
c, err = scr.Run()
|
c, err = scr.Run()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.Equal(t, "Foo", c.Get("out").Value())
|
assert.Equal(t, "Foo", c.Get("out").Value())
|
||||||
scr.SetBuiltinModules(nil)
|
scr.SetImports(nil)
|
||||||
_, err = scr.Run()
|
_, err = scr.Run()
|
||||||
assert.Error(t, err)
|
assert.Error(t, err)
|
||||||
|
|
||||||
|
|
|
@ -1,13 +1,12 @@
|
||||||
package script_test
|
package script_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"math"
|
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/d5/tengo/assert"
|
"github.com/d5/tengo/assert"
|
||||||
"github.com/d5/tengo/objects"
|
"github.com/d5/tengo/objects"
|
||||||
"github.com/d5/tengo/script"
|
"github.com/d5/tengo/script"
|
||||||
|
"github.com/d5/tengo/stdlib"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestScript_Add(t *testing.T) {
|
func TestScript_Add(t *testing.T) {
|
||||||
|
@ -52,56 +51,9 @@ func TestScript_Run(t *testing.T) {
|
||||||
compiledGet(t, c, "a", int64(5))
|
compiledGet(t, c, "a", int64(5))
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestScript_SetBuiltinFunctions(t *testing.T) {
|
func TestScript_BuiltinModules(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[4]})
|
|
||||||
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[4]})
|
|
||||||
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 := script.New([]byte(`math := import("math"); a := math.abs(-19.84)`))
|
||||||
s.SetBuiltinModules(map[string]*objects.ImmutableMap{
|
s.SetImports(map[string]objects.Importable{"math": stdlib.BuiltinModules["math"]})
|
||||||
"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()
|
c, err := s.Run()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.NotNil(t, c)
|
assert.NotNil(t, c)
|
||||||
|
@ -112,11 +64,29 @@ func TestScript_SetBuiltinModules(t *testing.T) {
|
||||||
assert.NotNil(t, c)
|
assert.NotNil(t, c)
|
||||||
compiledGet(t, c, "a", 19.84)
|
compiledGet(t, c, "a", 19.84)
|
||||||
|
|
||||||
s.SetBuiltinModules(map[string]*objects.ImmutableMap{"os": objectPtr(&objects.ImmutableMap{Value: map[string]objects.Object{}})})
|
s.SetImports(map[string]objects.Importable{"os": &objects.BuiltinModule{Attrs: map[string]objects.Object{}}})
|
||||||
_, err = s.Run()
|
_, err = s.Run()
|
||||||
assert.Error(t, err)
|
assert.Error(t, err)
|
||||||
|
|
||||||
s.SetBuiltinModules(nil)
|
s.SetImports(nil)
|
||||||
|
_, err = s.Run()
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestScript_SourceModules(t *testing.T) {
|
||||||
|
s := script.New([]byte(`
|
||||||
|
enum := import("enum")
|
||||||
|
a := enum.all([1,2,3], func(_, v) {
|
||||||
|
return v > 0
|
||||||
|
})
|
||||||
|
`))
|
||||||
|
s.SetImports(stdlib.GetModules("enum"))
|
||||||
|
c, err := s.Run()
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.NotNil(t, c)
|
||||||
|
compiledGet(t, c, "a", true)
|
||||||
|
|
||||||
|
s.SetImports(nil)
|
||||||
_, err = s.Run()
|
_, err = s.Run()
|
||||||
assert.Error(t, err)
|
assert.Error(t, err)
|
||||||
}
|
}
|
||||||
|
@ -154,7 +124,3 @@ func TestScript_SetMaxConstObjects(t *testing.T) {
|
||||||
_, err = s.Compile()
|
_, err = s.Compile()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
func objectPtr(o objects.Object) *objects.ImmutableMap {
|
|
||||||
return o.(*objects.ImmutableMap)
|
|
||||||
}
|
|
||||||
|
|
14
stdlib/builtin_modules.go
Normal file
14
stdlib/builtin_modules.go
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
package stdlib
|
||||||
|
|
||||||
|
import "github.com/d5/tengo/objects"
|
||||||
|
|
||||||
|
// BuiltinModules are builtin type standard library modules.
|
||||||
|
var BuiltinModules = map[string]*objects.BuiltinModule{
|
||||||
|
"math": {Attrs: mathModule},
|
||||||
|
"os": {Attrs: osModule},
|
||||||
|
"text": {Attrs: textModule},
|
||||||
|
"times": {Attrs: timesModule},
|
||||||
|
"rand": {Attrs: randModule},
|
||||||
|
"fmt": {Attrs: fmtModule},
|
||||||
|
"json": {Attrs: jsonModule},
|
||||||
|
}
|
7
stdlib/enum_test.go
Normal file
7
stdlib/enum_test.go
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
package stdlib_test
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestEnum(t *testing.T) {
|
||||||
|
|
||||||
|
}
|
|
@ -1,30 +1,20 @@
|
||||||
package objects
|
package stdlib
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"github.com/d5/tengo"
|
"github.com/d5/tengo"
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
)
|
)
|
||||||
|
|
||||||
func getPrintArgs(args ...Object) ([]interface{}, error) {
|
var fmtModule = map[string]objects.Object{
|
||||||
var printArgs []interface{}
|
"print": &objects.UserFunction{Name: "print", Value: fmtPrint},
|
||||||
l := 0
|
"printf": &objects.UserFunction{Name: "printf", Value: fmtPrintf},
|
||||||
for _, arg := range args {
|
"println": &objects.UserFunction{Name: "println", Value: fmtPrintln},
|
||||||
s, _ := ToString(arg)
|
"sprintf": &objects.UserFunction{Name: "sprintf", Value: fmtSprintf},
|
||||||
slen := len(s)
|
|
||||||
if l+slen > tengo.MaxStringLen { // make sure length does not exceed the limit
|
|
||||||
return nil, ErrStringLimit
|
|
||||||
}
|
|
||||||
l += slen
|
|
||||||
|
|
||||||
printArgs = append(printArgs, s)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return printArgs, nil
|
func fmtPrint(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
}
|
|
||||||
|
|
||||||
// print(args...)
|
|
||||||
func builtinPrint(args ...Object) (Object, error) {
|
|
||||||
printArgs, err := getPrintArgs(args...)
|
printArgs, err := getPrintArgs(args...)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
@ -35,29 +25,15 @@ func builtinPrint(args ...Object) (Object, error) {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// println(args...)
|
func fmtPrintf(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
func builtinPrintln(args ...Object) (Object, error) {
|
|
||||||
printArgs, err := getPrintArgs(args...)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
printArgs = append(printArgs, "\n")
|
|
||||||
_, _ = fmt.Print(printArgs...)
|
|
||||||
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// printf("format", args...)
|
|
||||||
func builtinPrintf(args ...Object) (Object, error) {
|
|
||||||
numArgs := len(args)
|
numArgs := len(args)
|
||||||
if numArgs == 0 {
|
if numArgs == 0 {
|
||||||
return nil, ErrWrongNumArguments
|
return nil, objects.ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
format, ok := args[0].(*String)
|
format, ok := args[0].(*objects.String)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, ErrInvalidArgumentType{
|
return nil, objects.ErrInvalidArgumentType{
|
||||||
Name: "format",
|
Name: "format",
|
||||||
Expected: "string",
|
Expected: "string",
|
||||||
Found: args[0].TypeName(),
|
Found: args[0].TypeName(),
|
||||||
|
@ -70,7 +46,7 @@ func builtinPrintf(args ...Object) (Object, error) {
|
||||||
|
|
||||||
formatArgs := make([]interface{}, numArgs-1, numArgs-1)
|
formatArgs := make([]interface{}, numArgs-1, numArgs-1)
|
||||||
for idx, arg := range args[1:] {
|
for idx, arg := range args[1:] {
|
||||||
formatArgs[idx] = ToInterface(arg)
|
formatArgs[idx] = objects.ToInterface(arg)
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Printf(format.Value, formatArgs...)
|
fmt.Printf(format.Value, formatArgs...)
|
||||||
|
@ -78,16 +54,27 @@ func builtinPrintf(args ...Object) (Object, error) {
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// sprintf("format", args...)
|
func fmtPrintln(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
func builtinSprintf(args ...Object) (Object, error) {
|
printArgs, err := getPrintArgs(args...)
|
||||||
numArgs := len(args)
|
if err != nil {
|
||||||
if numArgs == 0 {
|
return nil, err
|
||||||
return nil, ErrWrongNumArguments
|
|
||||||
}
|
}
|
||||||
|
|
||||||
format, ok := args[0].(*String)
|
printArgs = append(printArgs, "\n")
|
||||||
|
_, _ = fmt.Print(printArgs...)
|
||||||
|
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func fmtSprintf(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
numArgs := len(args)
|
||||||
|
if numArgs == 0 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
format, ok := args[0].(*objects.String)
|
||||||
if !ok {
|
if !ok {
|
||||||
return nil, ErrInvalidArgumentType{
|
return nil, objects.ErrInvalidArgumentType{
|
||||||
Name: "format",
|
Name: "format",
|
||||||
Expected: "string",
|
Expected: "string",
|
||||||
Found: args[0].TypeName(),
|
Found: args[0].TypeName(),
|
||||||
|
@ -99,14 +86,31 @@ func builtinSprintf(args ...Object) (Object, error) {
|
||||||
|
|
||||||
formatArgs := make([]interface{}, numArgs-1, numArgs-1)
|
formatArgs := make([]interface{}, numArgs-1, numArgs-1)
|
||||||
for idx, arg := range args[1:] {
|
for idx, arg := range args[1:] {
|
||||||
formatArgs[idx] = ToInterface(arg)
|
formatArgs[idx] = objects.ToInterface(arg)
|
||||||
}
|
}
|
||||||
|
|
||||||
s := fmt.Sprintf(format.Value, formatArgs...)
|
s := fmt.Sprintf(format.Value, formatArgs...)
|
||||||
|
|
||||||
if len(s) > tengo.MaxStringLen {
|
if len(s) > tengo.MaxStringLen {
|
||||||
return nil, ErrStringLimit
|
return nil, objects.ErrStringLimit
|
||||||
}
|
}
|
||||||
|
|
||||||
return &String{Value: s}, nil
|
return &objects.String{Value: s}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getPrintArgs(args ...objects.Object) ([]interface{}, error) {
|
||||||
|
var printArgs []interface{}
|
||||||
|
l := 0
|
||||||
|
for _, arg := range args {
|
||||||
|
s, _ := objects.ToString(arg)
|
||||||
|
slen := len(s)
|
||||||
|
if l+slen > tengo.MaxStringLen { // make sure length does not exceed the limit
|
||||||
|
return nil, objects.ErrStringLimit
|
||||||
|
}
|
||||||
|
l += slen
|
||||||
|
|
||||||
|
printArgs = append(printArgs, s)
|
||||||
|
}
|
||||||
|
|
||||||
|
return printArgs, nil
|
||||||
}
|
}
|
11
stdlib/fmt_test.go
Normal file
11
stdlib/fmt_test.go
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
package stdlib_test
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestFmtSprintf(t *testing.T) {
|
||||||
|
module(t, `fmt`).call("sprintf", "").expect("")
|
||||||
|
module(t, `fmt`).call("sprintf", "foo").expect("foo")
|
||||||
|
module(t, `fmt`).call("sprintf", `foo %d %v %s`, 1, 2, "bar").expect("foo 1 2 bar")
|
||||||
|
module(t, `fmt`).call("sprintf", "foo %v", `[1, "bar", true]`).expect(`foo [1, "bar", true]`)
|
||||||
|
module(t, `fmt`).call("sprintf", "foo %v %d", `[1, "bar", true]`, 19).expect(`foo [1, "bar", true] 19`)
|
||||||
|
}
|
54
stdlib/gensrcmods.go
Normal file
54
stdlib/gensrcmods.go
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
// +build ignore
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"io/ioutil"
|
||||||
|
"log"
|
||||||
|
"regexp"
|
||||||
|
)
|
||||||
|
|
||||||
|
var tengoModFileRE = regexp.MustCompile(`^srcmod_(\w+).tengo$`)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
modules := make(map[string]string)
|
||||||
|
|
||||||
|
// enumerate all Tengo module files
|
||||||
|
files, err := ioutil.ReadDir(".")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
for _, file := range files {
|
||||||
|
m := tengoModFileRE.FindStringSubmatch(file.Name())
|
||||||
|
if m != nil {
|
||||||
|
modName := m[1]
|
||||||
|
|
||||||
|
src, err := ioutil.ReadFile(file.Name())
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("file '%s' read error: %s", file.Name(), err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
modules[modName] = string(src)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var out bytes.Buffer
|
||||||
|
out.WriteString(`// Code generated using gensrcmods.go; DO NOT EDIT.
|
||||||
|
|
||||||
|
package stdlib
|
||||||
|
|
||||||
|
import "github.com/d5/tengo/objects"
|
||||||
|
|
||||||
|
// SourceModules are source type standard library modules.
|
||||||
|
var SourceModules = map[string]*objects.SourceModule{` + "\n")
|
||||||
|
for modName, modSrc := range modules {
|
||||||
|
out.WriteString("\t\"" + modName + "\": {Src: []byte(`" + modSrc + "`)},\n")
|
||||||
|
}
|
||||||
|
out.WriteString("}\n")
|
||||||
|
|
||||||
|
const target = "source_modules.go"
|
||||||
|
if err := ioutil.WriteFile(target, out.Bytes(), 0644); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
69
stdlib/json.go
Normal file
69
stdlib/json.go
Normal file
|
@ -0,0 +1,69 @@
|
||||||
|
package stdlib
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
"github.com/d5/tengo"
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
|
)
|
||||||
|
|
||||||
|
var jsonModule = map[string]objects.Object{
|
||||||
|
"parse": &objects.UserFunction{Name: "parse", Value: jsonParse},
|
||||||
|
"stringify": &objects.UserFunction{Name: "stringify", Value: jsonStringify},
|
||||||
|
}
|
||||||
|
|
||||||
|
func jsonParse(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var target interface{}
|
||||||
|
|
||||||
|
switch o := args[0].(type) {
|
||||||
|
case *objects.Bytes:
|
||||||
|
err := json.Unmarshal(o.Value, &target)
|
||||||
|
if err != nil {
|
||||||
|
return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
|
||||||
|
}
|
||||||
|
case *objects.String:
|
||||||
|
err := json.Unmarshal([]byte(o.Value), &target)
|
||||||
|
if err != nil {
|
||||||
|
return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
return nil, objects.ErrInvalidArgumentType{
|
||||||
|
Name: "first",
|
||||||
|
Expected: "bytes/string",
|
||||||
|
Found: args[0].TypeName(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := objects.FromInterface(target)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return res, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func jsonStringify(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
v := objects.ToInterface(args[0])
|
||||||
|
if vErr, isErr := v.(error); isErr {
|
||||||
|
v = vErr.Error()
|
||||||
|
}
|
||||||
|
|
||||||
|
res, err := json.Marshal(v)
|
||||||
|
if err != nil {
|
||||||
|
return &objects.Error{Value: &objects.String{Value: err.Error()}}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(res) > tengo.MaxBytesLen {
|
||||||
|
return nil, objects.ErrBytesLimit
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.String{Value: string(res)}, nil
|
||||||
|
}
|
33
stdlib/json_test.go
Normal file
33
stdlib/json_test.go
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
package stdlib_test
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestJSON(t *testing.T) {
|
||||||
|
module(t, "json").call("stringify", 5).expect("5")
|
||||||
|
module(t, "json").call("stringify", "foobar").expect(`"foobar"`)
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": 5}).expect("{\"foo\":5}")
|
||||||
|
module(t, "json").call("stringify", IMAP{"foo": 5}).expect("{\"foo\":5}")
|
||||||
|
module(t, "json").call("stringify", ARR{1, 2, 3}).expect("[1,2,3]")
|
||||||
|
module(t, "json").call("stringify", IARR{1, 2, 3}).expect("[1,2,3]")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": "bar"}).expect("{\"foo\":\"bar\"}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": 1.8}).expect("{\"foo\":1.8}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": true}).expect("{\"foo\":true}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": '8'}).expect("{\"foo\":56}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": []byte("foo")}).expect("{\"foo\":\"Zm9v\"}") // json encoding returns []byte as base64 encoded string
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": ARR{"bar", 1, 1.8, '8', true}}).expect("{\"foo\":[\"bar\",1,1.8,56,true]}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": IARR{"bar", 1, 1.8, '8', true}}).expect("{\"foo\":[\"bar\",1,1.8,56,true]}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": ARR{ARR{"bar", 1}, ARR{"bar", 1}}}).expect("{\"foo\":[[\"bar\",1],[\"bar\",1]]}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": MAP{"string": "bar", "int": 1, "float": 1.8, "char": '8', "bool": true}}).expect("{\"foo\":{\"bool\":true,\"char\":56,\"float\":1.8,\"int\":1,\"string\":\"bar\"}}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": IMAP{"string": "bar", "int": 1, "float": 1.8, "char": '8', "bool": true}}).expect("{\"foo\":{\"bool\":true,\"char\":56,\"float\":1.8,\"int\":1,\"string\":\"bar\"}}")
|
||||||
|
module(t, "json").call("stringify", MAP{"foo": MAP{"map1": MAP{"string": "bar"}, "map2": MAP{"int": "1"}}}).expect("{\"foo\":{\"map1\":{\"string\":\"bar\"},\"map2\":{\"int\":\"1\"}}}")
|
||||||
|
module(t, "json").call("stringify", ARR{ARR{"bar", 1}, ARR{"bar", 1}}).expect("[[\"bar\",1],[\"bar\",1]]")
|
||||||
|
|
||||||
|
module(t, "json").call("parse", `5`).expect(5.0)
|
||||||
|
module(t, "json").call("parse", `"foo"`).expect("foo")
|
||||||
|
module(t, "json").call("parse", `[1,2,3,"bar"]`).expect(ARR{1.0, 2.0, 3.0, "bar"})
|
||||||
|
module(t, "json").call("parse", `{"foo":5}`).expect(MAP{"foo": 5.0})
|
||||||
|
module(t, "json").call("parse", `{"foo":2.5}`).expect(MAP{"foo": 2.5})
|
||||||
|
module(t, "json").call("parse", `{"foo":true}`).expect(MAP{"foo": true})
|
||||||
|
module(t, "json").call("parse", `{"foo":"bar"}`).expect(MAP{"foo": "bar"})
|
||||||
|
module(t, "json").call("parse", `{"foo":[1,2,3,"bar"]}`).expect(MAP{"foo": ARR{1.0, 2.0, 3.0, "bar"}})
|
||||||
|
}
|
50
stdlib/source_modules.go
Normal file
50
stdlib/source_modules.go
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
// Code generated using gensrcmods.go; DO NOT EDIT.
|
||||||
|
|
||||||
|
package stdlib
|
||||||
|
|
||||||
|
import "github.com/d5/tengo/objects"
|
||||||
|
|
||||||
|
// SourceModules are source type standard library modules.
|
||||||
|
var SourceModules = map[string]*objects.SourceModule{
|
||||||
|
"enum": {Src: []byte(`export {
|
||||||
|
// all returns true if the given function fn evaluates to a truthy value on
|
||||||
|
// all of the items in the enumerable.
|
||||||
|
all: func(enumerable, fn) {
|
||||||
|
for k, v in enumerable {
|
||||||
|
if !fn(k, v) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
},
|
||||||
|
// any returns true if the given function fn evaluates to a truthy value on
|
||||||
|
// any of the items in the enumerable.
|
||||||
|
any: func(enumerable, fn) {
|
||||||
|
for k, v in enumerable {
|
||||||
|
if fn(k, v) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
// chunk returns an array of elements split into groups the length of size.
|
||||||
|
// If the enumerable can't be split evenly, the final chunk will be the
|
||||||
|
// remaining elements.
|
||||||
|
chunk: func(enumerable, size) {
|
||||||
|
numElements := len(enumerable)
|
||||||
|
|
||||||
|
if !numElements {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
|
||||||
|
res := []
|
||||||
|
idx := 0
|
||||||
|
for idx < numElements {
|
||||||
|
res = append(res, enumerable[idx:idx+size])
|
||||||
|
idx += size
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`)},
|
||||||
|
}
|
40
stdlib/srcmod_enum.tengo
Normal file
40
stdlib/srcmod_enum.tengo
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
export {
|
||||||
|
// all returns true if the given function fn evaluates to a truthy value on
|
||||||
|
// all of the items in the enumerable.
|
||||||
|
all: func(enumerable, fn) {
|
||||||
|
for k, v in enumerable {
|
||||||
|
if !fn(k, v) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
},
|
||||||
|
// any returns true if the given function fn evaluates to a truthy value on
|
||||||
|
// any of the items in the enumerable.
|
||||||
|
any: func(enumerable, fn) {
|
||||||
|
for k, v in enumerable {
|
||||||
|
if fn(k, v) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
},
|
||||||
|
// chunk returns an array of elements split into groups the length of size.
|
||||||
|
// If the enumerable can't be split evenly, the final chunk will be the
|
||||||
|
// remaining elements.
|
||||||
|
chunk: func(enumerable, size) {
|
||||||
|
numElements := len(enumerable)
|
||||||
|
|
||||||
|
if !numElements {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
|
||||||
|
res := []
|
||||||
|
idx := 0
|
||||||
|
for idx < numElements {
|
||||||
|
res = append(res, enumerable[idx:idx+size])
|
||||||
|
idx += size
|
||||||
|
}
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,20 +1,16 @@
|
||||||
package stdlib
|
package stdlib
|
||||||
|
|
||||||
import "github.com/d5/tengo/objects"
|
//go:generate go run gensrcmods.go
|
||||||
|
|
||||||
// Modules contain the standard modules.
|
import "github.com/d5/tengo/objects"
|
||||||
var Modules = map[string]*objects.ImmutableMap{
|
|
||||||
"math": &objects.ImmutableMap{Value: mathModule},
|
|
||||||
"os": &objects.ImmutableMap{Value: osModule},
|
|
||||||
"text": &objects.ImmutableMap{Value: textModule},
|
|
||||||
"times": &objects.ImmutableMap{Value: timesModule},
|
|
||||||
"rand": &objects.ImmutableMap{Value: randModule},
|
|
||||||
}
|
|
||||||
|
|
||||||
// AllModuleNames returns a list of all default module names.
|
// AllModuleNames returns a list of all default module names.
|
||||||
func AllModuleNames() []string {
|
func AllModuleNames() []string {
|
||||||
var names []string
|
var names []string
|
||||||
for name := range Modules {
|
for name := range BuiltinModules {
|
||||||
|
names = append(names, name)
|
||||||
|
}
|
||||||
|
for name := range SourceModules {
|
||||||
names = append(names, name)
|
names = append(names, name)
|
||||||
}
|
}
|
||||||
return names
|
return names
|
||||||
|
@ -22,12 +18,16 @@ func AllModuleNames() []string {
|
||||||
|
|
||||||
// GetModules returns the modules for the given names.
|
// GetModules returns the modules for the given names.
|
||||||
// Duplicate names and invalid names are ignore.
|
// Duplicate names and invalid names are ignore.
|
||||||
func GetModules(names ...string) map[string]*objects.ImmutableMap {
|
func GetModules(names ...string) map[string]objects.Importable {
|
||||||
modules := make(map[string]*objects.ImmutableMap)
|
modules := make(map[string]objects.Importable)
|
||||||
for _, name := range names {
|
for _, name := range names {
|
||||||
if mod := Modules[name]; mod != nil {
|
if mod := BuiltinModules[name]; mod != nil {
|
||||||
|
modules[name] = mod
|
||||||
|
}
|
||||||
|
if mod := SourceModules[name]; mod != nil {
|
||||||
modules[name] = mod
|
modules[name] = mod
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return modules
|
return modules
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,12 +18,9 @@ type IMAP map[string]interface{}
|
||||||
|
|
||||||
func TestAllModuleNames(t *testing.T) {
|
func TestAllModuleNames(t *testing.T) {
|
||||||
names := stdlib.AllModuleNames()
|
names := stdlib.AllModuleNames()
|
||||||
if !assert.Equal(t, len(stdlib.Modules), len(names)) {
|
if !assert.Equal(t, len(stdlib.BuiltinModules)+len(stdlib.SourceModules), len(names)) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
for _, name := range names {
|
|
||||||
assert.NotNil(t, stdlib.Modules[name], "name: %s", name)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestModulesRun(t *testing.T) {
|
func TestModulesRun(t *testing.T) {
|
||||||
|
@ -100,7 +97,7 @@ func TestGetModules(t *testing.T) {
|
||||||
|
|
||||||
type callres struct {
|
type callres struct {
|
||||||
t *testing.T
|
t *testing.T
|
||||||
o objects.Object
|
o interface{}
|
||||||
e error
|
e error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -109,12 +106,14 @@ func (c callres) call(funcName string, args ...interface{}) callres {
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
imap, ok := c.o.(*objects.ImmutableMap)
|
var oargs []objects.Object
|
||||||
if !ok {
|
for _, v := range args {
|
||||||
return c
|
oargs = append(oargs, object(v))
|
||||||
}
|
}
|
||||||
|
|
||||||
m, ok := imap.Value[funcName]
|
switch o := c.o.(type) {
|
||||||
|
case *objects.BuiltinModule:
|
||||||
|
m, ok := o.Attrs[funcName]
|
||||||
if !ok {
|
if !ok {
|
||||||
return callres{t: c.t, e: fmt.Errorf("function not found: %s", funcName)}
|
return callres{t: c.t, e: fmt.Errorf("function not found: %s", funcName)}
|
||||||
}
|
}
|
||||||
|
@ -124,14 +123,27 @@ func (c callres) call(funcName string, args ...interface{}) callres {
|
||||||
return callres{t: c.t, e: fmt.Errorf("non-callable: %s", funcName)}
|
return callres{t: c.t, e: fmt.Errorf("non-callable: %s", funcName)}
|
||||||
}
|
}
|
||||||
|
|
||||||
var oargs []objects.Object
|
res, err := f.Value(oargs...)
|
||||||
for _, v := range args {
|
return callres{t: c.t, o: res, e: err}
|
||||||
oargs = append(oargs, object(v))
|
case *objects.UserFunction:
|
||||||
|
res, err := o.Value(oargs...)
|
||||||
|
return callres{t: c.t, o: res, e: err}
|
||||||
|
case *objects.ImmutableMap:
|
||||||
|
m, ok := o.Value[funcName]
|
||||||
|
if !ok {
|
||||||
|
return callres{t: c.t, e: fmt.Errorf("function not found: %s", funcName)}
|
||||||
|
}
|
||||||
|
|
||||||
|
f, ok := m.(*objects.UserFunction)
|
||||||
|
if !ok {
|
||||||
|
return callres{t: c.t, e: fmt.Errorf("non-callable: %s", funcName)}
|
||||||
}
|
}
|
||||||
|
|
||||||
res, err := f.Value(oargs...)
|
res, err := f.Value(oargs...)
|
||||||
|
|
||||||
return callres{t: c.t, o: res, e: err}
|
return callres{t: c.t, o: res, e: err}
|
||||||
|
default:
|
||||||
|
panic(fmt.Errorf("unexpected object: %v (%T)", o, o))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c callres) expect(expected interface{}, msgAndArgs ...interface{}) bool {
|
func (c callres) expect(expected interface{}, msgAndArgs ...interface{}) bool {
|
||||||
|
@ -144,7 +156,7 @@ func (c callres) expectError() bool {
|
||||||
}
|
}
|
||||||
|
|
||||||
func module(t *testing.T, moduleName string) callres {
|
func module(t *testing.T, moduleName string) callres {
|
||||||
mod, ok := stdlib.Modules[moduleName]
|
mod, ok := stdlib.BuiltinModules[moduleName]
|
||||||
if !ok {
|
if !ok {
|
||||||
return callres{t: t, e: fmt.Errorf("module not found: %s", moduleName)}
|
return callres{t: t, e: fmt.Errorf("module not found: %s", moduleName)}
|
||||||
}
|
}
|
||||||
|
@ -219,7 +231,7 @@ func object(v interface{}) objects.Object {
|
||||||
|
|
||||||
func expect(t *testing.T, input string, expected interface{}) {
|
func expect(t *testing.T, input string, expected interface{}) {
|
||||||
s := script.New([]byte(input))
|
s := script.New([]byte(input))
|
||||||
s.SetBuiltinModules(stdlib.Modules)
|
s.SetImports(stdlib.GetModules(stdlib.AllModuleNames()...))
|
||||||
c, err := s.Run()
|
c, err := s.Run()
|
||||||
assert.NoError(t, err)
|
assert.NoError(t, err)
|
||||||
assert.NotNil(t, c)
|
assert.NotNil(t, c)
|
||||||
|
|
Loading…
Reference in a new issue