parent
8171d58071
commit
3f55a6b5b0
44 changed files with 1316 additions and 92 deletions
29
README.md
29
README.md
|
@ -186,6 +186,26 @@ if is_error(err1) { // 'is_error' builtin function
|
||||||
```
|
```
|
||||||
> [Run in Playground](https://tengolang.com/?s=5eaba4289c9d284d97704dd09cb15f4f03ad05c1)
|
> [Run in Playground](https://tengolang.com/?s=5eaba4289c9d284d97704dd09cb15f4f03ad05c1)
|
||||||
|
|
||||||
|
You can load other scripts as import modules using `import` expression.
|
||||||
|
|
||||||
|
Main script:
|
||||||
|
```golang
|
||||||
|
mod1 := import("./mod1") // assuming mod1.tengo file exists in the current directory
|
||||||
|
// same as 'import("./mod1.tengo")' or 'import("mod1")'
|
||||||
|
mod1.func1(a) // module function
|
||||||
|
a += mod1.foo // module variable
|
||||||
|
//mod1.foo = 5 // error: module variables are read-only
|
||||||
|
```
|
||||||
|
|
||||||
|
`mod1.tengo` file:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
func1 := func(x) { print(x) }
|
||||||
|
foo := 2
|
||||||
|
```
|
||||||
|
|
||||||
|
Basically `import` expression returns all the global variables defined in the module as a Map-like value. One can access the functions or variables defined in the module using `.` selector or `["key"]` indexer, but, module variables are immutable.
|
||||||
|
|
||||||
|
|
||||||
## Embedding Tengo in Go
|
## Embedding Tengo in Go
|
||||||
|
|
||||||
|
@ -245,9 +265,9 @@ func main() {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
In the example above, a variable `b` is defined by the user before compiliation using `Script.Add()` function. Then a compiled bytecode `c` is used to execute the bytecode and get the value of global variables. In thie example, the value of global variable `a` is read using `Compiled.Get()` function.
|
In the example above, a variable `b` is defined by the user before compiliation using `Script.Add()` function. Then a compiled bytecode `c` is used to execute the bytecode and get the value of global variables. In this example, the value of global variable `a` is read using `Compiled.Get()` function.
|
||||||
|
|
||||||
If you need the custom data types (outside Tengo's primitive types), you can define your own `struct` that implements `objects.Object` interface _(and optinoally `objects.Callable` if you want to make function-like invokable objects)_.
|
If you need the custom data types (outside Tengo's primitive types), you can define your own `struct` that implements `objects.Object` interface _(and optionally `objects.Callable` if you want to make function-like invokable objects)_.
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
import (
|
import (
|
||||||
|
@ -380,10 +400,9 @@ tengo
|
||||||
|
|
||||||
Development roadmap for Tengo:
|
Development roadmap for Tengo:
|
||||||
|
|
||||||
- Module system _(or packages)_
|
- Standard libraries _(modules)_
|
||||||
- Standard libraries
|
|
||||||
- Better documentations
|
- Better documentations
|
||||||
- More language constructs such as error handling, object methods, switch-case statements
|
- More language constructs such as destructuring assignment, `this` binding for object methods, switch-case statements
|
||||||
- Native executables compilation
|
- Native executables compilation
|
||||||
- Performance improvements
|
- Performance improvements
|
||||||
- Syntax highlighter for IDEs
|
- Syntax highlighter for IDEs
|
||||||
|
|
|
@ -26,10 +26,10 @@ func (e *ArrayLit) End() source.Pos {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *ArrayLit) String() string {
|
func (e *ArrayLit) String() string {
|
||||||
var elts []string
|
var elements []string
|
||||||
for _, m := range e.Elements {
|
for _, m := range e.Elements {
|
||||||
elts = append(elts, m.String())
|
elements = append(elements, m.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
return "[" + strings.Join(elts, ", ") + "]"
|
return "[" + strings.Join(elements, ", ") + "]"
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
||||||
|
|
||||||
import "github.com/d5/tengo/compiler/source"
|
import "github.com/d5/tengo/compiler/source"
|
||||||
|
|
||||||
// BoolLit represetns a boolean literal.
|
// BoolLit represents a boolean literal.
|
||||||
type BoolLit struct {
|
type BoolLit struct {
|
||||||
Value bool
|
Value bool
|
||||||
ValuePos source.Pos
|
ValuePos source.Pos
|
||||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
||||||
|
|
||||||
import "github.com/d5/tengo/compiler/source"
|
import "github.com/d5/tengo/compiler/source"
|
||||||
|
|
||||||
// ForStmt represetns a for statement.
|
// ForStmt represents a for statement.
|
||||||
type ForStmt struct {
|
type ForStmt struct {
|
||||||
ForPos source.Pos
|
ForPos source.Pos
|
||||||
Init Stmt
|
Init Stmt
|
||||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
||||||
|
|
||||||
import "github.com/d5/tengo/compiler/source"
|
import "github.com/d5/tengo/compiler/source"
|
||||||
|
|
||||||
// FuncType represetns a function type definition.
|
// FuncType represents a function type definition.
|
||||||
type FuncType struct {
|
type FuncType struct {
|
||||||
FuncPos source.Pos
|
FuncPos source.Pos
|
||||||
Params *IdentList
|
Params *IdentList
|
||||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
||||||
|
|
||||||
import "github.com/d5/tengo/compiler/source"
|
import "github.com/d5/tengo/compiler/source"
|
||||||
|
|
||||||
// Ident represetns an identifier.
|
// Ident represents an identifier.
|
||||||
type Ident struct {
|
type Ident struct {
|
||||||
Name string
|
Name string
|
||||||
NamePos source.Pos
|
NamePos source.Pos
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
"github.com/d5/tengo/compiler/source"
|
"github.com/d5/tengo/compiler/source"
|
||||||
)
|
)
|
||||||
|
|
||||||
// IdentList represetns a list of identifiers.
|
// IdentList represents a list of identifiers.
|
||||||
type IdentList struct {
|
type IdentList struct {
|
||||||
LParen source.Pos
|
LParen source.Pos
|
||||||
List []*Ident
|
List []*Ident
|
||||||
|
|
29
compiler/ast/import_expr.go
Normal file
29
compiler/ast/import_expr.go
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
package ast
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/d5/tengo/compiler/source"
|
||||||
|
"github.com/d5/tengo/compiler/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ImportExpr represents an import expression
|
||||||
|
type ImportExpr struct {
|
||||||
|
ModuleName string
|
||||||
|
Token token.Token
|
||||||
|
TokenPos source.Pos
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ImportExpr) exprNode() {}
|
||||||
|
|
||||||
|
// Pos returns the position of first character belonging to the node.
|
||||||
|
func (e *ImportExpr) Pos() source.Pos {
|
||||||
|
return e.TokenPos
|
||||||
|
}
|
||||||
|
|
||||||
|
// End returns the position of first character immediately after the node.
|
||||||
|
func (e *ImportExpr) End() source.Pos {
|
||||||
|
return source.Pos(int(e.TokenPos) + 10 + len(e.ModuleName)) // import("moduleName")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *ImportExpr) String() string {
|
||||||
|
return `import("` + e.ModuleName + `")"`
|
||||||
|
}
|
|
@ -2,7 +2,7 @@ package ast
|
||||||
|
|
||||||
import "github.com/d5/tengo/compiler/source"
|
import "github.com/d5/tengo/compiler/source"
|
||||||
|
|
||||||
// IntLit represetns an integer literal.
|
// IntLit represents an integer literal.
|
||||||
type IntLit struct {
|
type IntLit struct {
|
||||||
Value int64
|
Value int64
|
||||||
ValuePos source.Pos
|
ValuePos source.Pos
|
||||||
|
|
|
@ -26,10 +26,10 @@ func (e *MapLit) End() source.Pos {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *MapLit) String() string {
|
func (e *MapLit) String() string {
|
||||||
var elts []string
|
var elements []string
|
||||||
for _, m := range e.Elements {
|
for _, m := range e.Elements {
|
||||||
elts = append(elts, m.String())
|
elements = append(elements, m.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
return "{" + strings.Join(elts, ", ") + "}"
|
return "{" + strings.Join(elements, ", ") + "}"
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,16 +6,21 @@ import (
|
||||||
"reflect"
|
"reflect"
|
||||||
|
|
||||||
"github.com/d5/tengo/compiler/ast"
|
"github.com/d5/tengo/compiler/ast"
|
||||||
|
"github.com/d5/tengo/compiler/stdmods"
|
||||||
"github.com/d5/tengo/compiler/token"
|
"github.com/d5/tengo/compiler/token"
|
||||||
"github.com/d5/tengo/objects"
|
"github.com/d5/tengo/objects"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Compiler compiles the AST into a bytecode.
|
// Compiler compiles the AST into a bytecode.
|
||||||
type Compiler struct {
|
type Compiler struct {
|
||||||
|
parent *Compiler
|
||||||
|
moduleName string
|
||||||
constants []objects.Object
|
constants []objects.Object
|
||||||
symbolTable *SymbolTable
|
symbolTable *SymbolTable
|
||||||
scopes []CompilationScope
|
scopes []CompilationScope
|
||||||
scopeIndex int
|
scopeIndex int
|
||||||
|
moduleLoader ModuleLoader
|
||||||
|
compiledModules map[string]*objects.CompiledModule
|
||||||
loops []*Loop
|
loops []*Loop
|
||||||
loopIndex int
|
loopIndex int
|
||||||
trace io.Writer
|
trace io.Writer
|
||||||
|
@ -42,6 +47,7 @@ func NewCompiler(symbolTable *SymbolTable, trace io.Writer) *Compiler {
|
||||||
scopeIndex: 0,
|
scopeIndex: 0,
|
||||||
loopIndex: -1,
|
loopIndex: -1,
|
||||||
trace: trace,
|
trace: trace,
|
||||||
|
compiledModules: make(map[string]*objects.CompiledModule),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,11 +68,13 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.ExprStmt:
|
case *ast.ExprStmt:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
c.emit(OpPop)
|
c.emit(OpPop)
|
||||||
|
|
||||||
case *ast.IncDecStmt:
|
case *ast.IncDecStmt:
|
||||||
op := token.AddAssign
|
op := token.AddAssign
|
||||||
if node.Token == token.Dec {
|
if node.Token == token.Dec {
|
||||||
|
@ -74,10 +82,12 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
return c.compileAssign([]ast.Expr{node.Expr}, []ast.Expr{&ast.IntLit{Value: 1}}, op)
|
return c.compileAssign([]ast.Expr{node.Expr}, []ast.Expr{&ast.IntLit{Value: 1}}, op)
|
||||||
|
|
||||||
case *ast.ParenExpr:
|
case *ast.ParenExpr:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.BinaryExpr:
|
case *ast.BinaryExpr:
|
||||||
if node.Token == token.LAnd || node.Token == token.LOr {
|
if node.Token == token.LAnd || node.Token == token.LOr {
|
||||||
return c.compileLogical(node)
|
return c.compileLogical(node)
|
||||||
|
@ -149,22 +159,29 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.IntLit:
|
case *ast.IntLit:
|
||||||
c.emit(OpConstant, c.addConstant(&objects.Int{Value: node.Value}))
|
c.emit(OpConstant, c.addConstant(&objects.Int{Value: node.Value}))
|
||||||
|
|
||||||
case *ast.FloatLit:
|
case *ast.FloatLit:
|
||||||
c.emit(OpConstant, c.addConstant(&objects.Float{Value: node.Value}))
|
c.emit(OpConstant, c.addConstant(&objects.Float{Value: node.Value}))
|
||||||
|
|
||||||
case *ast.BoolLit:
|
case *ast.BoolLit:
|
||||||
if node.Value {
|
if node.Value {
|
||||||
c.emit(OpTrue)
|
c.emit(OpTrue)
|
||||||
} else {
|
} else {
|
||||||
c.emit(OpFalse)
|
c.emit(OpFalse)
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.StringLit:
|
case *ast.StringLit:
|
||||||
c.emit(OpConstant, c.addConstant(&objects.String{Value: node.Value}))
|
c.emit(OpConstant, c.addConstant(&objects.String{Value: node.Value}))
|
||||||
|
|
||||||
case *ast.CharLit:
|
case *ast.CharLit:
|
||||||
c.emit(OpConstant, c.addConstant(&objects.Char{Value: node.Value}))
|
c.emit(OpConstant, c.addConstant(&objects.Char{Value: node.Value}))
|
||||||
|
|
||||||
case *ast.UndefinedLit:
|
case *ast.UndefinedLit:
|
||||||
c.emit(OpNull)
|
c.emit(OpNull)
|
||||||
|
|
||||||
case *ast.UnaryExpr:
|
case *ast.UnaryExpr:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -182,6 +199,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.IfStmt:
|
case *ast.IfStmt:
|
||||||
// open new symbol table for the statement
|
// open new symbol table for the statement
|
||||||
c.symbolTable = c.symbolTable.Fork(true)
|
c.symbolTable = c.symbolTable.Fork(true)
|
||||||
|
@ -229,8 +247,10 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
|
|
||||||
case *ast.ForStmt:
|
case *ast.ForStmt:
|
||||||
return c.compileForStmt(node)
|
return c.compileForStmt(node)
|
||||||
|
|
||||||
case *ast.ForInStmt:
|
case *ast.ForInStmt:
|
||||||
return c.compileForInStmt(node)
|
return c.compileForInStmt(node)
|
||||||
|
|
||||||
case *ast.BranchStmt:
|
case *ast.BranchStmt:
|
||||||
if node.Token == token.Break {
|
if node.Token == token.Break {
|
||||||
curLoop := c.currentLoop()
|
curLoop := c.currentLoop()
|
||||||
|
@ -249,17 +269,19 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
} else {
|
} else {
|
||||||
return fmt.Errorf("unknown branch statement: %s", node.Token.String())
|
return fmt.Errorf("unknown branch statement: %s", node.Token.String())
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.BlockStmt:
|
case *ast.BlockStmt:
|
||||||
for _, stmt := range node.Stmts {
|
for _, stmt := range node.Stmts {
|
||||||
if err := c.Compile(stmt); err != nil {
|
if err := c.Compile(stmt); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
case *ast.AssignStmt:
|
|
||||||
|
|
||||||
|
case *ast.AssignStmt:
|
||||||
if err := c.compileAssign(node.LHS, node.RHS, node.Token); err != nil {
|
if err := c.compileAssign(node.LHS, node.RHS, node.Token); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.Ident:
|
case *ast.Ident:
|
||||||
symbol, _, ok := c.symbolTable.Resolve(node.Name)
|
symbol, _, ok := c.symbolTable.Resolve(node.Name)
|
||||||
if !ok {
|
if !ok {
|
||||||
|
@ -276,6 +298,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
case ScopeFree:
|
case ScopeFree:
|
||||||
c.emit(OpGetFree, symbol.Index)
|
c.emit(OpGetFree, symbol.Index)
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.ArrayLit:
|
case *ast.ArrayLit:
|
||||||
for _, elem := range node.Elements {
|
for _, elem := range node.Elements {
|
||||||
if err := c.Compile(elem); err != nil {
|
if err := c.Compile(elem); err != nil {
|
||||||
|
@ -284,6 +307,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(OpArray, len(node.Elements))
|
c.emit(OpArray, len(node.Elements))
|
||||||
|
|
||||||
case *ast.MapLit:
|
case *ast.MapLit:
|
||||||
for _, elt := range node.Elements {
|
for _, elt := range node.Elements {
|
||||||
// key
|
// key
|
||||||
|
@ -296,6 +320,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(OpMap, len(node.Elements)*2)
|
c.emit(OpMap, len(node.Elements)*2)
|
||||||
|
|
||||||
case *ast.SelectorExpr: // selector on RHS side
|
case *ast.SelectorExpr: // selector on RHS side
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -306,6 +331,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(OpIndex)
|
c.emit(OpIndex)
|
||||||
|
|
||||||
case *ast.IndexExpr:
|
case *ast.IndexExpr:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -316,6 +342,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(OpIndex)
|
c.emit(OpIndex)
|
||||||
|
|
||||||
case *ast.SliceExpr:
|
case *ast.SliceExpr:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -338,6 +365,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
c.emit(OpSliceIndex)
|
c.emit(OpSliceIndex)
|
||||||
|
|
||||||
case *ast.FuncLit:
|
case *ast.FuncLit:
|
||||||
c.enterScope()
|
c.enterScope()
|
||||||
|
|
||||||
|
@ -378,6 +406,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
} else {
|
} else {
|
||||||
c.emit(OpConstant, c.addConstant(compiledFunction))
|
c.emit(OpConstant, c.addConstant(compiledFunction))
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.ReturnStmt:
|
case *ast.ReturnStmt:
|
||||||
if c.symbolTable.Parent(true) == nil {
|
if c.symbolTable.Parent(true) == nil {
|
||||||
// outside the function
|
// outside the function
|
||||||
|
@ -396,6 +425,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("multi-value return not implemented")
|
return fmt.Errorf("multi-value return not implemented")
|
||||||
}
|
}
|
||||||
|
|
||||||
case *ast.CallExpr:
|
case *ast.CallExpr:
|
||||||
if err := c.Compile(node.Func); err != nil {
|
if err := c.Compile(node.Func); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -409,6 +439,21 @@ func (c *Compiler) Compile(node ast.Node) error {
|
||||||
|
|
||||||
c.emit(OpCall, len(node.Args))
|
c.emit(OpCall, len(node.Args))
|
||||||
|
|
||||||
|
case *ast.ImportExpr:
|
||||||
|
stdMod, ok := stdmods.Modules[node.ModuleName]
|
||||||
|
if ok {
|
||||||
|
// standard modules contain only globals with no code.
|
||||||
|
// so no need to compile anything
|
||||||
|
c.emit(OpConstant, c.addConstant(stdMod))
|
||||||
|
} else {
|
||||||
|
userMod, err := c.compileModule(node.ModuleName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
c.emit(OpModule, c.addConstant(userMod))
|
||||||
|
}
|
||||||
|
|
||||||
case *ast.ErrorExpr:
|
case *ast.ErrorExpr:
|
||||||
if err := c.Compile(node.Expr); err != nil {
|
if err := c.Compile(node.Expr); err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -428,7 +473,26 @@ func (c *Compiler) Bytecode() *Bytecode {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetModuleLoader sets or replaces the current module loader.
|
||||||
|
func (c *Compiler) SetModuleLoader(moduleLoader ModuleLoader) {
|
||||||
|
c.moduleLoader = moduleLoader
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Compiler) fork(moduleName string, symbolTable *SymbolTable) *Compiler {
|
||||||
|
child := NewCompiler(symbolTable, c.trace)
|
||||||
|
child.moduleName = moduleName // name of the module to compile
|
||||||
|
child.parent = c // parent to set to current compiler
|
||||||
|
child.moduleLoader = c.moduleLoader // share module loader
|
||||||
|
|
||||||
|
return child
|
||||||
|
}
|
||||||
|
|
||||||
func (c *Compiler) addConstant(o objects.Object) int {
|
func (c *Compiler) addConstant(o objects.Object) int {
|
||||||
|
if c.parent != nil {
|
||||||
|
// module compilers will use their parent's constants array
|
||||||
|
return c.parent.addConstant(o)
|
||||||
|
}
|
||||||
|
|
||||||
c.constants = append(c.constants, o)
|
c.constants = append(c.constants, o)
|
||||||
|
|
||||||
if c.trace != nil {
|
if c.trace != nil {
|
||||||
|
|
116
compiler/compiler_module.go
Normal file
116
compiler/compiler_module.go
Normal file
|
@ -0,0 +1,116 @@
|
||||||
|
package compiler
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/compiler/parser"
|
||||||
|
"github.com/d5/tengo/compiler/source"
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
fileSet = source.NewFileSet()
|
||||||
|
)
|
||||||
|
|
||||||
|
func (c *Compiler) compileModule(moduleName string) (*objects.CompiledModule, error) {
|
||||||
|
compiledModule, exists := c.loadCompiledModule(moduleName)
|
||||||
|
if exists {
|
||||||
|
return compiledModule, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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(moduleName); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var err error
|
||||||
|
moduleSrc, err = ioutil.ReadFile(moduleName)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := c.checkCyclicImports(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(moduleName string) error {
|
||||||
|
if c.moduleName == moduleName {
|
||||||
|
return fmt.Errorf("cyclic module import: %s", moduleName)
|
||||||
|
} else if c.parent != nil {
|
||||||
|
return c.parent.checkCyclicImports(moduleName)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Compiler) doCompileModule(moduleName string, src []byte) (*objects.CompiledModule, error) {
|
||||||
|
p := parser.NewParser(fileSet.AddFile(moduleName, -1, len(src)), src, nil)
|
||||||
|
file, err := p.ParseFile()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
symbolTable := NewSymbolTable()
|
||||||
|
globals := make(map[string]int)
|
||||||
|
|
||||||
|
moduleCompiler := c.fork(moduleName, symbolTable)
|
||||||
|
if err := moduleCompiler.Compile(file); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, name := range symbolTable.Names() {
|
||||||
|
symbol, _, _ := symbolTable.Resolve(name)
|
||||||
|
if symbol.Scope == ScopeGlobal {
|
||||||
|
globals[name] = symbol.Index
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.CompiledModule{
|
||||||
|
Instructions: moduleCompiler.Bytecode().Instructions,
|
||||||
|
Globals: globals,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Compiler) loadCompiledModule(moduleName string) (mod *objects.CompiledModule, ok bool) {
|
||||||
|
if c.parent != nil {
|
||||||
|
return c.parent.loadCompiledModule(moduleName)
|
||||||
|
}
|
||||||
|
|
||||||
|
mod, ok = c.compiledModules[moduleName]
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Compiler) storeCompiledModule(moduleName string, module *objects.CompiledModule) {
|
||||||
|
if c.parent != nil {
|
||||||
|
c.parent.storeCompiledModule(moduleName, module)
|
||||||
|
}
|
||||||
|
|
||||||
|
c.compiledModules[moduleName] = module
|
||||||
|
}
|
|
@ -875,11 +875,13 @@ func() {
|
||||||
intObject(0),
|
intObject(0),
|
||||||
intObject(1),
|
intObject(1),
|
||||||
intObject(1))))
|
intObject(1))))
|
||||||
|
|
||||||
|
expectError(t, `import("user1")`) // unknown module name
|
||||||
}
|
}
|
||||||
|
|
||||||
func concat(insts ...[]byte) []byte {
|
func concat(instructions ...[]byte) []byte {
|
||||||
concat := make([]byte, 0)
|
concat := make([]byte, 0)
|
||||||
for _, i := range insts {
|
for _, i := range instructions {
|
||||||
concat = append(concat, i...)
|
concat = append(concat, i...)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -913,6 +915,22 @@ func expect(t *testing.T, input string, expected *compiler.Bytecode) (ok bool) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func expectError(t *testing.T, input string) (ok bool) {
|
||||||
|
_, trace, err := traceCompile(input, nil)
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
if !ok {
|
||||||
|
for _, tr := range trace {
|
||||||
|
t.Log(tr)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
ok = assert.Error(t, err)
|
||||||
|
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
func equalBytecode(t *testing.T, expected, actual *compiler.Bytecode) bool {
|
func equalBytecode(t *testing.T, expected, actual *compiler.Bytecode) bool {
|
||||||
expectedInstructions := strings.Join(compiler.FormatInstructions(expected.Instructions, 0), "\n")
|
expectedInstructions := strings.Join(compiler.FormatInstructions(expected.Instructions, 0), "\n")
|
||||||
actualInstructions := strings.Join(compiler.FormatInstructions(actual.Instructions, 0), "\n")
|
actualInstructions := strings.Join(compiler.FormatInstructions(actual.Instructions, 0), "\n")
|
||||||
|
|
4
compiler/module_loader.go
Normal file
4
compiler/module_loader.go
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
package compiler
|
||||||
|
|
||||||
|
// ModuleLoader should take a module name and return the module data.
|
||||||
|
type ModuleLoader func(moduleName string) ([]byte, error)
|
|
@ -56,6 +56,7 @@ const (
|
||||||
OpIteratorNext // Iterator next
|
OpIteratorNext // Iterator next
|
||||||
OpIteratorKey // Iterator key
|
OpIteratorKey // Iterator key
|
||||||
OpIteratorValue // Iterator value
|
OpIteratorValue // Iterator value
|
||||||
|
OpModule // Module
|
||||||
)
|
)
|
||||||
|
|
||||||
// OpcodeNames is opcode names.
|
// OpcodeNames is opcode names.
|
||||||
|
@ -111,6 +112,7 @@ var OpcodeNames = [...]string{
|
||||||
OpIteratorNext: "ITNXT",
|
OpIteratorNext: "ITNXT",
|
||||||
OpIteratorKey: "ITKEY",
|
OpIteratorKey: "ITKEY",
|
||||||
OpIteratorValue: "ITVAL",
|
OpIteratorValue: "ITVAL",
|
||||||
|
OpModule: "MODULE",
|
||||||
}
|
}
|
||||||
|
|
||||||
// OpcodeOperands is the number of operands.
|
// OpcodeOperands is the number of operands.
|
||||||
|
@ -166,6 +168,7 @@ var OpcodeOperands = [...][]int{
|
||||||
OpIteratorNext: {},
|
OpIteratorNext: {},
|
||||||
OpIteratorKey: {},
|
OpIteratorKey: {},
|
||||||
OpIteratorValue: {},
|
OpIteratorValue: {},
|
||||||
|
OpModule: {2},
|
||||||
}
|
}
|
||||||
|
|
||||||
// ReadOperands reads operands from the bytecode.
|
// ReadOperands reads operands from the bytecode.
|
||||||
|
|
|
@ -334,6 +334,9 @@ func (p *Parser) parseOperand() ast.Expr {
|
||||||
p.next()
|
p.next()
|
||||||
return x
|
return x
|
||||||
|
|
||||||
|
case token.Import:
|
||||||
|
return p.parseImportExpr()
|
||||||
|
|
||||||
case token.LParen:
|
case token.LParen:
|
||||||
lparen := p.pos
|
lparen := p.pos
|
||||||
p.next()
|
p.next()
|
||||||
|
@ -366,6 +369,35 @@ func (p *Parser) parseOperand() ast.Expr {
|
||||||
return &ast.BadExpr{From: pos, To: p.pos}
|
return &ast.BadExpr{From: pos, To: p.pos}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Parser) parseImportExpr() ast.Expr {
|
||||||
|
pos := p.pos
|
||||||
|
|
||||||
|
p.next()
|
||||||
|
|
||||||
|
p.expect(token.LParen)
|
||||||
|
|
||||||
|
if p.token != token.String {
|
||||||
|
p.errorExpected(p.pos, "module name")
|
||||||
|
p.advance(stmtStart)
|
||||||
|
return &ast.BadExpr{From: pos, To: p.pos}
|
||||||
|
}
|
||||||
|
|
||||||
|
// module name
|
||||||
|
moduleName, _ := strconv.Unquote(p.tokenLit)
|
||||||
|
|
||||||
|
expr := &ast.ImportExpr{
|
||||||
|
ModuleName: moduleName,
|
||||||
|
Token: token.Import,
|
||||||
|
TokenPos: pos,
|
||||||
|
}
|
||||||
|
|
||||||
|
p.next()
|
||||||
|
|
||||||
|
p.expect(token.RParen)
|
||||||
|
|
||||||
|
return expr
|
||||||
|
}
|
||||||
|
|
||||||
func (p *Parser) parseCharLit() ast.Expr {
|
func (p *Parser) parseCharLit() ast.Expr {
|
||||||
if n := len(p.tokenLit); n >= 3 {
|
if n := len(p.tokenLit); n >= 3 {
|
||||||
if code, _, _, err := strconv.UnquoteChar(p.tokenLit[1:n-1], '\''); err == nil {
|
if code, _, _, err := strconv.UnquoteChar(p.tokenLit[1:n-1], '\''); err == nil {
|
||||||
|
@ -413,9 +445,9 @@ func (p *Parser) parseArrayLit() ast.Expr {
|
||||||
lbrack := p.expect(token.LBrack)
|
lbrack := p.expect(token.LBrack)
|
||||||
p.exprLevel++
|
p.exprLevel++
|
||||||
|
|
||||||
var elts []ast.Expr
|
var elements []ast.Expr
|
||||||
for p.token != token.RBrack && p.token != token.EOF {
|
for p.token != token.RBrack && p.token != token.EOF {
|
||||||
elts = append(elts, p.parseExpr())
|
elements = append(elements, p.parseExpr())
|
||||||
|
|
||||||
if !p.expectComma(token.RBrack, "array element") {
|
if !p.expectComma(token.RBrack, "array element") {
|
||||||
break
|
break
|
||||||
|
@ -426,7 +458,7 @@ func (p *Parser) parseArrayLit() ast.Expr {
|
||||||
rbrack := p.expect(token.RBrack)
|
rbrack := p.expect(token.RBrack)
|
||||||
|
|
||||||
return &ast.ArrayLit{
|
return &ast.ArrayLit{
|
||||||
Elements: elts,
|
Elements: elements,
|
||||||
LBrack: lbrack,
|
LBrack: lbrack,
|
||||||
RBrack: rbrack,
|
RBrack: rbrack,
|
||||||
}
|
}
|
||||||
|
@ -540,9 +572,9 @@ func (p *Parser) parseStmt() (stmt ast.Stmt) {
|
||||||
|
|
||||||
switch p.token {
|
switch p.token {
|
||||||
case // simple statements
|
case // simple statements
|
||||||
token.Func, token.Error, token.Ident, token.Int, token.Float, token.Char, token.String, token.True, token.False, token.Undefined, token.LParen, // operands
|
token.Func, token.Error, token.Ident, token.Int, token.Float, token.Char, token.String, token.True, token.False,
|
||||||
token.LBrace, token.LBrack, // composite types
|
token.Undefined, token.Import, token.LParen, token.LBrace, token.LBrack,
|
||||||
token.Add, token.Sub, token.Mul, token.And, token.Xor, token.Not: // unary operators
|
token.Add, token.Sub, token.Mul, token.And, token.Xor, token.Not:
|
||||||
s := p.parseSimpleStmt(false)
|
s := p.parseSimpleStmt(false)
|
||||||
p.expectSemi()
|
p.expectSemi()
|
||||||
return s
|
return s
|
||||||
|
@ -926,9 +958,9 @@ func (p *Parser) parseMapLit() *ast.MapLit {
|
||||||
lbrace := p.expect(token.LBrace)
|
lbrace := p.expect(token.LBrace)
|
||||||
p.exprLevel++
|
p.exprLevel++
|
||||||
|
|
||||||
var elts []*ast.MapElementLit
|
var elements []*ast.MapElementLit
|
||||||
for p.token != token.RBrace && p.token != token.EOF {
|
for p.token != token.RBrace && p.token != token.EOF {
|
||||||
elts = append(elts, p.parseMapElementLit())
|
elements = append(elements, p.parseMapElementLit())
|
||||||
|
|
||||||
if !p.expectComma(token.RBrace, "map element") {
|
if !p.expectComma(token.RBrace, "map element") {
|
||||||
break
|
break
|
||||||
|
@ -941,7 +973,7 @@ func (p *Parser) parseMapLit() *ast.MapLit {
|
||||||
return &ast.MapLit{
|
return &ast.MapLit{
|
||||||
LBrace: lbrace,
|
LBrace: lbrace,
|
||||||
RBrace: rbrace,
|
RBrace: rbrace,
|
||||||
Elements: elts,
|
Elements: elements,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ import (
|
||||||
"github.com/d5/tengo/compiler/token"
|
"github.com/d5/tengo/compiler/token"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestImport(t *testing.T) {
|
func TestError(t *testing.T) {
|
||||||
expect(t, `error(1234)`, func(p pfn) []ast.Stmt {
|
expect(t, `error(1234)`, func(p pfn) []ast.Stmt {
|
||||||
return stmts(
|
return stmts(
|
||||||
exprStmt(
|
exprStmt(
|
||||||
|
|
46
compiler/parser/parser_import_test.go
Normal file
46
compiler/parser/parser_import_test.go
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
package parser_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/compiler/ast"
|
||||||
|
"github.com/d5/tengo/compiler/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestImport(t *testing.T) {
|
||||||
|
expect(t, `a := import("mod1")`, func(p pfn) []ast.Stmt {
|
||||||
|
return stmts(
|
||||||
|
assignStmt(
|
||||||
|
exprs(ident("a", p(1, 1))),
|
||||||
|
exprs(importExpr("mod1", p(1, 6))),
|
||||||
|
token.Define, p(1, 3)))
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(t, `import("mod1").var1`, func(p pfn) []ast.Stmt {
|
||||||
|
return stmts(
|
||||||
|
exprStmt(
|
||||||
|
selectorExpr(
|
||||||
|
importExpr("mod1", p(1, 1)),
|
||||||
|
stringLit("var1", p(1, 16)))))
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(t, `import("mod1").func1()`, func(p pfn) []ast.Stmt {
|
||||||
|
return stmts(
|
||||||
|
exprStmt(
|
||||||
|
callExpr(
|
||||||
|
selectorExpr(
|
||||||
|
importExpr("mod1", p(1, 1)),
|
||||||
|
stringLit("func1", p(1, 16))),
|
||||||
|
p(1, 21), p(1, 22))))
|
||||||
|
})
|
||||||
|
|
||||||
|
expect(t, `for x, y in import("mod1") {}`, func(p pfn) []ast.Stmt {
|
||||||
|
return stmts(
|
||||||
|
forInStmt(
|
||||||
|
ident("x", p(1, 5)),
|
||||||
|
ident("y", p(1, 8)),
|
||||||
|
importExpr("mod1", p(1, 13)),
|
||||||
|
blockStmt(p(1, 28), p(1, 29)),
|
||||||
|
p(1, 1)))
|
||||||
|
})
|
||||||
|
}
|
|
@ -188,6 +188,10 @@ func unaryExpr(x ast.Expr, op token.Token, pos source.Pos) *ast.UnaryExpr {
|
||||||
return &ast.UnaryExpr{Expr: x, Token: op, TokenPos: pos}
|
return &ast.UnaryExpr{Expr: x, Token: op, TokenPos: pos}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func importExpr(moduleName string, pos source.Pos) *ast.ImportExpr {
|
||||||
|
return &ast.ImportExpr{ModuleName: moduleName, Token: token.Import, TokenPos: pos}
|
||||||
|
}
|
||||||
|
|
||||||
func exprs(list ...ast.Expr) []ast.Expr {
|
func exprs(list ...ast.Expr) []ast.Expr {
|
||||||
return list
|
return list
|
||||||
}
|
}
|
||||||
|
@ -385,6 +389,10 @@ func equalExpr(t *testing.T, expected, actual ast.Expr) bool {
|
||||||
case *ast.SelectorExpr:
|
case *ast.SelectorExpr:
|
||||||
return equalExpr(t, expected.Expr, actual.(*ast.SelectorExpr).Expr) &&
|
return equalExpr(t, expected.Expr, actual.(*ast.SelectorExpr).Expr) &&
|
||||||
equalExpr(t, expected.Sel, actual.(*ast.SelectorExpr).Sel)
|
equalExpr(t, expected.Sel, actual.(*ast.SelectorExpr).Sel)
|
||||||
|
case *ast.ImportExpr:
|
||||||
|
return assert.Equal(t, expected.ModuleName, actual.(*ast.ImportExpr).ModuleName) &&
|
||||||
|
assert.Equal(t, int(expected.TokenPos), int(actual.(*ast.ImportExpr).TokenPos)) &&
|
||||||
|
assert.Equal(t, expected.Token, actual.(*ast.ImportExpr).Token)
|
||||||
case *ast.ErrorExpr:
|
case *ast.ErrorExpr:
|
||||||
return equalExpr(t, expected.Expr, actual.(*ast.ErrorExpr).Expr) &&
|
return equalExpr(t, expected.Expr, actual.(*ast.ErrorExpr).Expr) &&
|
||||||
assert.Equal(t, int(expected.ErrorPos), int(actual.(*ast.ErrorExpr).ErrorPos)) &&
|
assert.Equal(t, int(expected.ErrorPos), int(actual.(*ast.ErrorExpr).ErrorPos)) &&
|
||||||
|
|
243
compiler/stdmods/func_typedefs.go
Normal file
243
compiler/stdmods/func_typedefs.go
Normal file
|
@ -0,0 +1,243 @@
|
||||||
|
package stdmods
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
|
)
|
||||||
|
|
||||||
|
// FuncAFRF transform a function of 'func(float64) float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFRF(fn func(float64) float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
return &objects.Float{Value: fn(float64(arg.Value))}, nil
|
||||||
|
case *objects.Float:
|
||||||
|
return &objects.Float{Value: fn(arg.Value)}, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncARF transform a function of 'func() float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncARF(fn func() float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 0 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Float{Value: fn()}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAIRF transform a function of 'func(int) float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAIRF(fn func(int) float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
return &objects.Float{Value: fn(int(arg.Value))}, nil
|
||||||
|
case *objects.Float:
|
||||||
|
return &objects.Float{Value: fn(int(arg.Value))}, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAFRI transform a function of 'func(float64) int' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFRI(fn func(float64) int) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
return &objects.Int{Value: int64(fn(float64(arg.Value)))}, nil
|
||||||
|
case *objects.Float:
|
||||||
|
return &objects.Int{Value: int64(fn(arg.Value))}, nil
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAFFRF transform a function of 'func(float64, float64) float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFFRF(fn func(float64, float64) float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 2 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg0, arg1 float64
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg0 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg0 = arg.Value
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
switch arg := args[1].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg1 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg1 = arg.Value
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Float{Value: fn(arg0, arg1)}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAIFRF transform a function of 'func(int, float64) float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAIFRF(fn func(int, float64) float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 2 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg0 int
|
||||||
|
var arg1 float64
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg0 = int(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg0 = int(arg.Value)
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
switch arg := args[1].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg1 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg1 = arg.Value
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Float{Value: fn(arg0, arg1)}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAFIRF transform a function of 'func(float64, int) float64' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFIRF(fn func(float64, int) float64) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 2 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg0 float64
|
||||||
|
var arg1 int
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg0 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg0 = float64(arg.Value)
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
switch arg := args[1].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg1 = int(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg1 = int(arg.Value)
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Float{Value: fn(arg0, arg1)}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAFIRB transform a function of 'func(float64, int) bool' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFIRB(fn func(float64, int) bool) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 2 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg0 float64
|
||||||
|
var arg1 int
|
||||||
|
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg0 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg0 = arg.Value
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
switch arg := args[1].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg1 = int(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg1 = int(arg.Value)
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Bool{Value: fn(arg0, arg1)}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// FuncAFRB transform a function of 'func(float64) bool' signature
|
||||||
|
// into a user function object.
|
||||||
|
func FuncAFRB(fn func(float64) bool) *objects.UserFunction {
|
||||||
|
return &objects.UserFunction{
|
||||||
|
Value: func(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
var arg0 float64
|
||||||
|
switch arg := args[0].(type) {
|
||||||
|
case *objects.Int:
|
||||||
|
arg0 = float64(arg.Value)
|
||||||
|
case *objects.Float:
|
||||||
|
arg0 = arg.Value
|
||||||
|
default:
|
||||||
|
return nil, fmt.Errorf("invalid argument type: %s", arg.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
return &objects.Bool{Value: fn(arg0)}, nil
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
124
compiler/stdmods/func_typedefs_test.go
Normal file
124
compiler/stdmods/func_typedefs_test.go
Normal file
|
@ -0,0 +1,124 @@
|
||||||
|
package stdmods_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/assert"
|
||||||
|
"github.com/d5/tengo/compiler/stdmods"
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestFuncARF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncARF(func() float64 {
|
||||||
|
return 10.0
|
||||||
|
})
|
||||||
|
ret, err := uf.Call()
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 10.0}, ret)
|
||||||
|
ret, err = uf.Call(objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFRF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFRF(func(a float64) float64 {
|
||||||
|
return a
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 10.0})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 10.0}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue, objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAIRF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAIRF(func(a int) float64 {
|
||||||
|
return float64(a)
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Int{Value: 10.0})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 10.0}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue, objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFRI(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFRI(func(a float64) int {
|
||||||
|
return int(a)
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 10.5})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Int{Value: 10}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue, objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFRB(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFRB(func(a float64) bool {
|
||||||
|
return a > 0.0
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 0.1})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Bool{Value: true}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue, objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFFRF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFFRF(func(a, b float64) float64 {
|
||||||
|
return a + b
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 10.0}, &objects.Float{Value: 20.0})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 30.0}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAIFRF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAIFRF(func(a int, b float64) float64 {
|
||||||
|
return float64(a) + b
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Int{Value: 10}, &objects.Float{Value: 20.0})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 30.0}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFIRF(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFIRF(func(a float64, b int) float64 {
|
||||||
|
return a + float64(b)
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 10.0}, &objects.Int{Value: 20})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Float{Value: 30.0}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestFuncAFIRB(t *testing.T) {
|
||||||
|
uf := stdmods.FuncAFIRB(func(a float64, b int) bool {
|
||||||
|
return a < float64(b)
|
||||||
|
})
|
||||||
|
ret, err := uf.Call(&objects.Float{Value: 10.0}, &objects.Int{Value: 20})
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, &objects.Bool{Value: true}, ret)
|
||||||
|
ret, err = uf.Call()
|
||||||
|
assert.Error(t, err)
|
||||||
|
ret, err = uf.Call(objects.TrueValue)
|
||||||
|
assert.Error(t, err)
|
||||||
|
}
|
84
compiler/stdmods/math.go
Normal file
84
compiler/stdmods/math.go
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
package stdmods
|
||||||
|
|
||||||
|
import (
|
||||||
|
"math"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/objects"
|
||||||
|
)
|
||||||
|
|
||||||
|
var mathModule = map[string]objects.Object{
|
||||||
|
"e": &objects.Float{Value: math.E},
|
||||||
|
"pi": &objects.Float{Value: math.Pi},
|
||||||
|
"phi": &objects.Float{Value: math.Phi},
|
||||||
|
"sqrt2": &objects.Float{Value: math.Sqrt2},
|
||||||
|
"sqrtE": &objects.Float{Value: math.SqrtE},
|
||||||
|
"sqrtPi": &objects.Float{Value: math.SqrtPi},
|
||||||
|
"sqrtPhi": &objects.Float{Value: math.SqrtPhi},
|
||||||
|
"ln2": &objects.Float{Value: math.Ln2},
|
||||||
|
"log2E": &objects.Float{Value: math.Log2E},
|
||||||
|
"ln10": &objects.Float{Value: math.Ln10},
|
||||||
|
"log10E": &objects.Float{Value: math.Log10E},
|
||||||
|
"abs": FuncAFRF(math.Abs),
|
||||||
|
"acos": FuncAFRF(math.Acos),
|
||||||
|
"acosh": FuncAFRF(math.Acosh),
|
||||||
|
"asin": FuncAFRF(math.Asin),
|
||||||
|
"asinh": FuncAFRF(math.Asinh),
|
||||||
|
"atan": FuncAFRF(math.Atan),
|
||||||
|
"atan2": FuncAFFRF(math.Atan2),
|
||||||
|
"atanh": FuncAFRF(math.Atanh),
|
||||||
|
"cbrt": FuncAFRF(math.Cbrt),
|
||||||
|
"ceil": FuncAFRF(math.Ceil),
|
||||||
|
"copysign": FuncAFFRF(math.Copysign),
|
||||||
|
"cos": FuncAFRF(math.Cos),
|
||||||
|
"cosh": FuncAFRF(math.Cosh),
|
||||||
|
"dim": FuncAFFRF(math.Dim),
|
||||||
|
"erf": FuncAFRF(math.Erf),
|
||||||
|
"erfc": FuncAFRF(math.Erfc),
|
||||||
|
"erfcinv": FuncAFRF(math.Erfcinv),
|
||||||
|
"erfinv": FuncAFRF(math.Erfinv),
|
||||||
|
"exp": FuncAFRF(math.Exp),
|
||||||
|
"exp2": FuncAFRF(math.Exp2),
|
||||||
|
"expm1": FuncAFRF(math.Expm1),
|
||||||
|
"floor": FuncAFRF(math.Floor),
|
||||||
|
"gamma": FuncAFRF(math.Gamma),
|
||||||
|
"hypot": FuncAFFRF(math.Hypot),
|
||||||
|
"ilogb": FuncAFRI(math.Ilogb),
|
||||||
|
"inf": FuncAIRF(math.Inf),
|
||||||
|
"is_inf": FuncAFIRB(math.IsInf),
|
||||||
|
"is_nan": FuncAFRB(math.IsNaN),
|
||||||
|
"j0": FuncAFRF(math.J0),
|
||||||
|
"j1": FuncAFRF(math.J1),
|
||||||
|
"jn": FuncAIFRF(math.Jn),
|
||||||
|
"ldexp": FuncAFIRF(math.Ldexp),
|
||||||
|
"log": FuncAFRF(math.Log),
|
||||||
|
"log10": FuncAFRF(math.Log10),
|
||||||
|
"log1p": FuncAFRF(math.Log1p),
|
||||||
|
"log2": FuncAFRF(math.Log2),
|
||||||
|
"logb": FuncAFRF(math.Logb),
|
||||||
|
"max": FuncAFFRF(math.Max),
|
||||||
|
"min": FuncAFFRF(math.Min),
|
||||||
|
"mod": FuncAFFRF(math.Mod),
|
||||||
|
"nan": FuncARF(math.NaN),
|
||||||
|
"nextafter": FuncAFFRF(math.Nextafter),
|
||||||
|
"pow": FuncAFFRF(math.Pow),
|
||||||
|
"pow10": FuncAIRF(math.Pow10),
|
||||||
|
"remainder": FuncAFFRF(math.Remainder),
|
||||||
|
"round": FuncAFRF(math.Round),
|
||||||
|
"round_to_even": FuncAFRF(math.RoundToEven),
|
||||||
|
"signbit": FuncAFRB(math.Signbit),
|
||||||
|
"sin": FuncAFRF(math.Sin),
|
||||||
|
"sinh": FuncAFRF(math.Sinh),
|
||||||
|
"sqrt": FuncAFRF(math.Sqrt),
|
||||||
|
"tan": FuncAFRF(math.Tan),
|
||||||
|
"tanh": FuncAFRF(math.Tanh),
|
||||||
|
"runct": FuncAFRF(math.Trunc),
|
||||||
|
"y0": FuncAFRF(math.Y0),
|
||||||
|
"y1": FuncAFRF(math.Y1),
|
||||||
|
"yn": FuncAIFRF(math.Yn),
|
||||||
|
// TODO: functions that have multiple returns
|
||||||
|
// Should these be tuple assignment? Or Map return?
|
||||||
|
//"frexp": nil,
|
||||||
|
//"lgamma": nil,
|
||||||
|
//"modf": nil,
|
||||||
|
//"sincos": nil,
|
||||||
|
}
|
8
compiler/stdmods/stdmods.go
Normal file
8
compiler/stdmods/stdmods.go
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
package stdmods
|
||||||
|
|
||||||
|
import "github.com/d5/tengo/objects"
|
||||||
|
|
||||||
|
// Modules contain the standard modules.
|
||||||
|
var Modules = map[string]*objects.ModuleMap{
|
||||||
|
"math": {Value: mathModule},
|
||||||
|
}
|
|
@ -82,6 +82,7 @@ const (
|
||||||
False
|
False
|
||||||
In
|
In
|
||||||
Undefined
|
Undefined
|
||||||
|
Import
|
||||||
_keywordEnd
|
_keywordEnd
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -156,6 +157,7 @@ var tokens = [...]string{
|
||||||
False: "false",
|
False: "false",
|
||||||
In: "in",
|
In: "in",
|
||||||
Undefined: "undefined",
|
Undefined: "undefined",
|
||||||
|
Import: "import",
|
||||||
}
|
}
|
||||||
|
|
||||||
func (tok Token) String() string {
|
func (tok Token) String() string {
|
||||||
|
|
|
@ -7,7 +7,7 @@ import (
|
||||||
// append(src, items...)
|
// append(src, items...)
|
||||||
func builtinAppend(args ...Object) (Object, error) {
|
func builtinAppend(args ...Object) (Object, error) {
|
||||||
if len(args) < 2 {
|
if len(args) < 2 {
|
||||||
return nil, fmt.Errorf("not enough arguments in call to append")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
|
|
@ -1,13 +1,12 @@
|
||||||
package objects
|
package objects
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
)
|
)
|
||||||
|
|
||||||
func builtinString(args ...Object) (Object, error) {
|
func builtinString(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
@ -22,7 +21,7 @@ func builtinString(args ...Object) (Object, error) {
|
||||||
|
|
||||||
func builtinInt(args ...Object) (Object, error) {
|
func builtinInt(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
@ -49,7 +48,7 @@ func builtinInt(args ...Object) (Object, error) {
|
||||||
|
|
||||||
func builtinFloat(args ...Object) (Object, error) {
|
func builtinFloat(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
@ -69,7 +68,7 @@ func builtinFloat(args ...Object) (Object, error) {
|
||||||
|
|
||||||
func builtinBool(args ...Object) (Object, error) {
|
func builtinBool(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
@ -82,7 +81,7 @@ func builtinBool(args ...Object) (Object, error) {
|
||||||
|
|
||||||
func builtinChar(args ...Object) (Object, error) {
|
func builtinChar(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
|
|
@ -1,11 +1,8 @@
|
||||||
package objects
|
package objects
|
||||||
|
|
||||||
import "errors"
|
|
||||||
|
|
||||||
func builtinCopy(args ...Object) (Object, error) {
|
func builtinCopy(args ...Object) (Object, error) {
|
||||||
// TODO: should multi arguments later?
|
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, errors.New("wrong number of arguments")
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
return args[0].Copy(), nil
|
return args[0].Copy(), nil
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
|
|
||||||
// BuiltinFunction represents a builtin function.
|
// BuiltinFunction represents a builtin function.
|
||||||
type BuiltinFunction struct {
|
type BuiltinFunction struct {
|
||||||
Value BuiltinFunc
|
Value CallableFunc
|
||||||
}
|
}
|
||||||
|
|
||||||
// TypeName returns the name of the type.
|
// TypeName returns the name of the type.
|
||||||
|
|
|
@ -6,7 +6,7 @@ import (
|
||||||
|
|
||||||
func builtinLen(args ...Object) (Object, error) {
|
func builtinLen(args ...Object) (Object, error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return nil, fmt.Errorf("wrong number of arguments (got=%d, want=1)", len(args))
|
return nil, ErrWrongNumArguments
|
||||||
}
|
}
|
||||||
|
|
||||||
switch arg := args[0].(type) {
|
switch arg := args[0].(type) {
|
||||||
|
|
|
@ -1,12 +1,9 @@
|
||||||
package objects
|
package objects
|
||||||
|
|
||||||
// BuiltinFunc is a function signature for the builtin functions.
|
|
||||||
type BuiltinFunc func(args ...Object) (ret Object, err error)
|
|
||||||
|
|
||||||
// Builtins contains all known builtin functions.
|
// Builtins contains all known builtin functions.
|
||||||
var Builtins = []struct {
|
var Builtins = []struct {
|
||||||
Name string
|
Name string
|
||||||
Func BuiltinFunc
|
Func CallableFunc
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
Name: "print",
|
Name: "print",
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
package objects
|
package objects
|
||||||
|
|
||||||
// Callable repesents an object that can be called like a function.
|
// Callable represents an object that can be called like a function.
|
||||||
type Callable interface {
|
type Callable interface {
|
||||||
// Call should take an arbitrary number of arguments
|
// Call should take an arbitrary number of arguments
|
||||||
// and returns a return value and/or an error,
|
// and returns a return value and/or an error,
|
||||||
|
|
4
objects/callable_func.go
Normal file
4
objects/callable_func.go
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
// CallableFunc is a function signature for the callable functions.
|
||||||
|
type CallableFunc func(args ...Object) (ret Object, err error)
|
50
objects/compiled_module.go
Normal file
50
objects/compiled_module.go
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/d5/tengo/compiler/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
// CompiledModule represents a compiled module.
|
||||||
|
type CompiledModule struct {
|
||||||
|
Instructions []byte // compiled instructions
|
||||||
|
Globals map[string]int // global variable name-to-index map
|
||||||
|
}
|
||||||
|
|
||||||
|
// TypeName returns the name of the type.
|
||||||
|
func (o *CompiledModule) TypeName() string {
|
||||||
|
return "compiled-module"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *CompiledModule) String() string {
|
||||||
|
return "<compiled-module>"
|
||||||
|
}
|
||||||
|
|
||||||
|
// BinaryOp returns another object that is the result of
|
||||||
|
// a given binary operator and a right-hand side object.
|
||||||
|
func (o *CompiledModule) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||||
|
return nil, ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy returns a copy of the type.
|
||||||
|
func (o *CompiledModule) Copy() Object {
|
||||||
|
globals := make(map[string]int, len(o.Globals))
|
||||||
|
for name, index := range o.Globals {
|
||||||
|
globals[name] = index
|
||||||
|
}
|
||||||
|
|
||||||
|
return &CompiledModule{
|
||||||
|
Instructions: append([]byte{}, o.Instructions...),
|
||||||
|
Globals: globals,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsFalsy returns true if the value of the type is falsy.
|
||||||
|
func (o *CompiledModule) IsFalsy() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equals returns true if the value of the type
|
||||||
|
// is equal to the value of another object.
|
||||||
|
func (o *CompiledModule) Equals(x Object) bool {
|
||||||
|
return false
|
||||||
|
}
|
|
@ -4,3 +4,6 @@ import "errors"
|
||||||
|
|
||||||
// ErrInvalidOperator represents an error for invalid operator usage.
|
// ErrInvalidOperator represents an error for invalid operator usage.
|
||||||
var ErrInvalidOperator = errors.New("invalid operator")
|
var ErrInvalidOperator = errors.New("invalid operator")
|
||||||
|
|
||||||
|
// ErrWrongNumArguments represents a wrong number of arguments error.
|
||||||
|
var ErrWrongNumArguments = errors.New("wrong number of arguments")
|
||||||
|
|
77
objects/module_map.go
Normal file
77
objects/module_map.go
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/compiler/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
// ModuleMap represents a module map object.
|
||||||
|
type ModuleMap struct {
|
||||||
|
Value map[string]Object
|
||||||
|
}
|
||||||
|
|
||||||
|
// TypeName returns the name of the type.
|
||||||
|
func (o *ModuleMap) TypeName() string {
|
||||||
|
return "module"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *ModuleMap) String() string {
|
||||||
|
var pairs []string
|
||||||
|
for k, v := range o.Value {
|
||||||
|
pairs = append(pairs, fmt.Sprintf("%s: %s", k, v.String()))
|
||||||
|
}
|
||||||
|
|
||||||
|
return fmt.Sprintf("{%s}", strings.Join(pairs, ", "))
|
||||||
|
}
|
||||||
|
|
||||||
|
// BinaryOp returns another object that is the result of
|
||||||
|
// a given binary operator and a right-hand side object.
|
||||||
|
func (o *ModuleMap) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||||
|
return nil, ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy returns a copy of the type.
|
||||||
|
func (o *ModuleMap) Copy() Object {
|
||||||
|
c := make(map[string]Object)
|
||||||
|
for k, v := range o.Value {
|
||||||
|
c[k] = v.Copy()
|
||||||
|
}
|
||||||
|
|
||||||
|
return &ModuleMap{Value: c}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsFalsy returns true if the value of the type is falsy.
|
||||||
|
func (o *ModuleMap) IsFalsy() bool {
|
||||||
|
return len(o.Value) == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get returns the value for the given key.
|
||||||
|
func (o *ModuleMap) Get(key string) (Object, bool) {
|
||||||
|
val, ok := o.Value[key]
|
||||||
|
|
||||||
|
return val, ok
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equals returns true if the value of the type
|
||||||
|
// is equal to the value of another object.
|
||||||
|
func (o *ModuleMap) Equals(x Object) bool {
|
||||||
|
t, ok := x.(*ModuleMap)
|
||||||
|
if !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(o.Value) != len(t.Value) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for k, v := range o.Value {
|
||||||
|
tv := t.Value[k]
|
||||||
|
if !v.Equals(tv) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
76
objects/module_map_iterator.go
Normal file
76
objects/module_map_iterator.go
Normal file
|
@ -0,0 +1,76 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
import "github.com/d5/tengo/compiler/token"
|
||||||
|
|
||||||
|
// ModuleMapIterator represents an iterator for the module map.
|
||||||
|
type ModuleMapIterator struct {
|
||||||
|
v map[string]Object
|
||||||
|
k []string
|
||||||
|
i int
|
||||||
|
l int
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewModuleMapIterator creates a module iterator.
|
||||||
|
func NewModuleMapIterator(v *ModuleMap) Iterator {
|
||||||
|
var keys []string
|
||||||
|
for k := range v.Value {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &ModuleMapIterator{
|
||||||
|
v: v.Value,
|
||||||
|
k: keys,
|
||||||
|
l: len(keys),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TypeName returns the name of the type.
|
||||||
|
func (i *ModuleMapIterator) TypeName() string {
|
||||||
|
return "module-iterator"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *ModuleMapIterator) String() string {
|
||||||
|
return "<module-iterator>"
|
||||||
|
}
|
||||||
|
|
||||||
|
// BinaryOp returns another object that is the result of
|
||||||
|
// a given binary operator and a right-hand side object.
|
||||||
|
func (i *ModuleMapIterator) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||||
|
return nil, ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsFalsy returns true if the value of the type is falsy.
|
||||||
|
func (i *ModuleMapIterator) IsFalsy() bool {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equals returns true if the value of the type
|
||||||
|
// is equal to the value of another object.
|
||||||
|
func (i *ModuleMapIterator) Equals(Object) bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy returns a copy of the type.
|
||||||
|
func (i *ModuleMapIterator) Copy() Object {
|
||||||
|
return &ModuleMapIterator{v: i.v, k: i.k, i: i.i, l: i.l}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Next returns true if there are more elements to iterate.
|
||||||
|
func (i *ModuleMapIterator) Next() bool {
|
||||||
|
i.i++
|
||||||
|
return i.i <= i.l
|
||||||
|
}
|
||||||
|
|
||||||
|
// Key returns the key or index value of the current element.
|
||||||
|
func (i *ModuleMapIterator) Key() Object {
|
||||||
|
k := i.k[i.i-1]
|
||||||
|
|
||||||
|
return &String{Value: k}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Value returns the value of the current element.
|
||||||
|
func (i *ModuleMapIterator) Value() Object {
|
||||||
|
k := i.k[i.i-1]
|
||||||
|
|
||||||
|
return i.v[k]
|
||||||
|
}
|
|
@ -23,19 +23,13 @@ func (o *String) String() string {
|
||||||
// BinaryOp returns another object that is the result of
|
// BinaryOp returns another object that is the result of
|
||||||
// a given binary operator and a right-hand side object.
|
// a given binary operator and a right-hand side object.
|
||||||
func (o *String) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
func (o *String) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||||
|
switch op {
|
||||||
|
case token.Add:
|
||||||
switch rhs := rhs.(type) {
|
switch rhs := rhs.(type) {
|
||||||
case *String:
|
case *String:
|
||||||
switch op {
|
|
||||||
case token.Add:
|
|
||||||
if rhs.Value == "" {
|
|
||||||
return o, nil
|
|
||||||
}
|
|
||||||
return &String{Value: o.Value + rhs.Value}, nil
|
return &String{Value: o.Value + rhs.Value}, nil
|
||||||
}
|
default:
|
||||||
case *Char:
|
return &String{Value: o.Value + rhs.String()}, nil
|
||||||
switch op {
|
|
||||||
case token.Add:
|
|
||||||
return &String{Value: o.Value + string(rhs.Value)}, nil
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
46
objects/user_function.go
Normal file
46
objects/user_function.go
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
package objects
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/d5/tengo/compiler/token"
|
||||||
|
)
|
||||||
|
|
||||||
|
// UserFunction represents a user function.
|
||||||
|
type UserFunction struct {
|
||||||
|
Value CallableFunc
|
||||||
|
}
|
||||||
|
|
||||||
|
// TypeName returns the name of the type.
|
||||||
|
func (o *UserFunction) TypeName() string {
|
||||||
|
return "user-function"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *UserFunction) String() string {
|
||||||
|
return "<user-function>"
|
||||||
|
}
|
||||||
|
|
||||||
|
// BinaryOp returns another object that is the result of
|
||||||
|
// a given binary operator and a right-hand side object.
|
||||||
|
func (o *UserFunction) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||||
|
return nil, ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy returns a copy of the type.
|
||||||
|
func (o *UserFunction) Copy() Object {
|
||||||
|
return &UserFunction{Value: o.Value}
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsFalsy returns true if the value of the type is falsy.
|
||||||
|
func (o *UserFunction) IsFalsy() bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Equals returns true if the value of the type
|
||||||
|
// is equal to the value of another object.
|
||||||
|
func (o *UserFunction) Equals(x Object) bool {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// Call executes a builtin function.
|
||||||
|
func (o *UserFunction) Call(args ...Object) (Object, error) {
|
||||||
|
return o.Value(args...)
|
||||||
|
}
|
|
@ -625,7 +625,26 @@ func (v *VM) Run() error {
|
||||||
case *objects.Map:
|
case *objects.Map:
|
||||||
key, ok := (*index).(*objects.String)
|
key, ok := (*index).(*objects.String)
|
||||||
if !ok {
|
if !ok {
|
||||||
return fmt.Errorf("non-string map key: %s", left.TypeName())
|
return fmt.Errorf("non-string key: %s", left.TypeName())
|
||||||
|
}
|
||||||
|
|
||||||
|
var res = objects.UndefinedValue
|
||||||
|
val, ok := left.Value[key.Value]
|
||||||
|
if ok {
|
||||||
|
res = val
|
||||||
|
}
|
||||||
|
|
||||||
|
if v.sp >= StackSize {
|
||||||
|
return ErrStackOverflow
|
||||||
|
}
|
||||||
|
|
||||||
|
v.stack[v.sp] = &res
|
||||||
|
v.sp++
|
||||||
|
|
||||||
|
case *objects.ModuleMap:
|
||||||
|
key, ok := (*index).(*objects.String)
|
||||||
|
if !ok {
|
||||||
|
return fmt.Errorf("non-string key: %s", left.TypeName())
|
||||||
}
|
}
|
||||||
|
|
||||||
var res = objects.UndefinedValue
|
var res = objects.UndefinedValue
|
||||||
|
@ -986,6 +1005,8 @@ func (v *VM) Run() error {
|
||||||
iterator = objects.NewArrayIterator(dst)
|
iterator = objects.NewArrayIterator(dst)
|
||||||
case *objects.Map:
|
case *objects.Map:
|
||||||
iterator = objects.NewMapIterator(dst)
|
iterator = objects.NewMapIterator(dst)
|
||||||
|
case *objects.ModuleMap:
|
||||||
|
iterator = objects.NewModuleMapIterator(dst)
|
||||||
case *objects.String:
|
case *objects.String:
|
||||||
iterator = objects.NewStringIterator(dst)
|
iterator = objects.NewStringIterator(dst)
|
||||||
default:
|
default:
|
||||||
|
@ -1041,6 +1062,14 @@ func (v *VM) Run() error {
|
||||||
v.stack[v.sp] = &val
|
v.stack[v.sp] = &val
|
||||||
v.sp++
|
v.sp++
|
||||||
|
|
||||||
|
case compiler.OpModule:
|
||||||
|
cidx := compiler.ReadUint16(v.curInsts[ip+1:])
|
||||||
|
v.curFrame.ip += 2
|
||||||
|
|
||||||
|
if err := v.importModule(v.constants[cidx].(*objects.CompiledModule)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("unknown opcode: %d", v.curInsts[ip])
|
return fmt.Errorf("unknown opcode: %d", v.curInsts[ip])
|
||||||
}
|
}
|
||||||
|
@ -1184,6 +1213,35 @@ func (v *VM) callFunction(fn *objects.CompiledFunction, freeVars []*objects.Obje
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: should reuse *objects.ModuleMap for the same imports?
|
||||||
|
func (v *VM) importModule(compiledModule *objects.CompiledModule) error {
|
||||||
|
// import module is basically to create a new instance of VM
|
||||||
|
// and run the module code and retrieve all global variables after execution.
|
||||||
|
moduleVM := NewVM(&compiler.Bytecode{
|
||||||
|
Instructions: compiledModule.Instructions,
|
||||||
|
Constants: v.constants,
|
||||||
|
}, nil)
|
||||||
|
if err := moduleVM.Run(); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
mmValue := make(map[string]objects.Object)
|
||||||
|
for name, index := range compiledModule.Globals {
|
||||||
|
mmValue[name] = *moduleVM.globals[index]
|
||||||
|
}
|
||||||
|
|
||||||
|
var mm objects.Object = &objects.ModuleMap{Value: mmValue}
|
||||||
|
|
||||||
|
if v.sp >= StackSize {
|
||||||
|
return ErrStackOverflow
|
||||||
|
}
|
||||||
|
|
||||||
|
v.stack[v.sp] = &mm
|
||||||
|
v.sp++
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func selectorAssign(dst, src *objects.Object, selectors []interface{}) error {
|
func selectorAssign(dst, src *objects.Object, selectors []interface{}) error {
|
||||||
numSel := len(selectors)
|
numSel := len(selectors)
|
||||||
|
|
||||||
|
|
|
@ -5,15 +5,21 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestForIn(t *testing.T) {
|
func TestForIn(t *testing.T) {
|
||||||
expect(t, `for i, x in [1, 2, 3] { out += i + x }`, 9)
|
// array
|
||||||
expect(t, `func() { for i, x in [1, 2, 3] { out += i + x } }()`, 9)
|
expect(t, `for x in [1, 2, 3] { out += x }`, 6) // value
|
||||||
|
expect(t, `for i, x in [1, 2, 3] { out += i + x }`, 9) // index, value
|
||||||
|
expect(t, `func() { for i, x in [1, 2, 3] { out += i + x } }()`, 9) // index, value
|
||||||
|
expect(t, `for i, _ in [1, 2, 3] { out += i }`, 3) // index, _
|
||||||
|
expect(t, `func() { for i, _ in [1, 2, 3] { out += i } }()`, 3) // index, _
|
||||||
|
|
||||||
expect(t, `for i, _ in [1, 2, 3] { out += i }`, 3)
|
// map
|
||||||
expect(t, `func() { for i, _ in [1, 2, 3] { out += i } }()`, 3)
|
expect(t, `for v in {a:2,b:3,c:4} { out += v }`, 9) // value
|
||||||
|
expect(t, `for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } }`, "b") // key, value
|
||||||
expect(t, `for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } }`, "b")
|
expect(t, `for k, _ in {a:2} { out += k }`, "a") // key, _
|
||||||
expect(t, `func() { for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } } }()`, "b")
|
expect(t, `for _, v in {a:2,b:3,c:4} { out += v }`, 9) // _, value
|
||||||
|
expect(t, `func() { for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } } }()`, "b") // key, value
|
||||||
|
|
||||||
|
// string
|
||||||
expect(t, `for c in "abcde" { out += c }`, "abcde")
|
expect(t, `for c in "abcde" { out += c }`, "abcde")
|
||||||
expect(t, `for i, c in "abcde" { if i == 2 { continue }; out += c }`, "abde")
|
expect(t, `for i, c in "abcde" { if i == 2 { continue }; out += c }`, "abde")
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,10 +10,13 @@ func TestIncDec(t *testing.T) {
|
||||||
expect(t, `a := 0; a++; out = a`, 1)
|
expect(t, `a := 0; a++; out = a`, 1)
|
||||||
expect(t, `a := 0; a++; a--; out = a`, 0)
|
expect(t, `a := 0; a++; a--; out = a`, 0)
|
||||||
|
|
||||||
|
// this seems strange but it works because 'a += b' is
|
||||||
|
// translated into 'a = a + b' and string type takes other types for + operator.
|
||||||
|
expect(t, `a := "foo"; a++; out = a`, "foo1")
|
||||||
|
expectError(t, `a := "foo"; a--`)
|
||||||
|
|
||||||
expectError(t, `a++`) // not declared
|
expectError(t, `a++`) // not declared
|
||||||
expectError(t, `a--`) // not declared
|
expectError(t, `a--`) // not declared
|
||||||
expectError(t, `a := "foo"; a++`) // invalid operand
|
|
||||||
//expectError(t, `a := 0; b := a++`) // inc-dec is statement not expression <- parser error
|
//expectError(t, `a := 0; b := a++`) // inc-dec is statement not expression <- parser error
|
||||||
|
|
||||||
expectError(t, `4++`)
|
expectError(t, `4++`)
|
||||||
}
|
}
|
||||||
|
|
82
runtime/vm_module_test.go
Normal file
82
runtime/vm_module_test.go
Normal file
|
@ -0,0 +1,82 @@
|
||||||
|
package runtime_test
|
||||||
|
|
||||||
|
import "testing"
|
||||||
|
|
||||||
|
func TestModule(t *testing.T) {
|
||||||
|
// stdmods
|
||||||
|
expect(t, `math := import("math"); out = math.abs(1)`, 1.0)
|
||||||
|
expect(t, `math := import("math"); out = math.abs(-1)`, 1.0)
|
||||||
|
expect(t, `math := import("math"); out = math.abs(1.0)`, 1.0)
|
||||||
|
expect(t, `math := import("math"); out = math.abs(-1.0)`, 1.0)
|
||||||
|
|
||||||
|
// user modules
|
||||||
|
expectWithUserModules(t, `out = import("mod1").bar()`, 5.0, map[string]string{
|
||||||
|
"mod1": `bar := func() { return 5.0 }`,
|
||||||
|
})
|
||||||
|
// (main) -> mod1 -> mod2
|
||||||
|
expectWithUserModules(t, `out = import("mod1").mod2.bar()`, 5.0, map[string]string{
|
||||||
|
"mod1": `mod2 := import("mod2")`,
|
||||||
|
"mod2": `bar := func() { return 5.0 }`,
|
||||||
|
})
|
||||||
|
// (main) -> mod1 -> mod2
|
||||||
|
// -> mod2
|
||||||
|
expectWithUserModules(t, `import("mod1"); out = import("mod2").bar()`, 5.0, map[string]string{
|
||||||
|
"mod1": `mod2 := import("mod2")`,
|
||||||
|
"mod2": `bar := func() { return 5.0 }`,
|
||||||
|
})
|
||||||
|
// (main) -> mod1 -> mod2 -> mod3
|
||||||
|
// -> mod2 -> mod3
|
||||||
|
expectWithUserModules(t, `import("mod1"); out = import("mod2").mod3.bar()`, 5.0, map[string]string{
|
||||||
|
"mod1": `mod2 := import("mod2")`,
|
||||||
|
"mod2": `mod3 := import("mod3")`,
|
||||||
|
"mod3": `bar := func() { return 5.0 }`,
|
||||||
|
})
|
||||||
|
|
||||||
|
// cyclic imports
|
||||||
|
// (main) -> mod1 -> mod2 -> mod1
|
||||||
|
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
||||||
|
"mod1": `import("mod2")`,
|
||||||
|
"mod2": `import("mod1")`,
|
||||||
|
})
|
||||||
|
// (main) -> mod1 -> mod2 -> mod3 -> mod1
|
||||||
|
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
||||||
|
"mod1": `import("mod2")`,
|
||||||
|
"mod2": `import("mod3")`,
|
||||||
|
"mod3": `import("mod1")`,
|
||||||
|
})
|
||||||
|
// (main) -> mod1 -> mod2 -> mod3 -> mod2
|
||||||
|
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
||||||
|
"mod1": `import("mod2")`,
|
||||||
|
"mod2": `import("mod3")`,
|
||||||
|
"mod3": `import("mod2")`,
|
||||||
|
})
|
||||||
|
|
||||||
|
// unknown modules
|
||||||
|
expectErrorWithUserModules(t, `import("mod0")`, map[string]string{
|
||||||
|
"mod1": `a := 5`,
|
||||||
|
})
|
||||||
|
expectErrorWithUserModules(t, `import("mod1")`, map[string]string{
|
||||||
|
"mod1": `import("mod2")`,
|
||||||
|
})
|
||||||
|
|
||||||
|
// for-in
|
||||||
|
expectWithUserModules(t, `for _, n in import("mod1") { out += n }`, 6, map[string]string{
|
||||||
|
"mod1": `a := 1; b := 2; c := 3`,
|
||||||
|
})
|
||||||
|
expectWithUserModules(t, `for k, _ in import("mod1") { out += k }`, "a", map[string]string{
|
||||||
|
"mod1": `a := 1`, // only 1 global variable because module map does not sort the keys
|
||||||
|
})
|
||||||
|
|
||||||
|
// mutating global variables inside the module does not affect exported values
|
||||||
|
expectWithUserModules(t, `m1 := import("mod1"); m1.mutate(); out = m1.a`, 3, map[string]string{
|
||||||
|
"mod1": `a := 3; mutate := func() { a = 10 }`,
|
||||||
|
})
|
||||||
|
|
||||||
|
// module map is immutable
|
||||||
|
expectErrorWithUserModules(t, `m1 := import("mod1"); m1.a = 5`, map[string]string{
|
||||||
|
"mod1": `a := 3`,
|
||||||
|
})
|
||||||
|
expectErrorWithUserModules(t, `m1 := import("mod1"); m1.a.b = 5`, map[string]string{
|
||||||
|
"mod1": `a := {b: 3}`,
|
||||||
|
})
|
||||||
|
}
|
|
@ -42,5 +42,22 @@ func TestString(t *testing.T) {
|
||||||
expectError(t, fmt.Sprintf("%s[:%d]", strStr, strLen+1))
|
expectError(t, fmt.Sprintf("%s[:%d]", strStr, strLen+1))
|
||||||
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 2, 1))
|
expectError(t, fmt.Sprintf("%s[%d:%d]", strStr, 2, 1))
|
||||||
|
|
||||||
|
// string concatenation with other types
|
||||||
|
expect(t, `out = "foo" + 1`, "foo1")
|
||||||
|
// Float.String() returns the smallest number of digits
|
||||||
|
// 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.5`, "foo1.5")
|
||||||
|
expect(t, `out = "foo" + true`, "footrue")
|
||||||
|
expect(t, `out = "foo" + 'X'`, "fooX")
|
||||||
|
expect(t, `out = "foo" + error(5)`, "fooerror: 5")
|
||||||
|
expect(t, `out = "foo" + undefined`, "foo<undefined>")
|
||||||
|
expect(t, `out = "foo" + [1,2,3]`, "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
|
||||||
|
expect(t, `out = "foo"; out += 1.5`, "foo1.5")
|
||||||
|
// string concats works only when string is LHS
|
||||||
|
expectError(t, `1 + "foo"`)
|
||||||
|
|
||||||
expectError(t, `"foo" - "bar"`)
|
expectError(t, `"foo" - "bar"`)
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,6 +24,10 @@ type MAP = map[string]interface{}
|
||||||
type ARR = []interface{}
|
type ARR = []interface{}
|
||||||
|
|
||||||
func expect(t *testing.T, input string, expected interface{}) {
|
func expect(t *testing.T, input string, expected interface{}) {
|
||||||
|
expectWithUserModules(t, input, expected, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func expectWithUserModules(t *testing.T, input string, expected interface{}, userModules map[string]string) {
|
||||||
// parse
|
// parse
|
||||||
file := parse(t, input)
|
file := parse(t, input)
|
||||||
if file == nil {
|
if file == nil {
|
||||||
|
@ -31,10 +35,14 @@ func expect(t *testing.T, input string, expected interface{}) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// compiler/VM
|
// compiler/VM
|
||||||
runVM(t, file, expected)
|
runVM(t, file, expected, userModules)
|
||||||
}
|
}
|
||||||
|
|
||||||
func expectError(t *testing.T, input string) {
|
func expectError(t *testing.T, input string) {
|
||||||
|
expectErrorWithUserModules(t, input, nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
func expectErrorWithUserModules(t *testing.T, input string, userModules map[string]string) {
|
||||||
// parse
|
// parse
|
||||||
program := parse(t, input)
|
program := parse(t, input)
|
||||||
if program == nil {
|
if program == nil {
|
||||||
|
@ -42,15 +50,15 @@ func expectError(t *testing.T, input string) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// compiler/VM
|
// compiler/VM
|
||||||
runVMError(t, program)
|
runVMError(t, program, userModules)
|
||||||
}
|
}
|
||||||
|
|
||||||
func runVM(t *testing.T, file *ast.File, expected interface{}) (ok bool) {
|
func runVM(t *testing.T, file *ast.File, expected interface{}, userModules map[string]string) (ok bool) {
|
||||||
expectedObj := toObject(expected)
|
expectedObj := toObject(expected)
|
||||||
|
|
||||||
res, trace, err := traceCompileRun(file, map[string]objects.Object{
|
res, trace, err := traceCompileRun(file, map[string]objects.Object{
|
||||||
testOut: objectZeroCopy(expectedObj),
|
testOut: objectZeroCopy(expectedObj),
|
||||||
})
|
}, userModules)
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
if !ok {
|
if !ok {
|
||||||
|
@ -68,8 +76,8 @@ func runVM(t *testing.T, file *ast.File, expected interface{}) (ok bool) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: should differentiate compile-time error, runtime error, and, error object returned
|
// TODO: should differentiate compile-time error, runtime error, and, error object returned
|
||||||
func runVMError(t *testing.T, file *ast.File) (ok bool) {
|
func runVMError(t *testing.T, file *ast.File, userModules map[string]string) (ok bool) {
|
||||||
_, trace, err := traceCompileRun(file, nil)
|
_, trace, err := traceCompileRun(file, nil, userModules)
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
if !ok {
|
if !ok {
|
||||||
|
@ -132,7 +140,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) (res map[string]objects.Object, trace []string, err error) {
|
func traceCompileRun(file *ast.File, symbols map[string]objects.Object, userModules map[string]string) (res map[string]objects.Object, trace []string, err error) {
|
||||||
var v *runtime.VM
|
var v *runtime.VM
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
|
@ -169,6 +177,13 @@ func traceCompileRun(file *ast.File, symbols map[string]objects.Object) (res map
|
||||||
|
|
||||||
tr := &tracer{}
|
tr := &tracer{}
|
||||||
c := compiler.NewCompiler(symTable, tr)
|
c := compiler.NewCompiler(symTable, 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 {
|
||||||
|
|
Loading…
Reference in a new issue