2019-04-19 09:33:23 +03:00
|
|
|
|
# Tengo Language Syntax
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Tengo's syntax is designed to be familiar to Go developers while being a bit
|
|
|
|
|
simpler and more streamlined.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
**You can test the Tengo code in online [Playground](https://tengolang.com).**
|
|
|
|
|
|
|
|
|
|
## Values and Value Types
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
In Tengo, everything is a value, and, all values are associated with a type.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
19 + 84 // int values
|
|
|
|
|
"aomame" + `kawa` // string values
|
|
|
|
|
-9.22 + 1e10 // float values
|
|
|
|
|
true || false // bool values
|
|
|
|
|
'九' > '9' // char values
|
|
|
|
|
[1, false, "foo"] // array value
|
|
|
|
|
{a: 12.34, b: "bar"} // map value
|
|
|
|
|
func() { /*...*/ } // function value
|
|
|
|
|
```
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
Here's a list of all available value types in Tengo.
|
|
|
|
|
|
|
|
|
|
| Tengo Type | Description | Equivalent Type in Go |
|
|
|
|
|
| :---: | :---: | :---: |
|
|
|
|
|
| int | signed 64-bit integer value | `int64` |
|
|
|
|
|
| float | 64-bit floating point value | `float64` |
|
|
|
|
|
| bool | boolean value | `bool` |
|
|
|
|
|
| char | unicode character | `rune` |
|
2020-03-04 17:53:38 +03:00
|
|
|
|
| string | unicode string | `string` |
|
2019-04-19 09:33:23 +03:00
|
|
|
|
| bytes | byte array | `[]byte` |
|
|
|
|
|
| error | [error](#error-values) value | - |
|
|
|
|
|
| time | time value | `time.Time` |
|
2023-08-22 07:26:27 +03:00
|
|
|
|
| array | value array _(mutable)_ | `[]any` |
|
2019-04-19 09:33:23 +03:00
|
|
|
|
| immutable array | [immutable](#immutable-values) array | - |
|
2023-08-22 07:26:27 +03:00
|
|
|
|
| map | value map with string keys _(mutable)_ | `map[string]any` |
|
2019-04-19 09:33:23 +03:00
|
|
|
|
| immutable map | [immutable](#immutable-values) map | - |
|
|
|
|
|
| undefined | [undefined](#undefined-values) value | - |
|
2021-11-14 02:13:39 +03:00
|
|
|
|
| function | [function](#function-values) value | - |
|
2019-04-19 09:33:23 +03:00
|
|
|
|
| _user-defined_ | value of [user-defined types](https://github.com/d5/tengo/blob/master/docs/objects.md) | - |
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Error Values
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
In Tengo, an error can be represented using "error" typed values. An error
|
|
|
|
|
value is created using `error` expression, and, it must have an underlying
|
|
|
|
|
value. The underlying value of an error value can be access using `.value`
|
2021-11-14 02:13:39 +03:00
|
|
|
|
selector.
|
2020-03-04 17:53:38 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
```golang
|
|
|
|
|
err1 := error("oops") // error with string value
|
|
|
|
|
err2 := error(1+2+3) // error with int value
|
|
|
|
|
if is_error(err1) { // 'is_error' builtin function
|
2020-03-04 17:53:38 +03:00
|
|
|
|
err_val := err1.value // get underlying value
|
2021-11-14 02:13:39 +03:00
|
|
|
|
}
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Immutable Values
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
In Tengo, basically all values (except for array and map) are immutable.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
s := "12345"
|
2020-03-04 17:53:38 +03:00
|
|
|
|
s[1] = 'b' // illegal: String is immutable
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
a := [1, 2, 3]
|
|
|
|
|
a[1] = "two" // ok: a is now [1, "two", 3]
|
2020-03-04 17:53:38 +03:00
|
|
|
|
```
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
An array or map value can be made immutable using `immutable` expression.
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
b := immutable([1, 2, 3])
|
|
|
|
|
b[1] = "foo" // illegal: 'b' references to an immutable array.
|
2020-03-04 17:53:38 +03:00
|
|
|
|
```
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Note that re-assigning a new value to the variable has nothing to do with the
|
|
|
|
|
value immutability.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
s := "abc"
|
|
|
|
|
s = "foo" // ok
|
|
|
|
|
a := immutable([1, 2, 3])
|
|
|
|
|
a = false // ok
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
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.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
a := immutable({b: 4, c: [1, 2, 3]})
|
|
|
|
|
a.b = 5 // illegal
|
|
|
|
|
a.c[1] = 5 // ok: because 'a.c' is not immutable
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
a = immutable({b: 4, c: immutable([1, 2, 3])})
|
2019-04-19 09:33:23 +03:00
|
|
|
|
a.c[1] = 5 // illegal
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Undefined Values
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
In Tengo, an "undefined" value can be used to represent an unexpected or
|
2021-11-14 02:13:39 +03:00
|
|
|
|
non-existing value:
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- A function that does not return a value explicitly considered to return
|
2020-03-04 17:53:38 +03:00
|
|
|
|
`undefined` value.
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- Indexer or selector on composite value types may return `undefined` if the
|
2021-11-14 02:13:39 +03:00
|
|
|
|
key or index does not exist.
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- Type conversion builtin functions without a default value will return
|
|
|
|
|
`undefined` if conversion fails.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
a := func() { b := 4 }() // a == undefined
|
|
|
|
|
b := [1, 2, 3][10] // b == undefined
|
|
|
|
|
c := {a: "foo"}["b"] // c == undefined
|
|
|
|
|
d := int("foo") // d == undefined
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Array Values
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
In Tengo, array is an ordered list of values of any types. Elements of an array
|
|
|
|
|
can be accessed using indexer `[]`.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
[1, 2, 3][0] // == 1
|
|
|
|
|
[1, 2, 3][2] // == 3
|
|
|
|
|
[1, 2, 3][3] // == undefined
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
["foo", "bar", [1, 2, 3]] // ok: array with an array element
|
|
|
|
|
```
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Map Values
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
In Tengo, map is a set of key-value pairs where key is string and the value is
|
|
|
|
|
of any value types. Value of a map can be accessed using indexer `[]` or
|
|
|
|
|
selector '.' operators.
|
2019-02-10 03:45:21 +03:00
|
|
|
|
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
m := { a: 1, b: false, c: "foo" }
|
|
|
|
|
m["b"] // == false
|
|
|
|
|
m.c // == "foo"
|
|
|
|
|
m.x // == undefined
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2021-11-14 02:13:39 +03:00
|
|
|
|
{a: [1,2,3], b: {c: "foo", d: "bar"}} // ok: map with an array element and a map element
|
|
|
|
|
```
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
### Function Values
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
In Tengo, function is a callable value with a number of function arguments and
|
|
|
|
|
a return value. Just like any other values, functions can be passed into or
|
2020-03-04 17:53:38 +03:00
|
|
|
|
returned from another function.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
my_func := func(arg1, arg2) {
|
|
|
|
|
return arg1 + arg2
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
adder := func(base) {
|
|
|
|
|
return func(x) { return base + x } // capturing 'base'
|
|
|
|
|
}
|
|
|
|
|
add5 := adder(5)
|
|
|
|
|
nine := add5(4) // == 9
|
|
|
|
|
```
|
2020-03-04 17:53:38 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
Unlike Go, Tengo does not have declarations. So the following code is illegal:
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
func my_func(arg1, arg2) { // illegal
|
|
|
|
|
return arg1 + arg2
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2019-04-26 07:28:27 +03:00
|
|
|
|
Tengo also supports variadic functions/closures:
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
variadic := func (a, b, ...c) {
|
|
|
|
|
return [a, b, c]
|
|
|
|
|
}
|
|
|
|
|
variadic(1, 2, 3, 4) // [1, 2, [3, 4]]
|
|
|
|
|
|
|
|
|
|
variadicClosure := func(a) {
|
|
|
|
|
return func(b, ...c) {
|
|
|
|
|
return [a, b, c]
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
variadicClosure(1)(2, 3, 4) // [1, 2, [3, 4]]
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Only the last parameter can be variadic. The following code is also illegal:
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
// illegal, because a is variadic and is not the last parameter
|
|
|
|
|
illegal := func(a..., b) { /*... */ }
|
|
|
|
|
```
|
|
|
|
|
|
2020-06-08 19:54:24 +03:00
|
|
|
|
When calling a function, the number of passing arguments must match that of
|
|
|
|
|
function definition.
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
f := func(a, b) {}
|
|
|
|
|
f(1, 2, 3) // Runtime Error: wrong number of arguments: want=2, got=3
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Like Go, you can use ellipsis `...` to pass array-type value as its last parameter:
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
f1 := func(a, b, c) { return a + b + c }
|
|
|
|
|
f1([1, 2, 3]...) // => 6
|
|
|
|
|
f1(1, [2, 3]...) // => 6
|
|
|
|
|
f1(1, 2, [3]...) // => 6
|
|
|
|
|
f1([1, 2]...) // Runtime Error: wrong number of arguments: want=3, got=2
|
|
|
|
|
|
|
|
|
|
f2 := func(a, ...b) {}
|
|
|
|
|
f2(1) // valid; a = 1, b = []
|
|
|
|
|
f2(1, 2) // valid; a = 1, b = [2]
|
|
|
|
|
f2(1, 2, 3) // valid; a = 1, b = [2, 3]
|
|
|
|
|
f2([1, 2, 3]...) // valid; a = 1, b = [2, 3]
|
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
## Variables and Scopes
|
|
|
|
|
|
|
|
|
|
A value can be assigned to a variable using assignment operator `:=` and `=`.
|
|
|
|
|
|
|
|
|
|
- `:=` operator defines a new variable in the scope and assigns a value.
|
2020-03-04 17:53:38 +03:00
|
|
|
|
- `=` operator assigns a new value to an existing variable in the scope.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Variables are defined either in global scope (defined outside function) or in
|
2020-03-04 17:53:38 +03:00
|
|
|
|
local scope (defined inside function).
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := "foo" // define 'a' in global scope
|
|
|
|
|
|
|
|
|
|
func() { // function scope A
|
|
|
|
|
b := 52 // define 'b' in function scope A
|
2021-11-14 02:13:39 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
func() { // function scope B
|
|
|
|
|
c := 19.84 // define 'c' in function scope B
|
2020-03-04 17:53:38 +03:00
|
|
|
|
|
|
|
|
|
a = "bee" // ok: assign new value to 'a' from global scope
|
2019-04-19 09:33:23 +03:00
|
|
|
|
b = 20 // ok: assign new value to 'b' from function scope A
|
2020-03-04 17:53:38 +03:00
|
|
|
|
|
|
|
|
|
b := true // ok: define new 'b' in function scope B
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// (shadowing 'b' from function scope A)
|
|
|
|
|
}
|
2021-11-14 02:13:39 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
a = "bar" // ok: assigne new value to 'a' from global scope
|
|
|
|
|
b = 10 // ok: assigne new value to 'b'
|
|
|
|
|
a := -100 // ok: define new 'a' in function scope A
|
|
|
|
|
// (shadowing 'a' from global scope)
|
2021-11-14 02:13:39 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
c = -9.1 // illegal: 'c' is not defined
|
|
|
|
|
b := [1, 2] // illegal: 'b' is already defined in the same scope
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
b = 25 // illegal: 'b' is not defined
|
|
|
|
|
a := {d: 2} // illegal: 'a' is already defined in the same scope
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Unlike Go, a variable can be assigned a value of different types.
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := 123 // assigned 'int'
|
|
|
|
|
a = "123" // re-assigned 'string'
|
|
|
|
|
a = [1, 2, 3] // re-assigned 'array'
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
## Type Conversions
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Although the type is not directly specified in Tengo, one can use type
|
|
|
|
|
conversion
|
|
|
|
|
[builtin functions](https://github.com/d5/tengo/blob/master/docs/builtins.md)
|
2020-03-04 17:53:38 +03:00
|
|
|
|
to convert between value types.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
s1 := string(1984) // "1984"
|
|
|
|
|
i2 := int("-999") // -999
|
|
|
|
|
f3 := float(-51) // -51.0
|
|
|
|
|
b4 := bool(1) // true
|
|
|
|
|
c5 := char("X") // 'X'
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
See [Operators](https://github.com/d5/tengo/blob/master/docs/operators.md)
|
|
|
|
|
for more details on type coercions.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
## Operators
|
|
|
|
|
|
|
|
|
|
### Unary Operators
|
|
|
|
|
|
|
|
|
|
| Operator | Usage | Types |
|
|
|
|
|
| :---: | :---: | :---: |
|
|
|
|
|
| `+` | same as `0 + x` | int, float |
|
|
|
|
|
| `-` | same as `0 - x` | int, float |
|
|
|
|
|
| `!` | logical NOT | all types* |
|
|
|
|
|
| `^` | bitwise complement | int |
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
_In Tengo, all values can be either
|
|
|
|
|
[truthy or falsy](https://github.com/d5/tengo/blob/d5-patch-1/docs/runtime-types.md#objectisfalsy)._
|
2019-01-28 04:33:24 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
### Binary Operators
|
|
|
|
|
|
|
|
|
|
| Operator | Usage | Types |
|
|
|
|
|
| :---: | :---: | :---: |
|
|
|
|
|
| `==` | equal | all types |
|
|
|
|
|
| `!=` | not equal | all types |
|
|
|
|
|
| `&&` | logical AND | all types |
|
|
|
|
|
| `\|\|` | logical OR | all types |
|
|
|
|
|
| `+` | add/concat | int, float, string, char, time, array |
|
|
|
|
|
| `-` | subtract | int, float, char, time |
|
|
|
|
|
| `*` | multiply | int, float |
|
|
|
|
|
| `/` | divide | int, float |
|
|
|
|
|
| `&` | bitwise AND | int |
|
|
|
|
|
| `\|` | bitwise OR | int |
|
|
|
|
|
| `^` | bitwise XOR | int |
|
|
|
|
|
| `&^` | bitclear (AND NOT) | int |
|
|
|
|
|
| `<<` | shift left | int |
|
|
|
|
|
| `>>` | shift right | int |
|
2020-05-25 20:53:52 +03:00
|
|
|
|
| `<` | less than | int, float, char, time, string |
|
|
|
|
|
| `<=` | less than or equal to | int, float, char, time, string |
|
|
|
|
|
| `>` | greater than | int, float, char, time, string |
|
|
|
|
|
| `>=` | greater than or equal to | int, float, char, time, string |
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
2020-05-25 20:53:52 +03:00
|
|
|
|
_See [Operators](https://github.com/d5/tengo/blob/master/docs/operators.md)
|
2019-12-20 22:40:38 +03:00
|
|
|
|
for more details._
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
### Ternary Operators
|
|
|
|
|
|
|
|
|
|
Tengo has a ternary conditional operator `(condition expression) ? (true expression) : (false expression)`.
|
2019-01-28 04:33:24 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := true ? 1 : -1 // a == 1
|
|
|
|
|
|
|
|
|
|
min := func(a, b) {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
return a < b ? a : b
|
2019-01-28 04:33:24 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
b := min(5, 10) // b == 5
|
2019-01-28 04:33:24 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
### Assignment and Increment Operators
|
|
|
|
|
|
|
|
|
|
| Operator | Usage |
|
|
|
|
|
| :---: | :---: |
|
|
|
|
|
| `+=` | `(lhs) = (lhs) + (rhs)` |
|
|
|
|
|
| `-=` | `(lhs) = (lhs) - (rhs)` |
|
|
|
|
|
| `*=` | `(lhs) = (lhs) * (rhs)` |
|
|
|
|
|
| `/=` | `(lhs) = (lhs) / (rhs)` |
|
|
|
|
|
| `%=` | `(lhs) = (lhs) % (rhs)` |
|
|
|
|
|
| `&=` | `(lhs) = (lhs) & (rhs)` |
|
|
|
|
|
| `\|=` | `(lhs) = (lhs) \| (rhs)` |
|
|
|
|
|
| `&^=` | `(lhs) = (lhs) &^ (rhs)` |
|
|
|
|
|
| `^=` | `(lhs) = (lhs) ^ (rhs)` |
|
|
|
|
|
| `<<=` | `(lhs) = (lhs) << (rhs)` |
|
|
|
|
|
| `>>=` | `(lhs) = (lhs) >> (rhs)` |
|
|
|
|
|
| `++` | `(lhs) = (lhs) + 1` |
|
|
|
|
|
| `--` | `(lhs) = (lhs) - 1` |
|
|
|
|
|
|
|
|
|
|
### Operator Precedences
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Unary operators have the highest precedence, and, ternary operator has the
|
|
|
|
|
lowest precedence. There are five precedence levels for binary operators.
|
|
|
|
|
Multiplication operators bind strongest, followed by addition operators,
|
|
|
|
|
comparison operators, `&&` (logical AND), and finally `||` (logical OR):
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
| Precedence | Operator |
|
|
|
|
|
| :---: | :---: |
|
|
|
|
|
| 5 | `*` `/` `%` `<<` `>>` `&` `&^` |
|
|
|
|
|
| 4 | `+` `-` `\|` `^` |
|
|
|
|
|
| 3 | `==` `!=` `<` `<=` `>` `>=` |
|
|
|
|
|
| 2 | `&&` |
|
|
|
|
|
| 1 | `\|\|` |
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Like Go, `++` and `--` operators form statements, not expressions, they fall
|
2020-03-04 17:53:38 +03:00
|
|
|
|
outside the operator hierarchy.
|
2019-01-28 04:33:24 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
### Selector and Indexer
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
One can use selector (`.`) and indexer (`[]`) operators to read or write
|
|
|
|
|
elements of composite types (array, map, string, bytes).
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
["one", "two", "three"][1] // == "two"
|
|
|
|
|
|
|
|
|
|
m := {
|
|
|
|
|
a: 1,
|
|
|
|
|
b: [2, 3, 4],
|
|
|
|
|
c: func() { return 10 }
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
m.a // == 1
|
|
|
|
|
m["b"][1] // == 3
|
|
|
|
|
m.c() // == 10
|
|
|
|
|
m.x = 5 // add 'x' to map 'm'
|
|
|
|
|
m["b"][5] // == undefined
|
|
|
|
|
m["b"][5].d // == undefined
|
|
|
|
|
m.b[5] = 0 // == undefined
|
2020-03-04 17:53:38 +03:00
|
|
|
|
m.x.y.z // == undefined
|
2019-04-19 09:33:23 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Like Go, one can use slice operator `[:]` for sequence value types such as
|
|
|
|
|
array, string, bytes.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := [1, 2, 3, 4, 5][1:3] // == [2, 3]
|
|
|
|
|
b := [1, 2, 3, 4, 5][3:] // == [4, 5]
|
|
|
|
|
c := [1, 2, 3, 4, 5][:3] // == [1, 2, 3]
|
|
|
|
|
d := "hello world"[2:10] // == "llo worl"
|
|
|
|
|
c := [1, 2, 3, 4, 5][-1:10] // == [1, 2, 3, 4, 5]
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2020-06-08 19:54:24 +03:00
|
|
|
|
**Note: Keywords cannot be used as selectors.**
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := {in: true} // Parse Error: expected map key, found 'in'
|
|
|
|
|
a.func = "" // Parse Error: expected selector, found 'func'
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Use double quotes and indexer to use keywords with maps.
|
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
a := {"in": true}
|
|
|
|
|
a["func"] = ""
|
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
## Statements
|
|
|
|
|
|
|
|
|
|
### If Statement
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
"If" statement is very similar to Go.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
if a < 0 {
|
2020-03-04 17:53:38 +03:00
|
|
|
|
// execute if 'a' is negative
|
2019-01-24 03:30:41 +03:00
|
|
|
|
} else if a == 0 {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// execute if 'a' is zero
|
2019-01-24 03:30:41 +03:00
|
|
|
|
} else {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// execute if 'a' is positive
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
```
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Like Go, the condition expression may be preceded by a simple statement,
|
2020-03-04 17:53:38 +03:00
|
|
|
|
which executes before the expression is evaluated.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
if a := foo(); a < 0 {
|
|
|
|
|
// execute if 'a' is negative
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### For Statement
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
"For" statement is very similar to Go.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
// for (init); (condition); (post) {}
|
2019-01-24 03:30:41 +03:00
|
|
|
|
for a:=0; a<10; a++ {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// ...
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// for (condition) {}
|
2019-01-24 03:30:41 +03:00
|
|
|
|
for a < 10 {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// ...
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
// for {}
|
|
|
|
|
for {
|
|
|
|
|
// ...
|
2019-01-24 03:30:41 +03:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
### For-In Statement
|
2019-01-25 06:31:17 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
"For-In" statement is new in Tengo. It's similar to Go's `for range` statement.
|
|
|
|
|
"For-In" statement can iterate any iterable value types (array, map, bytes,
|
2021-11-14 02:13:39 +03:00
|
|
|
|
string, undefined).
|
2019-01-25 06:31:17 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
for v in [1, 2, 3] { // array: element
|
|
|
|
|
// 'v' is value
|
|
|
|
|
}
|
|
|
|
|
for i, v in [1, 2, 3] { // array: index and element
|
|
|
|
|
// 'i' is index
|
2021-11-14 02:13:39 +03:00
|
|
|
|
// 'v' is value
|
2020-03-04 17:53:38 +03:00
|
|
|
|
}
|
2019-04-19 09:33:23 +03:00
|
|
|
|
for k, v in {k1: 1, k2: 2} { // map: key and value
|
|
|
|
|
// 'k' is key
|
|
|
|
|
// 'v' is value
|
|
|
|
|
}
|
2019-01-25 06:31:17 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-01-24 03:30:41 +03:00
|
|
|
|
## Modules
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Module is the basic compilation unit in Tengo. A module can import another
|
2020-03-04 17:53:38 +03:00
|
|
|
|
module using `import` expression.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
Main module:
|
2019-02-02 10:27:29 +03:00
|
|
|
|
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```golang
|
2019-04-19 09:33:23 +03:00
|
|
|
|
sum := import("./sum") // load module from a local file
|
2020-03-04 17:53:38 +03:00
|
|
|
|
fmt.print(sum(10)) // module function
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
Another module in `sum.tengo` file:
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2019-02-02 10:27:29 +03:00
|
|
|
|
base := 5
|
|
|
|
|
|
|
|
|
|
export func(x) {
|
2019-04-19 09:33:23 +03:00
|
|
|
|
return x + base
|
2019-02-02 10:27:29 +03:00
|
|
|
|
}
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
|
|
|
|
|
2021-11-14 02:13:39 +03:00
|
|
|
|
By default, `import` solves the missing extension name of a module file as
|
|
|
|
|
"`.tengo`"[^note].
|
|
|
|
|
Thus, `sum := import("./sum")` is equivalent to `sum := import("./sum.tengo")`.
|
|
|
|
|
|
|
|
|
|
[^note]:
|
|
|
|
|
If using Tengo as a library in Go, the file extension name "`.tengo`" can
|
|
|
|
|
be customized. In that case, use the `SetImportFileExt` function of the
|
|
|
|
|
`Compiler` type.
|
|
|
|
|
See the [Go reference](https://pkg.go.dev/github.com/d5/tengo/v2) for details.
|
|
|
|
|
|
2019-02-02 10:27:29 +03:00
|
|
|
|
In Tengo, modules are very similar to functions.
|
|
|
|
|
|
2020-03-04 17:53:38 +03:00
|
|
|
|
- `import` expression loads the module code and execute it like a function.
|
2019-02-02 10:27:29 +03:00
|
|
|
|
- Module should return a value using `export` statement.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
- Module can return `export` a value of any types: int, map, function, etc.
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- `export` in a module is like `return` in a function: it stops execution and
|
|
|
|
|
return a value to the importing code.
|
2019-04-19 09:33:23 +03:00
|
|
|
|
- `export`-ed values are always immutable.
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- If the module does not have any `export` statement, `import` expression
|
2021-11-14 02:13:39 +03:00
|
|
|
|
simply returns `undefined`. _(Just like the function that has no `return`.)_
|
2019-12-20 22:40:38 +03:00
|
|
|
|
- Note that `export` statement is completely ignored and not evaluated if
|
2021-11-14 02:13:39 +03:00
|
|
|
|
the code is executed as a main module.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Also, you can use `import` expression to load the
|
|
|
|
|
[Standard Library](https://github.com/d5/tengo/blob/master/docs/stdlib.md) as
|
|
|
|
|
well.
|
2019-01-24 03:30:41 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
|
|
|
|
math := import("math")
|
2019-04-19 09:33:23 +03:00
|
|
|
|
a := math.abs(-19.84) // == 19.84
|
|
|
|
|
```
|
2020-03-04 17:53:38 +03:00
|
|
|
|
|
2019-04-19 09:33:23 +03:00
|
|
|
|
## Comments
|
|
|
|
|
|
2019-12-20 22:40:38 +03:00
|
|
|
|
Like Go, Tengo supports line comments (`//...`) and block comments
|
|
|
|
|
(`/* ... */`).
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
```golang
|
2020-03-04 17:53:38 +03:00
|
|
|
|
/*
|
|
|
|
|
multi-line block comments
|
2019-04-19 09:33:23 +03:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
a := 5 // line comments
|
2019-01-24 03:30:41 +03:00
|
|
|
|
```
|
2019-04-19 09:33:23 +03:00
|
|
|
|
|
|
|
|
|
## Differences from Go
|
|
|
|
|
|
|
|
|
|
Unlike Go, Tengo does not have the following:
|
|
|
|
|
|
|
|
|
|
- Declarations
|
|
|
|
|
- Imaginary values
|
|
|
|
|
- Structs
|
|
|
|
|
- Pointers
|
|
|
|
|
- Channels
|
|
|
|
|
- Goroutines
|
|
|
|
|
- Tuple assignment
|
|
|
|
|
- Variable parameters
|
|
|
|
|
- Switch statement
|
|
|
|
|
- Goto statement
|
|
|
|
|
- Defer statement
|
2020-03-04 17:53:38 +03:00
|
|
|
|
- Panic
|
2019-04-19 09:33:23 +03:00
|
|
|
|
- Type assertion
|