East expressions
East expressions are the primary way of adding logic to your Elara solutions. Expressions are created in the Elara Development Kit (EDK) and sent to the Elara platform as a part of the solution template. When performing a task involving user-defined logic, Elara will compile and execute your expressions as a part of that task.
This immediately provides some constraints for how the East programming language was designed and implemented. To integerate with the EDK, we need out-of-the-box support within VS Code and TypeScript (the language of the EDK). The expressions need to be easy to send accross the internet and easy to embed into tasks on the Elara platform. It needs to be easy to check correctness of expressions with the EDK, and any code evaluated on the Elara platform must be safe and secure.
Expressions in the EDK
The EDK is a TypeScript framework to build Elara solutions.
Developers use the EDK to construct solutions Templates consisting of data streams, tasks, portal layouts, etc.
The various components are constructed through Builder
s, which make heavy use of the fluent interfaces (a.k.a. the builder pattern).
Inside components like data pipelines or simulations Elara needs to evaluate your custom logic.
This logic is provided inline in TypeScript using a combination fluent interfaces and East functions exported by the EDK.
East expressions are enterred directly in TypeScript with functions like Add(1, 2)
.
We do not use strings like "1 + 2"
to representing East expressions, and there are no seperate East source files to worry about.
That means East is a programming language without a syntax.
East is a language built entirely inside TypeScript.
We expect that it make take some time for you to get used to working with a language within another language.
Our choice to embed East directly within TypeScript is to minimize complexity for you and us:
- The structure and logic of your solutions can live together in the same
.ts
source file, making it easy to reason about. - TypeScript can check the East types of your expressions and VS Code can provide live programming feedback, for example warning you when your East types don't match within a complex expression.
- The process of constructing solution templates, checking correctness, etc is immensely simplified.
The consequence of this is that East expressions may "look" a little different to other programming languages you have learned, but they fundamentally use the same structure as any other language.
Simple expressions
At their core, East expressions are a simple "tree" of instructions to follow in order to compute some result. Consider these examples:
Add(1, 2)
Multiply(5, Add(1, 2))
Add(Mutply(5, 1), 2)
The first represents 1 + 2 = 3
.
The second is 5 * (1 + 2) = 15
, and the third is (5 * 1) + 2 = 7
.
The first thing to note is that East doesn't support operators like +
, -
, *
or /
.
Because of this, there are no need for "extra" parentheses like used in mathematical expressions - East expressions are either nested inside another expression, or are the unique "root" expression.
The "leaves" of the tree are evaluated first, working our way up towards the root.
In computer science, this tree is often called an "abstract syntax tree" (AST) because it's an abstract tree representation of a programming language's syntax after it is parsed and processed (East doesn't have a concrete syntax).
Our expressions were originally named "Elara AST"s, and the acronym East quickly followed.
There is no limit to how deep or complex expressions can be.
East is a simple expression language where each node is one of a number built-in or "fundamental" operations.
Complex expressions are constructed out of these building blocks.
All expressions have the TypeScript type EastFunction<T extends EastType>
.
There are fundamental East functions for each of the East data types, including:
- Logical operators for Boolean values
- Mathematical operations on floats and integers
- String manipulation and parsing functions
- Construction and deconstruction of structs
- Construction and deconstruction of variants
- Array creation, lookup values, filter, map, reduce, sort, etc.
- Set creation, check for inclusion, filter, map, reduce, union, intersect, etc.
- Dictionary creation, lookup values, filter, map, reduce, etc.
- Equality and comparison for every type
With these building blocks, it is possible to craft intricate logic and transform complex data. A single EDK project will involve tens to thousands of expressions, spread across many tasks to be executed on the Elara platform. Leveraging this, it is possible to construct any program imaginable with Elara.
Note that while the East expression language doesn't allow for defining and calling functions, it is possible to simulate this in TypeScript.
One can simplify code and make EDK projects more modular but crafting TypeScript functions that return complex East expressions as a function of inputs.
A simple example is const Double = (x: EastFunction<FloatType>) => Multiply(x, 2)
, which doubles the input x
.
TypeScript will helpfully infer the type of the output of expressions for us.
A note on values vs. expressions
One thing that we've glossed over is how we represent values in a the AST.
Every node of the AST has TypeScript type EastFunction<T>
.
To make values (including JavaScript objects) unambiguous there is a special node called Const
.
A Const
is a wrapper for a value, along with its type.
For example Const(true)
is an EastFunction<BooleanType>
that contains value true
and type BooleanType
(the East type can optionally be inferred from the value).
The expressions like Add(1, 2)
really are just shorthand for Add(Const(1, FloatType), Const(2, FloatType))
.
This "shorthand" is provided in most places a primitive value is expected.
For more complex values (like structs or arrays) it will usually be necessary to use Const
explicitly.
Similarly, for more complex expressions like Struct
it will be necessary to use Const
even when the values are primitive.
The difference between values and Const
can be tricky when first learning East.
When in doubt, if you are experiencing problems entering an expression try wrap your values in Const
to see if that helps.
Working with variables
The expressions we've seen so far are quite simple.
In fact, they can all be evaluated in advance!
We know that Add(1, 2)
will always evaluate to 3
.
In contrast, solutions deployed on Elara will be data driven and expressions will depend on run-time data.
Like all programming languages, East uses variables to represent data that varies at run time.
Each variable has a distinct name to identify, as well as a fixed EastType
.
A Variable<T>
is one of the possible nodes in an expression; it is an EastFunction<T>
.
Variables are used extensively throughout the EDK.
Consider this simple pipeline that takes a datastream input_stream
which contains an integer (input_stream
is a Stream<IntegerType>
) and increments it by 1:
new PipelineBuilder("increment")
.from(input_stream)
.transform(x => Add(x, 1n))
Here we see the EDK and East expressions working together.
The .transform
method is used to evaluate an arbitrary East expression on the inputs.
In the above the EDK user has provided the function x => Add(x, 1n)
.
The EDK will construct a variable representing the run-time value, and call this function with this variable, resulting in an EastFunction
.
TypeScript will infer the type of x
is Variable<IntegerType>
and that the return type of the expression is a EastFunction<IntegerType>
(In this way we can automatically infer the output type of the pipeline is also IntegerType
).
Once deployed, the pipeline task will inject the value from input_stream
into a variable of the given name and evaluate the East expression.
One thing to note is how in this case the name of the variable is generated automatically by the EDK.
Note that the EDK does not "see" the name of the TypeScript variable x
as it builds your solution template (it simply executes the JavaScript code without reference to the source).
Only sometimes (in FunctionBuilder
and ProcessBuilder
) you will have the opportunity to name your the Variable
s yourself with the .let
method.
In either case, the name of the variable used is unimportant, as the output value does not depend on the Variable
name.
Expressions that create new variables
Users can also create variables to use within expressions.
Certain East functions will result in the creation of new variables.
The most important one to learn is Let
.
Like let
in TypeScript/JavaScript, the Let
function creates a new variable that can be accessed within the current scope.
That means that any variables constructed by Let
go "out of scope" once the expression is fully evaluated.
East expressions do not have code blocks like most modern langauges, so the way Let
works might seem a bit foreign at first.
To clarify, here is an example of Let
in action:
Let(
Multiply(8n, 5n),
x => Add(x, 2n)
)
What this does is first evaluate the first expression.
Once that value is known it creates a new variable (the x
above is a Variable<IntegerType>
) and uses that variable when evaluating the second expression.
The variable x
is only "in scope" within the second expression.
It immediately goes out of scope once the Let
node has been fully evaluated.
The Let
expression returns the final result of the second expression.
In this case we find that x = 8n * 5n
or 40n
, and return x + 2n
which is 42n
.
When is this useful? Sometimes a value is difficult to compute (the code is complex or it takes a long time to compute). When that value needs to be reused, it can be better to assign it to variable and refer to it multiple times rather than compute it all over again. This can result in simpler or faster code.
Expressions that branch
Code that "branches" is code that executes conditionally.
The simplest example is the IfElse
function, which returns results depending on if a "predicate" expression evaluates to true
or false
.
For example, suppose we had two floating-point variables x
and y
and wanted to know which was bigger.
We could construct an appropriate string depending on the run-time values via:
IfElse(
Greater(x, y),
Const("x is bigger"),
Const("y is bigger"),
)
Note the usage of Const
in the second and third expressions.
The second expression (true branch) and third expression (false branch) can be arbitrarily complex.
East will only evaluate the one of the two subexpressions based on the result of the predicate (it will not evaluate both the true branch and the false branch).
IfElse(x, y, z)
is similar to IF(x, y, z)
in Excel and the "ternary" operator x ? y : z
in TypeScript/JavaScript and many other languages.
Note that for branching functions the East data types of the results must be compatibile.
For example, we cannot have the true
branch of IfElse
return a string and the false
branch return an integer.
The East type of the result must be well defined.
Note that "compatible" doesn't mean "identical".
East can automatically merge null
with any type to make it nullable, or merge together different variants to result in a VariantType
that covers all cases.
The other EastFunction
s that branch are IfNull
(for nullable types) and Match
(for variants).
These functions are more complex as they also inject variables into their branch expressions.
Expressions that loop
East expressions are a lot more powerful than some "simple" expression languages.
In East it is possible to iterate over a collection to filter, transform and aggregate results.
However, in East these are not achieved using a for
or while
loop as in many imperative languages.
There are no while
loops in East expressions (though you can use this and other imperative constructs in FunctionBuilder
and ProcessBuilder
).
Instead, East uses constructs from functional programming languages to iterate over collections, like Filter
, Map
and Reduce
.
Take for example Filter
:
Filter(
Const([1, 2, 3, 4, 5]),
x => LessEqual(x, 3)
)
This operation loops over the values in the input array (1
through 5
) and evaluates the expression LessEqual(x, 3)
for each of them.
This returns true
for 1
, 2
and 3
and returns false
for 4
and 5
.
The Filter
function retains the true
values and excludes the false
values, returning a new collection (the input collection is not modified).
The final result is [1, 2, 3]
.
Note that although East does not support first-class functions, the EDK does have higher-order functions.
The EDK manages variable mapping to enforce the appropriate scoping rules (referencial transparency) expected when functions map arguments to parameters as they are called.
Advanved EDK users can use TypeScript to construct their own higher-order functions with care (by using Let
on the arguments to ensure they are not calculated multiple times).
That covers the various "patterns" to expect when constructing East expressions with the EDK.
JSON representation of the AST
Here we provide some insight into how Elara represents expressions in JavaScript and JSON. It is included for completeness; you do not need to understand this to use Elara effectively. For more advanced programmers it might provide some context of how the system works and is designed. Otherwise, feel free to skip this section.
Let's suppose you wanted to encode the expression for 1 + 2
as JSON.
Expressions are represented directly as an "abstract syntax tree", without relying on a source code language.
The EDK provides the Add
function for addition.
Calling Add(1, 2)
results in the following simple JavaScript object:
{
ast_type: "Add",
type: FloatType,
first: Const(1),
second: Const(2),
}
The Const
function also creates simple JavaScript objects, like:
{
ast_type: "Const",
type: FloatType,
value: 1,
}
Note that the value
in Const
is always encoded in a our canonical JSON format. For floats that are not NaN
, Infinity
or -Infinity
this is just a number.
Finally, the types themselves are simple JavaScript objects, such as FloatType
:
{
type: "Float",
value: null,
}
All of these objects are straightforward to serialize with JSON.toString()
and deserialize with JSON.parse()
.
You can inspect the TypeScript definitions of EastFunction
and EastType
to learn more.
Expressions like the above are included inside task definitions inside the solution template, which itself is a JSON document sent to the Elara platform to instantiate a workspace into a hosted solution.
From there, expressions can be evaluated by a variety of techniques, from simple tree-walking interpretation of individual expressions to optimized native-code generation of entire tasks.
Next steps
Now that understand how East expressions allow you to perform safe and performant logic on the Elara platform, continue to the next section to learn how to use and manipulate all the primitive data types in East.