Skip to main content

Structs

Structs are data "structures" containing multiple values associated with field names. Such structures are ubiquitous in computing, from the original struct in C to dynamic objects in JavaScript. They are used to compose data together into larger, more meaningful objects.

Struct values and types

Structs have a fixed set of fields of given name and type. There are an infinite number of distinct struct types. The East type for a struct is created via StructType(fields), where fields is an object mapping field names to East types.

For example StructType({ a: IntegerType, b: StringType }) is the type of a struct containing field a with integer values and field b with string values. The JavaScript representation is a plain object, such as the value { a: 1n, b: "abc" }. The TypeScript type of such values is { a: bigint, b: string }.

East typeExample value (JavaScript)East object notation
StructType({}){}()
StructType({ a: IntegerType, b: StringType }){ a: 1n, b: "abc" }(a = 1, b = "abc",)

Structs are very flexible as you are free to choose the number, names and types of the fields yourself. You will use structs extensively in the EDK.

Structs are plain values

East is not an object-oriented language, and does not support classes. It is not possible to attach methods or functions to structs to augment their behavior. Structs stand alone as values – as plain data.

Furthermore, East uses structural typing and struct types are not nominal types. Two structs with the same field names and types are considered to have the same type. Their values can be compared and used interchangeably.

It may be helpful to consider a struct as like the type of a row in a table. A table has named columns, and the data in each column is expected to have a certain type. A table has many rows, and each row has a set of fields containing a value. In both East and SQL, you can treat any two structs or rows as having the same type if the names and types of the fields agree.

Constructing and deconstructing structs

You can build a new struct with Struct, and get a field from an existing struct with GetField.

East functionDescriptionExample usageResult
StructCreate a new structStruct({ a: Add(1n, 3n), b: Const("abc") })(a = 4, b = "abc",)
GetFieldGet a field value from a structGetField(Const({ a: 4n, b: "abc" }), "a")4n

Note that the Struct function allows you to write an expression to compute each field (like Add above). Direct usage of a JavaScript object will be assumed to be a struct literal value by the EDK. Each field will assumed to be a value, not an expression. If any of the fields need to be computed at runtime, it is essential to use the Struct expression to create a new struct. The EDK will automatically infer the field names and types for you.

The GetField expression is much like the . operator in common programming languages. For example x.a returns the field named a from the struct x.

(Sadly, the EDK cannot use JavaScript syntax like {} or . for constructing East expressions, much like you cannot use + for Add or * for Multiply, so users need to write out Struct and GetField verbosely. Similar constructors exist for variants, arrays, sets and dictionaries.)

Setting fields

Structs in East are immutable. You cannot modify the fields in an existing struct.

However you will often find yourself needing to construct a new value much like an existing struct but with small modifications. For example, you may want to change just one field in a struct. Instead of manually copying each field, the EDK exports the SetField helper function to achieve this goal. This helper function is just nicer "syntax" for constructing a (possibly quite large) Struct expression.

East functionDescriptionExample usageResult
SetFieldCreate a new struct with an updated field valueSetField(Const({ a: 1n, b: "abc" }), "a", 42n)(a = 42, b = "abc",)

You can think of this operation as being like the "spread" syntax { ...struct, field: value } in JavaScript. The original struct is not modified, but a new one with an updated field is created.

Empty structs

Note that StructType({}) is a valid East type. This struct containing zero fields. It has only one possible value, (). Much like null, it takes zero bits of memory to represent such a value.

Comparison and ordering

Two structs (having the same field names) are compared be comparing the values for each field. For the purpose of ordering, the first field is considered first, followed by the second field, and so-on. This is a kind of "lexical ordering".

Note that this ordering allows for e.g. sorting or grouping by a secondary key. An array with values of type StructType({ sort_key: DateTimeType, unique_key: StringType, ... }) can be sorted with Sort. The values will be ordered first by sort_key and then by the (distinct) unique_key. Such patterns enable efficient algorithms (similar to those used by SQL databases) that you can use in your Elara solutions.

Representation

Structs are represented as JSON objects in East's canonical JSON representation, with field names for keys and the values encoded recursively. East's object notation, produced by Print, uses parentheses to delimit structs, comma-seperated entries, and = between the field name and value, like (a = 1, b = 2,).

Next steps

Continue to the next tutorial to understand how to use and manipulate sum types with East and Elara using variants.