commit
61e534eb67
8 changed files with 399 additions and 214 deletions
|
@ -66,7 +66,7 @@ _* See [here](https://github.com/d5/tengobench) for commands/codes used_
|
||||||
## References
|
## References
|
||||||
|
|
||||||
- [Language Syntax](https://github.com/d5/tengo/blob/master/docs/tutorial.md)
|
- [Language Syntax](https://github.com/d5/tengo/blob/master/docs/tutorial.md)
|
||||||
- [Tengo Objects](https://github.com/d5/tengo/blob/master/docs/objects.md)
|
- [Object Types](https://github.com/d5/tengo/blob/master/docs/objects.md)
|
||||||
- [Runtime Types](https://github.com/d5/tengo/blob/master/docs/runtime-types.md)
|
- [Runtime Types](https://github.com/d5/tengo/blob/master/docs/runtime-types.md)
|
||||||
- [Builtin Functions](https://github.com/d5/tengo/blob/master/docs/builtins.md)
|
- [Builtin Functions](https://github.com/d5/tengo/blob/master/docs/builtins.md)
|
||||||
- [Interoperability](https://github.com/d5/tengo/blob/master/docs/interoperability.md)
|
- [Interoperability](https://github.com/d5/tengo/blob/master/docs/interoperability.md)
|
||||||
|
|
|
@ -4,7 +4,8 @@
|
||||||
|
|
||||||
- [Using Scripts](#using-scripts)
|
- [Using Scripts](#using-scripts)
|
||||||
- [Type Conversion Table](#type-conversion-table)
|
- [Type Conversion Table](#type-conversion-table)
|
||||||
- [User Types](#user-types)
|
- [User Types](#user-types)
|
||||||
|
- [Importing Scripts](#importing-scripts)
|
||||||
- [Sandbox Environments](#sandbox-environments)
|
- [Sandbox Environments](#sandbox-environments)
|
||||||
- [Compiler and VM](#compiler-and-vm)
|
- [Compiler and VM](#compiler-and-vm)
|
||||||
|
|
||||||
|
@ -90,6 +91,7 @@ When adding a Variable _([Script.Add](https://godoc.org/github.com/d5/tengo/scri
|
||||||
|`byte`|`Char`||
|
|`byte`|`Char`||
|
||||||
|`float64`|`Float`||
|
|`float64`|`Float`||
|
||||||
|`[]byte`|`Bytes`||
|
|`[]byte`|`Bytes`||
|
||||||
|
|`time.Time`|`Time`||
|
||||||
|`error`|`Error{String}`|use `error.Error()` as String value|
|
|`error`|`Error{String}`|use `error.Error()` as String value|
|
||||||
|`map[string]Object`|`Map`||
|
|`map[string]Object`|`Map`||
|
||||||
|`map[string]interface{}`|`Map`|individual elements converted to Tengo objects|
|
|`map[string]interface{}`|`Map`|individual elements converted to Tengo objects|
|
||||||
|
@ -98,9 +100,23 @@ When adding a Variable _([Script.Add](https://godoc.org/github.com/d5/tengo/scri
|
||||||
|`Object`|`Object`|_(no type conversion performed)_|
|
|`Object`|`Object`|_(no type conversion performed)_|
|
||||||
|
|
||||||
|
|
||||||
## User Types
|
### User Types
|
||||||
|
|
||||||
One can easily add and use customized value types in Tengo code by implementing [Object](https://godoc.org/github.com/d5/tengo/objects#Object) interface. Tengo runtime will treat the user types exactly in the same way it does to the runtime types with no performance overhead. See [Tengo Objects](https://github.com/d5/tengo/blob/master/docs/objects.md) for more details.
|
Users can add and use a custom user type in Tengo code by implementing [Object](https://godoc.org/github.com/d5/tengo/objects#Object) interface. Tengo runtime will treat the user types in the same way it does to the runtime types with no performance overhead. See [Object Types](https://github.com/d5/tengo/blob/master/docs/objects.md) for more details.
|
||||||
|
|
||||||
|
### Importing Scripts
|
||||||
|
|
||||||
|
A script can import and use another script in the same way it can load the standard library or the user module. `Script.AddModule` function adds another script as a named module.
|
||||||
|
|
||||||
|
```golang
|
||||||
|
mod1Script := script.New([]byte(`a := 5`)) // mod1 script
|
||||||
|
|
||||||
|
mainScript := script.New([]byte(`print(import("mod1").a)`)) // main script
|
||||||
|
mainScript.AddModule("mod1", mod1Script) // add mod1 using name "mod1"
|
||||||
|
mainScript.Run() // prints "5"
|
||||||
|
```
|
||||||
|
|
||||||
|
Note that the script modules added using `Script.AddModule` will be compiled and run right before the main script is compiled.
|
||||||
|
|
||||||
## Sandbox Environments
|
## Sandbox Environments
|
||||||
|
|
||||||
|
|
423
docs/objects.md
423
docs/objects.md
|
@ -1,19 +1,22 @@
|
||||||
# Tengo Objects
|
# Object Types
|
||||||
|
|
||||||
## Table of Contents
|
## Table of Contents
|
||||||
|
|
||||||
- [Objects](#objects)
|
- [Tengo Objects](#tengo-objects)
|
||||||
- [Runtime Object Types](#runtime-object-types)
|
- [Object Interface](#object-interface)
|
||||||
- [User Object Types](#user-object-types)
|
- [Callable Interface](#callable-interface)
|
||||||
- [Callable Objects](#callable-objects)
|
- [Indexable Interface](#indexable-interface)
|
||||||
- [Indexable Objects](#indexable-objects)
|
- [Index-Assignable Interface](#index-assignable-interface)
|
||||||
- [Index-Assignable Objects](#index-assignable-objects)
|
- [Iterable Interface](#iterable-interface)
|
||||||
- [Iterable Objects](#iterable-objects)
|
|
||||||
- [Iterator Interface](#iterator-interface)
|
- [Iterator Interface](#iterator-interface)
|
||||||
|
- [Runtime Object Types](#runtime-object-types)
|
||||||
|
- [User Object Types](#user-object-types)
|
||||||
|
|
||||||
## Objects
|
## Tengo Objects
|
||||||
|
|
||||||
All object types in Tengo implement [Object](https://godoc.org/github.com/d5/tengo/objects#Object) interface.
|
In Tengo, all object types _(both [runtime types](#runtime-object-types) and [user types](#user-object-types))_ must implement [Object](https://godoc.org/github.com/d5/tengo/objects#Object) interface. And some types may implement other optional interfaces ([Callable](https://godoc.org/github.com/d5/tengo/objects#Callable), [Indexable](https://godoc.org/github.com/d5/tengo/objects#Indexable), [IndexAssignable](https://godoc.org/github.com/d5/tengo/objects#IndexAssignable), [Iterable](https://godoc.org/github.com/d5/tengo/objects#Iterable)) to support additional language features.
|
||||||
|
|
||||||
|
### Object Interface
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
TypeName() string
|
TypeName() string
|
||||||
|
@ -56,104 +59,10 @@ Copy() Object
|
||||||
|
|
||||||
Copy method should a _new_ copy of the same object. All primitive and composite value types implement this method to return a deep-copy of the value, which is recommended for other user types _(as `copy` builtin function uses this Copy method)_, but, it's not a strict requirement by the runtime.
|
Copy method should a _new_ copy of the same object. All primitive and composite value types implement this method to return a deep-copy of the value, which is recommended for other user types _(as `copy` builtin function uses this Copy method)_, but, it's not a strict requirement by the runtime.
|
||||||
|
|
||||||
### Runtime Object Types
|
|
||||||
|
|
||||||
These are the Tengo runtime object types:
|
### Callable Interface
|
||||||
|
|
||||||
- Primitive value types: [Int](https://godoc.org/github.com/d5/tengo/objects#Int), [String](https://godoc.org/github.com/d5/tengo/objects#String), [Float](https://godoc.org/github.com/d5/tengo/objects#Float), [Bool](https://godoc.org/github.com/d5/tengo/objects#ArrayIterator), [Char](https://godoc.org/github.com/d5/tengo/objects#Char), [Bytes](https://godoc.org/github.com/d5/tengo/objects#Bytes)
|
If the type implements [Callable](https://godoc.org/github.com/d5/tengo/objects#Callable) interface, its values can be invoked as if they were functions.
|
||||||
- Composite value types: [Array](https://godoc.org/github.com/d5/tengo/objects#Array), [Map](https://godoc.org/github.com/d5/tengo/objects#Map), [ImmutableMap](https://godoc.org/github.com/d5/tengo/objects#ImmutableMap)
|
|
||||||
- Functions: [CompiledFunction](https://godoc.org/github.com/d5/tengo/objects#CompiledFunction), [BuiltinFunction](https://godoc.org/github.com/d5/tengo/objects#BuiltinFunction), [UserFunction](https://godoc.org/github.com/d5/tengo/objects#UserFunction)
|
|
||||||
- [Iterators](https://godoc.org/github.com/d5/tengo/objects#Iterator): [StringIterator](https://godoc.org/github.com/d5/tengo/objects#StringIterator), [ArrayIterator](https://godoc.org/github.com/d5/tengo/objects#ArrayIterator), [MapIterator](https://godoc.org/github.com/d5/tengo/objects#MapIterator), [ImmutableMapIterator](https://godoc.org/github.com/d5/tengo/objects#ImmutableMapIterator)
|
|
||||||
- [Error](https://godoc.org/github.com/d5/tengo/objects#Error)
|
|
||||||
- [Undefined](https://godoc.org/github.com/d5/tengo/objects#Undefined)
|
|
||||||
- Other internal objects: [Closure](https://godoc.org/github.com/d5/tengo/objects#Closure), [CompiledModule](https://godoc.org/github.com/d5/tengo/objects#CompiledModule), [Break](https://godoc.org/github.com/d5/tengo/objects#Break), [Continue](https://godoc.org/github.com/d5/tengo/objects#Continue), [ReturnValue](https://godoc.org/github.com/d5/tengo/objects#ReturnValue)
|
|
||||||
|
|
||||||
### User Object Types
|
|
||||||
|
|
||||||
Basically Tengo runtime treats and manages both the runtime types and user types exactly the same way as long as they implement Object interface. You can add values of the custom user types (via either [Script.Add](https://godoc.org/github.com/d5/tengo/script#Script.Add) method or by directly manipulating the symbol table and the global variables), and, use them directly in Tengo code.
|
|
||||||
|
|
||||||
Here's an example user type, `Time`:
|
|
||||||
|
|
||||||
```golang
|
|
||||||
type Time struct {
|
|
||||||
Value time.Time
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) TypeName() string {
|
|
||||||
return "time"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) String() string {
|
|
||||||
return t.Value.Format(time.RFC3339)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) BinaryOp(op token.Token, rhs objects.Object) (objects.Object, error) {
|
|
||||||
switch rhs := rhs.(type) {
|
|
||||||
case *Time:
|
|
||||||
switch op {
|
|
||||||
case token.Sub:
|
|
||||||
return &objects.Int{
|
|
||||||
Value: t.Value.Sub(rhs.Value).Nanoseconds(),
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
case *objects.Int:
|
|
||||||
switch op {
|
|
||||||
case token.Add:
|
|
||||||
return &Time{
|
|
||||||
Value: t.Value.Add(time.Duration(rhs.Value)),
|
|
||||||
}, nil
|
|
||||||
case token.Sub:
|
|
||||||
return &Time{
|
|
||||||
Value: t.Value.Add(-time.Duration(rhs.Value)),
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return nil, objects.ErrInvalidOperator
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) IsFalsy() bool {
|
|
||||||
return t.Value.IsZero()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) Equals(o objects.Object) bool {
|
|
||||||
if o, ok := o.(*Time); ok {
|
|
||||||
return t.Value.Equal(o.Value)
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t *Time) Copy() objects.Object {
|
|
||||||
return &Time{Value: t.Value}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Now the Tengo runtime recognizes `Time` type, and, any `Time` values can be used directly in the Tengo code:
|
|
||||||
|
|
||||||
```golang
|
|
||||||
s := script.New([]byte(`
|
|
||||||
a := currentTime + 10000 // Time + Int = Time
|
|
||||||
b := a - currentTime // Time - Time = Int
|
|
||||||
`))
|
|
||||||
|
|
||||||
// add Time value 'currentTime'
|
|
||||||
err := s.Add("currentTime", &Time{Value: time.Now()})
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
c, err := s.Run()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Println(c.Get("b")) // "10000"
|
|
||||||
```
|
|
||||||
|
|
||||||
## Callable Objects
|
|
||||||
|
|
||||||
Any types that implement [Callable](https://godoc.org/github.com/d5/tengo/objects#Callable) interface (in addition to Object interface), values of such types can be used as if they are functions.
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
type Callable interface {
|
type Callable interface {
|
||||||
|
@ -161,48 +70,9 @@ type Callable interface {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
To make `Time` a callable value, add Call method to the previous implementation:
|
### Indexable Interface
|
||||||
|
|
||||||
```golang
|
If the type implements [Indexable](https://godoc.org/github.com/d5/tengo/objects#Indexable) interface, its values support dot selector (`value = object.index`) and indexer (`value = object[index]`) syntax.
|
||||||
func (t *Time) Call(args ...objects.Object) (ret objects.Object, err error) {
|
|
||||||
if len(args) != 1 {
|
|
||||||
return nil, objects.ErrWrongNumArguments
|
|
||||||
}
|
|
||||||
|
|
||||||
format, ok := objects.ToString(args[0])
|
|
||||||
if !ok {
|
|
||||||
return nil, objects.ErrInvalidTypeConversion
|
|
||||||
}
|
|
||||||
|
|
||||||
return &objects.String{Value: t.Value.Format(format)}, nil
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Now `Time` values can be "called" like this:
|
|
||||||
|
|
||||||
```golang
|
|
||||||
s := script.New([]byte(`
|
|
||||||
a := currentTime + 10000 // Time + Int = Time
|
|
||||||
b := a("15:04:05") // call 'a'
|
|
||||||
`))
|
|
||||||
|
|
||||||
// add Time value 'currentTime'
|
|
||||||
err := s.Add("currentTime", &Time{Value: time.Now()})
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
c, err := s.Run()
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
fmt.Println(c.Get("b")) // something like "21:15:27"
|
|
||||||
```
|
|
||||||
|
|
||||||
## Indexable Objects
|
|
||||||
|
|
||||||
If the type implements [Indexable](https://godoc.org/github.com/d5/tengo/objects#Indexable) interface, it enables dot selector (`value = object.index`) or indexer (`value = object[index]`) syntax for its values.
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
type Indexable interface {
|
type Indexable interface {
|
||||||
|
@ -210,32 +80,15 @@ type Indexable interface {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
If the implementation returns an error (`err`), the VM will treat it as a run-time error. Many runtime types such as Map and Array also implement the same interface:
|
If `IndexGet` returns an error (`err`), the VM will treat it as a run-time error.
|
||||||
|
|
||||||
```golang
|
Array and Map implementation forces the type of index Object to be Int and String respectively, but, it's not a required behavior of the VM. It is completely okay to take various index types as long as it is consistent.
|
||||||
func (o *Map) IndexGet(index Object) (res Object, err error) {
|
|
||||||
strIdx, ok := index.(*String)
|
|
||||||
if !ok {
|
|
||||||
err = ErrInvalidIndexType
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
val, ok := o.Value[strIdx.Value]
|
By convention, Array or Array-like types return `ErrIndexOutOfBounds` error (as a runtime error) when the index is invalid (out of the bounds), and, Map or Map-like types return `Undefined` value (instead of a run-time error) when the key does not exist. But, again, this is not a required behavior.
|
||||||
if !ok {
|
|
||||||
val = UndefinedValue
|
|
||||||
}
|
|
||||||
|
|
||||||
return val, nil
|
### Index-Assignable Interface
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Array and Map implementation forces the type of index Object (Int and String respectively), but, it's not required behavior by the VM. It is completely okay to take various index types (or to do type coercion) as long as its result is consistent.
|
If the type implements [IndexAssignable](https://godoc.org/github.com/d5/tengo/objects#IndexAssignable) interface, its values support assignment using dot selector (`object.index = value`) and indexer (`object[index] = value`) in the assignment statements.
|
||||||
|
|
||||||
By convention, Array or Array-like types return `ErrIndexOutOfBounds` error (as a runtime error) when the index is invalid (out of the bounds), and, Map or Map-like types return `Undefined` value when the key does not exist. But, again this is not a requirement, and, the type can implement the behavior however it fits.
|
|
||||||
|
|
||||||
## Index-Assignable Objects
|
|
||||||
|
|
||||||
If the type implements [IndexAssignable](https://godoc.org/github.com/d5/tengo/objects#IndexAssignable) interface, the values of that type allow assignment using dot selector (`object.index = value`) or indexer (`object[index] = value`) in the assignment statements.
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
type IndexAssignable interface {
|
type IndexAssignable interface {
|
||||||
|
@ -243,30 +96,13 @@ type IndexAssignable interface {
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
Map, Array, and a couple of other runtime types also implement the same interface:
|
Array and Map implementation forces the type of index Object to be Int and String respectively, but, it's not a required behavior of the VM. It is completely okay to take various index types as long as it is consistent.
|
||||||
|
|
||||||
```golang
|
By convention, Array or Array-like types return `ErrIndexOutOfBounds` error (as a runtime error) when the index is invalid (out of the bounds).
|
||||||
func (o *Map) IndexSet(index, value Object) (err error) {
|
|
||||||
strIdx, ok := ToString(index)
|
|
||||||
if !ok {
|
|
||||||
err = ErrInvalidTypeConversion
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
o.Value[strIdx] = value
|
### Iterable Interface
|
||||||
|
|
||||||
return nil
|
If the type implements [Iterable](https://godoc.org/github.com/d5/tengo/objects#Iterable) interface, its values can be used in `for-in` statements (`for key, value in object { ... }`).
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
Array and Map implementation forces the type of index Object (Int and String respectively), but, it's not required behavior by the VM. It is completely okay to take various index types (or to do type coercion) as long as its result is consistent.
|
|
||||||
|
|
||||||
By convention, Array or Array-like types return `ErrIndexOutOfBounds` error (as a runtime error) when the index is invalid (out of the bounds). But, this is not a requirement, and, the type can implement the behavior however it fits.
|
|
||||||
|
|
||||||
|
|
||||||
## Iterable Objects
|
|
||||||
|
|
||||||
Values of the types that implement [Iterable](https://godoc.org/github.com/d5/tengo/objects#Iterable) interface can be used in `for-in` statements (`for key, value in object { ... }`).
|
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
type Iterable interface {
|
type Iterable interface {
|
||||||
|
@ -276,7 +112,7 @@ type Iterable interface {
|
||||||
|
|
||||||
This Iterate method should return another object that implements [Iterator](https://godoc.org/github.com/d5/tengo/objects#Iterator) interface.
|
This Iterate method should return another object that implements [Iterator](https://godoc.org/github.com/d5/tengo/objects#Iterator) interface.
|
||||||
|
|
||||||
### Iterator Interface
|
#### Iterator Interface
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
Next() bool
|
Next() bool
|
||||||
|
@ -295,3 +131,208 @@ Value() Object
|
||||||
```
|
```
|
||||||
|
|
||||||
Value method should return a value Object for the current element of the underlying object. It should return the same value until Next method is called again.
|
Value method should return a value Object for the current element of the underlying object. It should return the same value until Next method is called again.
|
||||||
|
|
||||||
|
## Runtime Object Types
|
||||||
|
|
||||||
|
These are the basic types Tengo runtime supports out of the box:
|
||||||
|
|
||||||
|
- Primitive value types: [Int](https://godoc.org/github.com/d5/tengo/objects#Int), [String](https://godoc.org/github.com/d5/tengo/objects#String), [Float](https://godoc.org/github.com/d5/tengo/objects#Float), [Bool](https://godoc.org/github.com/d5/tengo/objects#ArrayIterator), [Char](https://godoc.org/github.com/d5/tengo/objects#Char), [Bytes](https://godoc.org/github.com/d5/tengo/objects#Bytes), [Time](https://godoc.org/github.com/d5/tengo/objects#Time)
|
||||||
|
- Composite value types: [Array](https://godoc.org/github.com/d5/tengo/objects#Array), [ImmutableArray](https://godoc.org/github.com/d5/tengo/objects#ImmutableArray), [Map](https://godoc.org/github.com/d5/tengo/objects#Map), [ImmutableMap](https://godoc.org/github.com/d5/tengo/objects#ImmutableMap)
|
||||||
|
- Functions: [CompiledFunction](https://godoc.org/github.com/d5/tengo/objects#CompiledFunction), [BuiltinFunction](https://godoc.org/github.com/d5/tengo/objects#BuiltinFunction), [UserFunction](https://godoc.org/github.com/d5/tengo/objects#UserFunction)
|
||||||
|
- [Iterators](https://godoc.org/github.com/d5/tengo/objects#Iterator): [StringIterator](https://godoc.org/github.com/d5/tengo/objects#StringIterator), [ArrayIterator](https://godoc.org/github.com/d5/tengo/objects#ArrayIterator), [MapIterator](https://godoc.org/github.com/d5/tengo/objects#MapIterator), [ImmutableMapIterator](https://godoc.org/github.com/d5/tengo/objects#ImmutableMapIterator)
|
||||||
|
- [Error](https://godoc.org/github.com/d5/tengo/objects#Error)
|
||||||
|
- [Undefined](https://godoc.org/github.com/d5/tengo/objects#Undefined)
|
||||||
|
- Other internal objects: [Closure](https://godoc.org/github.com/d5/tengo/objects#Closure), [CompiledModule](https://godoc.org/github.com/d5/tengo/objects#CompiledModule), [Break](https://godoc.org/github.com/d5/tengo/objects#Break), [Continue](https://godoc.org/github.com/d5/tengo/objects#Continue), [ReturnValue](https://godoc.org/github.com/d5/tengo/objects#ReturnValue)
|
||||||
|
|
||||||
|
See [Runtime Types](https://github.com/d5/tengo/blob/master/docs/runtime-types.md) for more details on these runtime types.
|
||||||
|
|
||||||
|
## User Object Types
|
||||||
|
|
||||||
|
Users can easily extend and add their own types by implementing the same [Object](https://godoc.org/github.com/d5/tengo/objects#Object) interface, and, Tengo runtime will treat them in the same way as its runtime types with no performance overhead.
|
||||||
|
|
||||||
|
Here's an example user type implementation, `StringArray`:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
type StringArray struct {
|
||||||
|
Value []string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) String() string {
|
||||||
|
return strings.Join(o.Value, ", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) BinaryOp(op token.Token, rhs objects.Object) (objects.Object, error) {
|
||||||
|
if rhs, ok := rhs.(*StringArray); ok {
|
||||||
|
switch op {
|
||||||
|
case token.Add:
|
||||||
|
if len(rhs.Value) == 0 {
|
||||||
|
return o, nil
|
||||||
|
}
|
||||||
|
return &StringArray{Value: append(o.Value, rhs.Value...)}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, objects.ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) IsFalsy() bool {
|
||||||
|
return len(o.Value) == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) Equals(x objects.Object) bool {
|
||||||
|
if x, ok := x.(*StringArray); ok {
|
||||||
|
if len(o.Value) != len(x.Value) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range o.Value {
|
||||||
|
if v != x.Value[i] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) Copy() objects.Object {
|
||||||
|
return &StringArray{
|
||||||
|
Value: append([]string{}, o.Value...),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) TypeName() string {
|
||||||
|
return "string-array"
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
You can use a user type via either [Script.Add](https://godoc.org/github.com/d5/tengo/script#Script.Add) or by directly manipulating the symbol table and the global variables. Here's an example code to add `StringArray` to the script:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
// script that uses 'my_list'
|
||||||
|
s := script.New([]byte(`
|
||||||
|
print(my_list + "three")
|
||||||
|
`))
|
||||||
|
|
||||||
|
myList := &StringArray{Value: []string{"one", "two"}}
|
||||||
|
s.Add("my_list", myList) // add StringArray value 'my_list'
|
||||||
|
s.Run() // prints "one, two, three"
|
||||||
|
```
|
||||||
|
|
||||||
|
It can also implement `Indexable` and `IndexAssinable` interfaces:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
func (o *StringArray) IndexGet(index objects.Object) (objects.Object, error) {
|
||||||
|
intIdx, ok := index.(*objects.Int)
|
||||||
|
if ok {
|
||||||
|
if intIdx.Value >= 0 && intIdx.Value < int64(len(o.Value)) {
|
||||||
|
return &objects.String{Value: o.Value[intIdx.Value]}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, objects.ErrIndexOutOfBounds
|
||||||
|
}
|
||||||
|
|
||||||
|
strIdx, ok := index.(*objects.String)
|
||||||
|
if ok {
|
||||||
|
for vidx, str := range o.Value {
|
||||||
|
if strIdx.Value == str {
|
||||||
|
return &objects.Int{Value: int64(vidx)}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects.UndefinedValue, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, objects.ErrInvalidIndexType
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) IndexSet(index, value objects.Object) error {
|
||||||
|
strVal, ok := objects.ToString(value)
|
||||||
|
if !ok {
|
||||||
|
return objects.ErrInvalidTypeConversion
|
||||||
|
}
|
||||||
|
|
||||||
|
intIdx, ok := index.(*objects.Int)
|
||||||
|
if ok {
|
||||||
|
if intIdx.Value >= 0 && intIdx.Value < int64(len(o.Value)) {
|
||||||
|
o.Value[intIdx.Value] = strVal
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects.ErrIndexOutOfBounds
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects.ErrInvalidIndexType
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
If we implement `Callabale` interface:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
func (o *StringArray) Call(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
s1, ok := objects.ToString(args[0])
|
||||||
|
if !ok {
|
||||||
|
return nil, objects.ErrInvalidTypeConversion
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range o.Value {
|
||||||
|
if v == s1 {
|
||||||
|
return &objects.Int{Value: int64(i)}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects.UndefinedValue, nil
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Then it can be "invoked":
|
||||||
|
|
||||||
|
```golang
|
||||||
|
s := script.New([]byte(`
|
||||||
|
print(my_list("two"))
|
||||||
|
`))
|
||||||
|
|
||||||
|
myList := &StringArray{Value: []string{"one", "two", "three"}}
|
||||||
|
s.Add("my_list", myList) // add StringArray value 'my_list'
|
||||||
|
s.Run() // prints "1" (index of "two")
|
||||||
|
```
|
||||||
|
|
||||||
|
We can also make `StringArray` iterable:
|
||||||
|
|
||||||
|
```golang
|
||||||
|
func (o *StringArray) Iterate() objects.Iterator {
|
||||||
|
return &StringArrayIterator{
|
||||||
|
strArr: o,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type StringArrayIterator struct {
|
||||||
|
objectImpl
|
||||||
|
strArr *StringArray
|
||||||
|
idx int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *StringArrayIterator) TypeName() string {
|
||||||
|
return "string-array-iterator"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *StringArrayIterator) Next() bool {
|
||||||
|
i.idx++
|
||||||
|
return i.idx <= len(i.strArr.Value)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *StringArrayIterator) Key() objects.Object {
|
||||||
|
return &objects.Int{Value: int64(i.idx - 1)}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (i *StringArrayIterator) Value() objects.Object {
|
||||||
|
return &objects.String{Value: i.strArr.Value[i.idx-1]}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
|
@ -188,7 +188,7 @@ b = "foo" // ok: this is not mutating the value of array
|
||||||
// but updating reference 'b' with different value
|
// but updating reference 'b' with different value
|
||||||
```
|
```
|
||||||
|
|
||||||
Not that, if you copy (using `copy` builtin function) an immutable value, it will return a "mutable" copy. Also, immutability is not applied to the individual elements of the array or map value, unless they are explicitly made immutable.
|
Note that, if you copy (using `copy` builtin function) an immutable value, it will return a "mutable" copy. Also, immutability is not applied to the individual elements of the array or map value, unless they are explicitly made immutable.
|
||||||
|
|
||||||
```golang
|
```golang
|
||||||
a := immutable({b: 4, c: [1, 2, 3]})
|
a := immutable({b: 4, c: [1, 2, 3]})
|
||||||
|
@ -232,7 +232,7 @@ func1 := func(x) { print(x) }
|
||||||
foo := 2
|
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.
|
Basically, `import` expression returns all the global variables defined in the module as an ImmutableMap value.
|
||||||
|
|
||||||
Also, you can use `import` to load the [Standard Library](https://github.com/d5/tengo/blob/master/docs/stdlib.md).
|
Also, you can use `import` to load the [Standard Library](https://github.com/d5/tengo/blob/master/docs/stdlib.md).
|
||||||
|
|
||||||
|
|
|
@ -6,16 +6,4 @@ func TestCall(t *testing.T) {
|
||||||
expect(t, `a := { b: func(x) { return x + 2 } }; out = a.b(5)`, 7)
|
expect(t, `a := { b: func(x) { return x + 2 } }; out = a.b(5)`, 7)
|
||||||
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a.b.c(5)`, 7)
|
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a.b.c(5)`, 7)
|
||||||
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a["b"].c(5)`, 7)
|
expect(t, `a := { b: { c: func(x) { return x + 2 } } }; out = a["b"].c(5)`, 7)
|
||||||
|
|
||||||
// "this" binding
|
|
||||||
// expect(t, `
|
|
||||||
//a = {
|
|
||||||
// b: {
|
|
||||||
// c: func(x) {
|
|
||||||
// return x + this.d // this -> a.b
|
|
||||||
// },
|
|
||||||
// d: 5
|
|
||||||
// }
|
|
||||||
//}
|
|
||||||
//out = a["b"].c(2)`, 7)
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -104,10 +104,55 @@ func (o *StringCircle) IndexSet(index, value objects.Object) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
type StringArray struct {
|
type StringArray struct {
|
||||||
objectImpl
|
|
||||||
Value []string
|
Value []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) String() string {
|
||||||
|
return strings.Join(o.Value, ", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) BinaryOp(op token.Token, rhs objects.Object) (objects.Object, error) {
|
||||||
|
if rhs, ok := rhs.(*StringArray); ok {
|
||||||
|
switch op {
|
||||||
|
case token.Add:
|
||||||
|
if len(rhs.Value) == 0 {
|
||||||
|
return o, nil
|
||||||
|
}
|
||||||
|
return &StringArray{Value: append(o.Value, rhs.Value...)}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil, objects.ErrInvalidOperator
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) IsFalsy() bool {
|
||||||
|
return len(o.Value) == 0
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) Equals(x objects.Object) bool {
|
||||||
|
if x, ok := x.(*StringArray); ok {
|
||||||
|
if len(o.Value) != len(x.Value) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range o.Value {
|
||||||
|
if v != x.Value[i] {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) Copy() objects.Object {
|
||||||
|
return &StringArray{
|
||||||
|
Value: append([]string{}, o.Value...),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (o *StringArray) TypeName() string {
|
func (o *StringArray) TypeName() string {
|
||||||
return "string-array"
|
return "string-array"
|
||||||
}
|
}
|
||||||
|
@ -155,6 +200,25 @@ func (o *StringArray) IndexSet(index, value objects.Object) error {
|
||||||
return objects.ErrInvalidIndexType
|
return objects.ErrInvalidIndexType
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (o *StringArray) Call(args ...objects.Object) (ret objects.Object, err error) {
|
||||||
|
if len(args) != 1 {
|
||||||
|
return nil, objects.ErrWrongNumArguments
|
||||||
|
}
|
||||||
|
|
||||||
|
s1, ok := objects.ToString(args[0])
|
||||||
|
if !ok {
|
||||||
|
return nil, objects.ErrInvalidTypeConversion
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, v := range o.Value {
|
||||||
|
if v == s1 {
|
||||||
|
return &objects.Int{Value: int64(i)}, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return objects.UndefinedValue, nil
|
||||||
|
}
|
||||||
|
|
||||||
func TestIndexable(t *testing.T) {
|
func TestIndexable(t *testing.T) {
|
||||||
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
dict := func() *StringDict { return &StringDict{Value: map[string]string{"a": "foo", "b": "bar"}} }
|
||||||
expectWithSymbols(t, `out = dict["a"]`, "foo", SYM{"dict": dict()})
|
expectWithSymbols(t, `out = dict["a"]`, "foo", SYM{"dict": dict()})
|
||||||
|
|
|
@ -17,6 +17,7 @@ type Script struct {
|
||||||
variables map[string]*Variable
|
variables map[string]*Variable
|
||||||
removedBuiltins map[string]bool
|
removedBuiltins map[string]bool
|
||||||
removedStdModules map[string]bool
|
removedStdModules map[string]bool
|
||||||
|
scriptModules map[string]*Script
|
||||||
userModuleLoader compiler.ModuleLoader
|
userModuleLoader compiler.ModuleLoader
|
||||||
input []byte
|
input []byte
|
||||||
}
|
}
|
||||||
|
@ -79,9 +80,22 @@ func (s *Script) SetUserModuleLoader(loader compiler.ModuleLoader) {
|
||||||
s.userModuleLoader = loader
|
s.userModuleLoader = loader
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// AddModule adds another script as a module. Script module will be
|
||||||
|
// compiled and run right before the main script s is compiled.
|
||||||
|
func (s *Script) AddModule(name string, scriptModule *Script) {
|
||||||
|
if s.scriptModules == nil {
|
||||||
|
s.scriptModules = make(map[string]*Script)
|
||||||
|
}
|
||||||
|
|
||||||
|
s.scriptModules[name] = scriptModule
|
||||||
|
}
|
||||||
|
|
||||||
// Compile compiles the script with all the defined variables, and, returns Compiled object.
|
// Compile compiles the script with all the defined variables, and, returns Compiled object.
|
||||||
func (s *Script) Compile() (*Compiled, error) {
|
func (s *Script) Compile() (*Compiled, error) {
|
||||||
symbolTable, stdModules, globals := s.prepCompile()
|
symbolTable, stdModules, globals, err := s.prepCompile()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
fileSet := source.NewFileSet()
|
fileSet := source.NewFileSet()
|
||||||
|
|
||||||
|
@ -132,7 +146,7 @@ func (s *Script) RunContext(ctx context.Context) (compiled *Compiled, err error)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Script) prepCompile() (symbolTable *compiler.SymbolTable, stdModules map[string]*objects.ImmutableMap, globals []*objects.Object) {
|
func (s *Script) prepCompile() (symbolTable *compiler.SymbolTable, stdModules map[string]*objects.ImmutableMap, globals []*objects.Object, err error) {
|
||||||
var names []string
|
var names []string
|
||||||
for name := range s.variables {
|
for name := range s.variables {
|
||||||
names = append(names, name)
|
names = append(names, name)
|
||||||
|
@ -151,6 +165,38 @@ func (s *Script) prepCompile() (symbolTable *compiler.SymbolTable, stdModules ma
|
||||||
stdModules[name] = mod
|
stdModules[name] = mod
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for name, scriptModule := range s.scriptModules {
|
||||||
|
if scriptModule == nil {
|
||||||
|
err = fmt.Errorf("script module must not be nil: %s", name)
|
||||||
|
}
|
||||||
|
|
||||||
|
var compiledModule *Compiled
|
||||||
|
compiledModule, err = scriptModule.Compile()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
err = compiledModule.Run()
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
mod := &objects.ImmutableMap{
|
||||||
|
Value: make(map[string]objects.Object),
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, symbolName := range compiledModule.symbolTable.Names() {
|
||||||
|
symbol, _, ok := compiledModule.symbolTable.Resolve(symbolName)
|
||||||
|
if ok && symbol.Scope == compiler.ScopeGlobal {
|
||||||
|
value := compiledModule.machine.Globals()[symbol.Index]
|
||||||
|
if value != nil {
|
||||||
|
mod.Value[symbolName] = *value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stdModules[name] = mod
|
||||||
|
}
|
||||||
|
|
||||||
globals = make([]*objects.Object, len(names), len(names))
|
globals = make([]*objects.Object, len(names), len(names))
|
||||||
|
|
||||||
|
|
30
script/script_module_test.go
Normal file
30
script/script_module_test.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package script_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/d5/tengo/assert"
|
||||||
|
"github.com/d5/tengo/script"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestScript_AddModule(t *testing.T) {
|
||||||
|
// mod1 module
|
||||||
|
mod1 := script.New([]byte(`a := 5`))
|
||||||
|
|
||||||
|
// script1 imports "mod1"
|
||||||
|
scr1 := script.New([]byte(`mod1 := import("mod1"); out := mod1.a`))
|
||||||
|
scr1.AddModule("mod1", mod1)
|
||||||
|
c, err := scr1.Run()
|
||||||
|
assert.Equal(t, int64(5), c.Get("out").Value())
|
||||||
|
|
||||||
|
// mod2 module imports "mod1"
|
||||||
|
mod2 := script.New([]byte(`mod1 := import("mod1"); b := mod1.a * 2`))
|
||||||
|
mod2.AddModule("mod1", mod1)
|
||||||
|
|
||||||
|
// script2 imports "mod2" (which imports "mod1")
|
||||||
|
scr2 := script.New([]byte(`mod2 := import("mod2"); out := mod2.b`))
|
||||||
|
scr2.AddModule("mod2", mod2)
|
||||||
|
c, err = scr2.Run()
|
||||||
|
assert.NoError(t, err)
|
||||||
|
assert.Equal(t, int64(10), c.Get("out").Value())
|
||||||
|
}
|
Loading…
Reference in a new issue