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)
|
||||
|
||||
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
|
||||
|
||||
|
@ -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
|
||||
import (
|
||||
|
@ -380,10 +400,9 @@ tengo
|
|||
|
||||
Development roadmap for Tengo:
|
||||
|
||||
- Module system _(or packages)_
|
||||
- Standard libraries
|
||||
- Standard libraries _(modules)_
|
||||
- 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
|
||||
- Performance improvements
|
||||
- Syntax highlighter for IDEs
|
||||
|
|
|
@ -26,10 +26,10 @@ func (e *ArrayLit) End() source.Pos {
|
|||
}
|
||||
|
||||
func (e *ArrayLit) String() string {
|
||||
var elts []string
|
||||
var elements []string
|
||||
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"
|
||||
|
||||
// BoolLit represetns a boolean literal.
|
||||
// BoolLit represents a boolean literal.
|
||||
type BoolLit struct {
|
||||
Value bool
|
||||
ValuePos source.Pos
|
||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
|||
|
||||
import "github.com/d5/tengo/compiler/source"
|
||||
|
||||
// ForStmt represetns a for statement.
|
||||
// ForStmt represents a for statement.
|
||||
type ForStmt struct {
|
||||
ForPos source.Pos
|
||||
Init Stmt
|
||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
|||
|
||||
import "github.com/d5/tengo/compiler/source"
|
||||
|
||||
// FuncType represetns a function type definition.
|
||||
// FuncType represents a function type definition.
|
||||
type FuncType struct {
|
||||
FuncPos source.Pos
|
||||
Params *IdentList
|
||||
|
|
|
@ -2,7 +2,7 @@ package ast
|
|||
|
||||
import "github.com/d5/tengo/compiler/source"
|
||||
|
||||
// Ident represetns an identifier.
|
||||
// Ident represents an identifier.
|
||||
type Ident struct {
|
||||
Name string
|
||||
NamePos source.Pos
|
||||
|
|
|
@ -6,7 +6,7 @@ import (
|
|||
"github.com/d5/tengo/compiler/source"
|
||||
)
|
||||
|
||||
// IdentList represetns a list of identifiers.
|
||||
// IdentList represents a list of identifiers.
|
||||
type IdentList struct {
|
||||
LParen source.Pos
|
||||
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"
|
||||
|
||||
// IntLit represetns an integer literal.
|
||||
// IntLit represents an integer literal.
|
||||
type IntLit struct {
|
||||
Value int64
|
||||
ValuePos source.Pos
|
||||
|
|
|
@ -26,10 +26,10 @@ func (e *MapLit) End() source.Pos {
|
|||
}
|
||||
|
||||
func (e *MapLit) String() string {
|
||||
var elts []string
|
||||
var elements []string
|
||||
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"
|
||||
|
||||
"github.com/d5/tengo/compiler/ast"
|
||||
"github.com/d5/tengo/compiler/stdmods"
|
||||
"github.com/d5/tengo/compiler/token"
|
||||
"github.com/d5/tengo/objects"
|
||||
)
|
||||
|
||||
// Compiler compiles the AST into a bytecode.
|
||||
type Compiler struct {
|
||||
parent *Compiler
|
||||
moduleName string
|
||||
constants []objects.Object
|
||||
symbolTable *SymbolTable
|
||||
scopes []CompilationScope
|
||||
scopeIndex int
|
||||
moduleLoader ModuleLoader
|
||||
compiledModules map[string]*objects.CompiledModule
|
||||
loops []*Loop
|
||||
loopIndex int
|
||||
trace io.Writer
|
||||
|
@ -42,6 +47,7 @@ func NewCompiler(symbolTable *SymbolTable, trace io.Writer) *Compiler {
|
|||
scopeIndex: 0,
|
||||
loopIndex: -1,
|
||||
trace: trace,
|
||||
compiledModules: make(map[string]*objects.CompiledModule),
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -62,11 +68,13 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
return err
|
||||
}
|
||||
}
|
||||
|
||||
case *ast.ExprStmt:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
}
|
||||
c.emit(OpPop)
|
||||
|
||||
case *ast.IncDecStmt:
|
||||
op := token.AddAssign
|
||||
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)
|
||||
|
||||
case *ast.ParenExpr:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
case *ast.BinaryExpr:
|
||||
if node.Token == token.LAnd || node.Token == token.LOr {
|
||||
return c.compileLogical(node)
|
||||
|
@ -149,22 +159,29 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
default:
|
||||
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
||||
}
|
||||
|
||||
case *ast.IntLit:
|
||||
c.emit(OpConstant, c.addConstant(&objects.Int{Value: node.Value}))
|
||||
|
||||
case *ast.FloatLit:
|
||||
c.emit(OpConstant, c.addConstant(&objects.Float{Value: node.Value}))
|
||||
|
||||
case *ast.BoolLit:
|
||||
if node.Value {
|
||||
c.emit(OpTrue)
|
||||
} else {
|
||||
c.emit(OpFalse)
|
||||
}
|
||||
|
||||
case *ast.StringLit:
|
||||
c.emit(OpConstant, c.addConstant(&objects.String{Value: node.Value}))
|
||||
|
||||
case *ast.CharLit:
|
||||
c.emit(OpConstant, c.addConstant(&objects.Char{Value: node.Value}))
|
||||
|
||||
case *ast.UndefinedLit:
|
||||
c.emit(OpNull)
|
||||
|
||||
case *ast.UnaryExpr:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
|
@ -182,6 +199,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
default:
|
||||
return fmt.Errorf("unknown operator: %s", node.Token.String())
|
||||
}
|
||||
|
||||
case *ast.IfStmt:
|
||||
// open new symbol table for the statement
|
||||
c.symbolTable = c.symbolTable.Fork(true)
|
||||
|
@ -229,8 +247,10 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
|
||||
case *ast.ForStmt:
|
||||
return c.compileForStmt(node)
|
||||
|
||||
case *ast.ForInStmt:
|
||||
return c.compileForInStmt(node)
|
||||
|
||||
case *ast.BranchStmt:
|
||||
if node.Token == token.Break {
|
||||
curLoop := c.currentLoop()
|
||||
|
@ -249,17 +269,19 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
} else {
|
||||
return fmt.Errorf("unknown branch statement: %s", node.Token.String())
|
||||
}
|
||||
|
||||
case *ast.BlockStmt:
|
||||
for _, stmt := range node.Stmts {
|
||||
if err := c.Compile(stmt); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
case *ast.AssignStmt:
|
||||
|
||||
case *ast.AssignStmt:
|
||||
if err := c.compileAssign(node.LHS, node.RHS, node.Token); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
case *ast.Ident:
|
||||
symbol, _, ok := c.symbolTable.Resolve(node.Name)
|
||||
if !ok {
|
||||
|
@ -276,6 +298,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
case ScopeFree:
|
||||
c.emit(OpGetFree, symbol.Index)
|
||||
}
|
||||
|
||||
case *ast.ArrayLit:
|
||||
for _, elem := range node.Elements {
|
||||
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))
|
||||
|
||||
case *ast.MapLit:
|
||||
for _, elt := range node.Elements {
|
||||
// key
|
||||
|
@ -296,6 +320,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
}
|
||||
|
||||
c.emit(OpMap, len(node.Elements)*2)
|
||||
|
||||
case *ast.SelectorExpr: // selector on RHS side
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
|
@ -306,6 +331,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
}
|
||||
|
||||
c.emit(OpIndex)
|
||||
|
||||
case *ast.IndexExpr:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
|
@ -316,6 +342,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
}
|
||||
|
||||
c.emit(OpIndex)
|
||||
|
||||
case *ast.SliceExpr:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
return err
|
||||
|
@ -338,6 +365,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
}
|
||||
|
||||
c.emit(OpSliceIndex)
|
||||
|
||||
case *ast.FuncLit:
|
||||
c.enterScope()
|
||||
|
||||
|
@ -378,6 +406,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
} else {
|
||||
c.emit(OpConstant, c.addConstant(compiledFunction))
|
||||
}
|
||||
|
||||
case *ast.ReturnStmt:
|
||||
if c.symbolTable.Parent(true) == nil {
|
||||
// outside the function
|
||||
|
@ -396,6 +425,7 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
default:
|
||||
return fmt.Errorf("multi-value return not implemented")
|
||||
}
|
||||
|
||||
case *ast.CallExpr:
|
||||
if err := c.Compile(node.Func); err != nil {
|
||||
return err
|
||||
|
@ -409,6 +439,21 @@ func (c *Compiler) Compile(node ast.Node) error {
|
|||
|
||||
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:
|
||||
if err := c.Compile(node.Expr); err != nil {
|
||||
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 {
|
||||
if c.parent != nil {
|
||||
// module compilers will use their parent's constants array
|
||||
return c.parent.addConstant(o)
|
||||
}
|
||||
|
||||
c.constants = append(c.constants, o)
|
||||
|
||||
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(1),
|
||||
intObject(1))))
|
||||
|
||||
expectError(t, `import("user1")`) // unknown module name
|
||||
}
|
||||
|
||||
func concat(insts ...[]byte) []byte {
|
||||
func concat(instructions ...[]byte) []byte {
|
||||
concat := make([]byte, 0)
|
||||
for _, i := range insts {
|
||||
for _, i := range instructions {
|
||||
concat = append(concat, i...)
|
||||
}
|
||||
|
||||
|
@ -913,6 +915,22 @@ func expect(t *testing.T, input string, expected *compiler.Bytecode) (ok bool) {
|
|||
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 {
|
||||
expectedInstructions := strings.Join(compiler.FormatInstructions(expected.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
|
||||
OpIteratorKey // Iterator key
|
||||
OpIteratorValue // Iterator value
|
||||
OpModule // Module
|
||||
)
|
||||
|
||||
// OpcodeNames is opcode names.
|
||||
|
@ -111,6 +112,7 @@ var OpcodeNames = [...]string{
|
|||
OpIteratorNext: "ITNXT",
|
||||
OpIteratorKey: "ITKEY",
|
||||
OpIteratorValue: "ITVAL",
|
||||
OpModule: "MODULE",
|
||||
}
|
||||
|
||||
// OpcodeOperands is the number of operands.
|
||||
|
@ -166,6 +168,7 @@ var OpcodeOperands = [...][]int{
|
|||
OpIteratorNext: {},
|
||||
OpIteratorKey: {},
|
||||
OpIteratorValue: {},
|
||||
OpModule: {2},
|
||||
}
|
||||
|
||||
// ReadOperands reads operands from the bytecode.
|
||||
|
|
|
@ -334,6 +334,9 @@ func (p *Parser) parseOperand() ast.Expr {
|
|||
p.next()
|
||||
return x
|
||||
|
||||
case token.Import:
|
||||
return p.parseImportExpr()
|
||||
|
||||
case token.LParen:
|
||||
lparen := p.pos
|
||||
p.next()
|
||||
|
@ -366,6 +369,35 @@ func (p *Parser) parseOperand() ast.Expr {
|
|||
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 {
|
||||
if n := len(p.tokenLit); n >= 3 {
|
||||
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)
|
||||
p.exprLevel++
|
||||
|
||||
var elts []ast.Expr
|
||||
var elements []ast.Expr
|
||||
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") {
|
||||
break
|
||||
|
@ -426,7 +458,7 @@ func (p *Parser) parseArrayLit() ast.Expr {
|
|||
rbrack := p.expect(token.RBrack)
|
||||
|
||||
return &ast.ArrayLit{
|
||||
Elements: elts,
|
||||
Elements: elements,
|
||||
LBrack: lbrack,
|
||||
RBrack: rbrack,
|
||||
}
|
||||
|
@ -540,9 +572,9 @@ func (p *Parser) parseStmt() (stmt ast.Stmt) {
|
|||
|
||||
switch p.token {
|
||||
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.LBrace, token.LBrack, // composite types
|
||||
token.Add, token.Sub, token.Mul, token.And, token.Xor, token.Not: // unary operators
|
||||
token.Func, token.Error, token.Ident, token.Int, token.Float, token.Char, token.String, token.True, token.False,
|
||||
token.Undefined, token.Import, token.LParen, token.LBrace, token.LBrack,
|
||||
token.Add, token.Sub, token.Mul, token.And, token.Xor, token.Not:
|
||||
s := p.parseSimpleStmt(false)
|
||||
p.expectSemi()
|
||||
return s
|
||||
|
@ -926,9 +958,9 @@ func (p *Parser) parseMapLit() *ast.MapLit {
|
|||
lbrace := p.expect(token.LBrace)
|
||||
p.exprLevel++
|
||||
|
||||
var elts []*ast.MapElementLit
|
||||
var elements []*ast.MapElementLit
|
||||
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") {
|
||||
break
|
||||
|
@ -941,7 +973,7 @@ func (p *Parser) parseMapLit() *ast.MapLit {
|
|||
return &ast.MapLit{
|
||||
LBrace: lbrace,
|
||||
RBrace: rbrace,
|
||||
Elements: elts,
|
||||
Elements: elements,
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@ import (
|
|||
"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 {
|
||||
return stmts(
|
||||
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}
|
||||
}
|
||||
|
||||
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 {
|
||||
return list
|
||||
}
|
||||
|
@ -385,6 +389,10 @@ func equalExpr(t *testing.T, expected, actual ast.Expr) bool {
|
|||
case *ast.SelectorExpr:
|
||||
return equalExpr(t, expected.Expr, actual.(*ast.SelectorExpr).Expr) &&
|
||||
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:
|
||||
return equalExpr(t, expected.Expr, actual.(*ast.ErrorExpr).Expr) &&
|
||||
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
|
||||
In
|
||||
Undefined
|
||||
Import
|
||||
_keywordEnd
|
||||
)
|
||||
|
||||
|
@ -156,6 +157,7 @@ var tokens = [...]string{
|
|||
False: "false",
|
||||
In: "in",
|
||||
Undefined: "undefined",
|
||||
Import: "import",
|
||||
}
|
||||
|
||||
func (tok Token) String() string {
|
||||
|
|
|
@ -7,7 +7,7 @@ import (
|
|||
// append(src, items...)
|
||||
func builtinAppend(args ...Object) (Object, error) {
|
||||
if len(args) < 2 {
|
||||
return nil, fmt.Errorf("not enough arguments in call to append")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
|
|
@ -1,13 +1,12 @@
|
|||
package objects
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"strconv"
|
||||
)
|
||||
|
||||
func builtinString(args ...Object) (Object, error) {
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
@ -22,7 +21,7 @@ func builtinString(args ...Object) (Object, error) {
|
|||
|
||||
func builtinInt(args ...Object) (Object, error) {
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
@ -49,7 +48,7 @@ func builtinInt(args ...Object) (Object, error) {
|
|||
|
||||
func builtinFloat(args ...Object) (Object, error) {
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
@ -69,7 +68,7 @@ func builtinFloat(args ...Object) (Object, error) {
|
|||
|
||||
func builtinBool(args ...Object) (Object, error) {
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
@ -82,7 +81,7 @@ func builtinBool(args ...Object) (Object, error) {
|
|||
|
||||
func builtinChar(args ...Object) (Object, error) {
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
switch arg := args[0].(type) {
|
||||
|
|
|
@ -1,11 +1,8 @@
|
|||
package objects
|
||||
|
||||
import "errors"
|
||||
|
||||
func builtinCopy(args ...Object) (Object, error) {
|
||||
// TODO: should multi arguments later?
|
||||
if len(args) != 1 {
|
||||
return nil, errors.New("wrong number of arguments")
|
||||
return nil, ErrWrongNumArguments
|
||||
}
|
||||
|
||||
return args[0].Copy(), nil
|
||||
|
|
|
@ -6,7 +6,7 @@ import (
|
|||
|
||||
// BuiltinFunction represents a builtin function.
|
||||
type BuiltinFunction struct {
|
||||
Value BuiltinFunc
|
||||
Value CallableFunc
|
||||
}
|
||||
|
||||
// TypeName returns the name of the type.
|
||||
|
|
|
@ -6,7 +6,7 @@ import (
|
|||
|
||||
func builtinLen(args ...Object) (Object, error) {
|
||||
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) {
|
||||
|
|
|
@ -1,12 +1,9 @@
|
|||
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.
|
||||
var Builtins = []struct {
|
||||
Name string
|
||||
Func BuiltinFunc
|
||||
Func CallableFunc
|
||||
}{
|
||||
{
|
||||
Name: "print",
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
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 {
|
||||
// Call should take an arbitrary number of arguments
|
||||
// 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.
|
||||
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
|
||||
// a given binary operator and a right-hand side object.
|
||||
func (o *String) BinaryOp(op token.Token, rhs Object) (Object, error) {
|
||||
switch op {
|
||||
case token.Add:
|
||||
switch rhs := rhs.(type) {
|
||||
case *String:
|
||||
switch op {
|
||||
case token.Add:
|
||||
if rhs.Value == "" {
|
||||
return o, nil
|
||||
}
|
||||
return &String{Value: o.Value + rhs.Value}, nil
|
||||
}
|
||||
case *Char:
|
||||
switch op {
|
||||
case token.Add:
|
||||
return &String{Value: o.Value + string(rhs.Value)}, nil
|
||||
default:
|
||||
return &String{Value: o.Value + rhs.String()}, 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:
|
||||
key, ok := (*index).(*objects.String)
|
||||
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
|
||||
|
@ -986,6 +1005,8 @@ func (v *VM) Run() error {
|
|||
iterator = objects.NewArrayIterator(dst)
|
||||
case *objects.Map:
|
||||
iterator = objects.NewMapIterator(dst)
|
||||
case *objects.ModuleMap:
|
||||
iterator = objects.NewModuleMapIterator(dst)
|
||||
case *objects.String:
|
||||
iterator = objects.NewStringIterator(dst)
|
||||
default:
|
||||
|
@ -1041,6 +1062,14 @@ func (v *VM) Run() error {
|
|||
v.stack[v.sp] = &val
|
||||
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:
|
||||
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
|
||||
}
|
||||
|
||||
// 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 {
|
||||
numSel := len(selectors)
|
||||
|
||||
|
|
|
@ -5,15 +5,21 @@ import (
|
|||
)
|
||||
|
||||
func TestForIn(t *testing.T) {
|
||||
expect(t, `for i, x in [1, 2, 3] { out += i + x }`, 9)
|
||||
expect(t, `func() { for i, x in [1, 2, 3] { out += i + x } }()`, 9)
|
||||
// array
|
||||
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)
|
||||
expect(t, `func() { for i, _ in [1, 2, 3] { out += i } }()`, 3)
|
||||
|
||||
expect(t, `for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } }`, "b")
|
||||
expect(t, `func() { for k, v in {a:2,b:3,c:4} { out = k; if v==3 { break } } }()`, "b")
|
||||
// map
|
||||
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, _ in {a:2} { out += k }`, "a") // key, _
|
||||
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 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++; 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 := "foo"; a++`) // invalid operand
|
||||
//expectError(t, `a := 0; b := a++`) // inc-dec is statement not expression <- parser error
|
||||
|
||||
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:%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"`)
|
||||
}
|
||||
|
|
|
@ -24,6 +24,10 @@ type MAP = map[string]interface{}
|
|||
type ARR = []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
|
||||
file := parse(t, input)
|
||||
if file == nil {
|
||||
|
@ -31,10 +35,14 @@ func expect(t *testing.T, input string, expected interface{}) {
|
|||
}
|
||||
|
||||
// compiler/VM
|
||||
runVM(t, file, expected)
|
||||
runVM(t, file, expected, userModules)
|
||||
}
|
||||
|
||||
func expectError(t *testing.T, input string) {
|
||||
expectErrorWithUserModules(t, input, nil)
|
||||
}
|
||||
|
||||
func expectErrorWithUserModules(t *testing.T, input string, userModules map[string]string) {
|
||||
// parse
|
||||
program := parse(t, input)
|
||||
if program == nil {
|
||||
|
@ -42,15 +50,15 @@ func expectError(t *testing.T, input string) {
|
|||
}
|
||||
|
||||
// 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)
|
||||
|
||||
res, trace, err := traceCompileRun(file, map[string]objects.Object{
|
||||
testOut: objectZeroCopy(expectedObj),
|
||||
})
|
||||
}, userModules)
|
||||
|
||||
defer func() {
|
||||
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
|
||||
func runVMError(t *testing.T, file *ast.File) (ok bool) {
|
||||
_, trace, err := traceCompileRun(file, nil)
|
||||
func runVMError(t *testing.T, file *ast.File, userModules map[string]string) (ok bool) {
|
||||
_, trace, err := traceCompileRun(file, nil, userModules)
|
||||
|
||||
defer func() {
|
||||
if !ok {
|
||||
|
@ -132,7 +140,7 @@ func (o *tracer) Write(p []byte) (n int, err error) {
|
|||
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
|
||||
|
||||
defer func() {
|
||||
|
@ -169,6 +177,13 @@ func traceCompileRun(file *ast.File, symbols map[string]objects.Object) (res map
|
|||
|
||||
tr := &tracer{}
|
||||
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)
|
||||
trace = append(trace, fmt.Sprintf("\n[Compiler Trace]\n\n%s", strings.Join(tr.Out, "")))
|
||||
if err != nil {
|
||||
|
|
Loading…
Reference in a new issue