wael-lib
v0.0.25
Published
Well-Known Text Arithmetic Expression Language
Maintainers
Readme
Well-Known Text Arithmetic Expression Language (WAEL)
The Well-Known Text Arithmetic Expression Language (WAEL - pronounced like "whale") is an experimental, domain-specific language for generating and manipulating geometry patterns. The language provides spatial geometry types as core data structures and has a syntax similar to Well-Known Text (WKT), with added support for programming features like variables, basic arithmetic, functions and comments. Geometries are immutable and can be transformed using array programming features like geometry arithmetic, or by using pipe transformations (see the Syntax section below for details).
Basic support is currently available for the following 2D geometries: POINT, LINESTRING, POLYGON, MULTIPOINT, MULTILINESTRING, GEOMETRYCOLLECTION. Sections with an ⚠ experimental feature label indicate features that could be updated or modified in future versions.
Try out the language in a CLI interpreter at anthonydgj.github.io/wael
Use the language with an interactive map view at geojsonscript.io with the WAEL code editor option selected.
Usage
Install dependency:
npm install wael-libTo run the interpreter CLI tool (see CLI Usage):
npx waelTo evaluate code from JavaScript:
import { Wael } from 'wael-lib';
const result = Wael.evaluate(`Point(1 1) + Point(2 2)`); // POINT (3 3)Examples
The following examples use language constructs and standard library functions to generate geometry patterns.
Create a 20x10 grid of points with 2-unit spacing starting from coordinates -110, 38:
Point(-110 38) + PointGrid(20, 10, 2)Create the same grid and introduce random offsets:
offset = () => (1 - Math:random() * 2);
Point(-110 38) + PointGrid(20, 10, 2)
|| p => (p + Point(offset() offset()))Rotate a 20x10 grid of points around origin by 23 degrees:
PointGrid(20, 10, 4) | Rotate(23)Create several nested circle polygons:
numRings = 5;
numRings >> (i => (
ring = numRings - i;
PointCircle((ring * 2), (ring * 10)) | ToPolygon
))Command Line Interface (CLI) Usage
The wael CLI tool can be used to evaluate code:
npx wael --helpFollowing the build instructions, a wael binary application can be used:
./wael --helpTo evaluate code and output the resulting WKT, specify one or more input files:
npx wael ./myScript.waelTo output GeoJSON instead of WKT, add the --geojson flag:
npx wael ./myScript.wael --geojsonTo evaluate expressions interactively in a read-eval-print loop (REPL), use the --interactive (or -i) flag.
npx wael -iAll evaluated files, including the interactive environment, will share the same scope. This means that any variables defined in a script file will be accessible in following scripts and the interactive environment, if specified. For example, in the following command, myConstants.wael variables will be accessible to myFunctions.wael, and variables in both scripts will be accessible in the interactive environment.
npx wael ./myConstants.wael ./myFunctions.wael -iExpressions can be passed in directly with the --evaluate (or -e) flag.
npx wael -e "Point(1 1) + Point(2 2)"Any variables defined in the --evaluate script can be used in following script files. For example, the following path.wael script references an undefined start variable:
start ++ GeometryCollection(Point(2 2), Point(3 3), Point(4 4))When evaluated with the following command:
npx wael -e "start = Point(1 1)" path.waelThe start variable will be defined in the --evaluate argument and the output will be:
GEOMETRYCOLLECTION (POINT (1 1), POINT (2 2), POINT (3 3), POINT (4 4))Metadata Variables
The interpreter also provides metadata stored in special variables:
$VERSION: the language version running in the interpreter$SCOPE: a module exporting all variables currently in scope (except numbered history)$HISTORY: a module exporting all numbered historical evaluations$?: the last evaluation result$0,$1, ...$n: previous evaluation results
Syntax
Define geometries using WKT syntax expressions:
GEOMETRYCOLLECTION(
POINT (30 10),
LINESTRING (30 10, 10 30, 40 40),
POLYGON ((30 10, 40 40, 20 40, 10 20, 30 10)),
POLYGON ((35 10, 45 45, 15 40, 10 20, 35 10),
(20 30, 35 35, 30 20, 20 30))
)Multiple expressions are separated by a semi-colon (;) and the last expression is returned after evaluation. For example, evaluating the code:
POINT (1 2);
LINESTRING (1 2, 3 4) will result in LINESTRING (1 2, 3 4)
Expressions are white-space insensitive and case-insensitive, so the following syntax is also valid:
LineString (
1 2 ,
3 4
)Comments
Comments begin with the # character:
# Napoli, Italy
Point(14.19 40.828)Arithmetic
Coordinate values can be expressed using basic numeric arithmetic (+ - * / ^ %):
Point((8 * 3) (-12 + 5)) # POINT (24 -7)Geometries also support basic arithmetic:
Point(1 2) + Point(3 4) # POINT (4 6)Array-like geometries support array programming operations:
LineString(1 1, 2 2, 3 3) + LineString(10 10, 10 10, 10 10); # LINESTRING (11 11, 12 12, 13 13)
LineString(1 1, 2 2, 3 3) - Point(10 10); # LINESTRING (-9 -9, -8 -8, -7 -7)Concatenation
⚠ experimental feature
Array-like geometries can be combined using the concatenate (++) operator:
LineString(1 1, 2 2) ++ LineString(3 3, 4 4); # LINESTRING (1 1, 2 2, 3 3, 4 4)
MultiPoint(1 1, 2 2) ++ MultiPoint(3 3, 4 4); # MULTIPOINT (1 1, 2 2, 3 3, 4 4)
GeometryCollection(Point(1 1)) ++ GeometryCollection(Point(2 2)); # GEOMETRYCOLLECTION(POINT (1 1),POINT (2 2))Points can be appended to point array-like geometries:
LineString(1 1, 2 2) ++ Point(3 3); # LINESTRING (1 1, 2 2, 3 3)
MultiPoint(1 1, 2 2) ++ Point(3 3); # MULTIPOINT (1 1, 2 2, 3 3)
GeometryCollection(Point(1 1)) ++ Point(2 2); # GEOMETRYCOLLECTION(POINT (1 1),POINT (2 2))Non array-like geometries are concatenated into a GEOMETRYCOLLECTION:
Point(1 1) ++ Point(2 2); # GEOMETRYCOLLECTION(POINT (1 1),POINT (2 2))
Point(1 1) ++ Polygon((2 2, 3 3, 4 4, 2 2)); # GEOMETRYCOLLECTION(POINT (1 1),POLYGON ((2 2, 3 3, 4 4, 2 2)))Variables
Variables are defined and assigned a value using the equal (=) operator. Supported data types include:
- Number
- Boolean:
True,False - Geometry:
Point,MultiPoint,LineString,MultiLineString,Polygon,GeometryCollection - Function
longitude = 2; # Number
bool = True; # Boolean
p = Point(longitude 3); # Geometry
fn = (p) => (p + 1); # Function
path = "./my-lib.wael"; # StringThe assignment value is also the evaluation result:
p1 = Point(1 1) # POINT (1 1)Geometry Literals
The GEOMETRYCOLLECTION and POINT geometry types can be created without keywords.
1 1 # POINT (1 1)(1 1, 2 2, 3 3) # GEOMETRYCOLLECTION (POINT (1 1), POINT (2 2), POINT (3 3))Functions
Functions are first-class and declared using the arrow symbol (=>), with a parameter list on the left and function body on the right:
getEquatorPoint = (longitude) => (Point(longitude 0));They can be invoked using parentheses ():
getEquatorPoint(14.19) # POINT (14.19 0)Functions can also accept multiple parameters and have function bodies spanning multiple lines. Similar to top-level expressions outside of a function, the last expression in the function body is used as the return value.
myFn = (x, y, last) => (
first = Point(x y);
LineString(first, last)
);
myFn(1, 2, Point(3 4)) # LINESTRING (1 2, 3 4)The spread operator ... can be used to collect all function arguments into a geometry collection:
Offset = (...g) => (g + 1);
Offset(1 1, 2 2, 3 3) # GEOMETRYCOLLECTION (POINT (2 2), POINT (3 3), POINT (4 4))Variable Scope
Variables have lexical scope and can be accessed or re-assigned from any nested functions:
a = 1;
() => (a = 2)();
a ## 2To assign a variable explicitly within the current scope (and shadow any existing variables in scope with the same name), the let keyword can be used:
a = 1;
() => (let a = 2)();
a ## 1This scoping allows for closures, which can be used to emulate mutable state:
Counter = () => (let c = 0; () => (c = c + 1));
c = Counter();
c();
c() # 2Properties and Methods
Geometry properties and methods can be accessed using the accessor (:) operator:
p = Point(3 4);
p:type; # Point
p:x; # 3
p:y; # 4
g = GeometryCollection(Point(1 2), Point(3 4));
g:type; # GeometryCollection
g:numGeometries; # 2
g:geometryN(1); # POINT (3 4)
l = LineString(1 2, 3 4);
l:type; # LineString
l:numPoints; # 2
l:pointN(1); # POINT (3 4)Geometry properties can be set by calling a method with an appropriate parameter. Since geometries are immutable, a new geometry instance is returned using the updated value:
p = Point(3 4);
p:x(5); # POINT (5 4)
p:y(6); # POINT (3 6)
p # POINT (3 4)Conditional Expressions
Boolean values True and False can be used in logical And, Or or negation ! expressions:
a = True;
b = False;
a And b; # False
a Or b; # True
!a; # FalseNumeric values can be used in comparison expressions < <= > >= == !=, which return a boolean value:
a = Point(1 2);
b = Point(3 4);
a:x < b:x # trueControl flow can be dictated using if-then-else expressions:
result = if (Point(1 2):x < 3)
then (LineString(1 1, 2 2, 3 3))
else (Point(0 0));
result # LINESTRING(1 1, 2 2, 3 3)All three parts of the if-then-else expression are required. The then and else blocks can contain multiple lines, similar to a function body.
points = GeometryCollection(Point(0 0), Point(0 0), Point(0 0), Point(0 0), Point(0 0));
if (points:numGeometries > 3) then (
a = Point(1 2);
b = Point(3 4);
a + b
) else (
a = LineString(1 1, 2 2);
b = LineString(3 3, 4 4);
a + b
) # POINT (4 6)Iteration
⚠ experimental feature
Multiple geometries can be generated using the Iterate operator >> by specifying an iteration count and either a geometry or a function that returns a geometry. The set of all geometries returned from a Generate expression are collected into a GEOMETRYCOLLECTION.
3 >> Point(0 0); # GEOMETRYCOLLECTION(POINT (0 0),POINT (0 0),POINT (0 0))
3 >> (x => Point(x x)) # GEOMETRYCOLLECTION(POINT (0 0),POINT (1 1),POINT (2 2))
3 >> ((x, count) => Point(x x) + count) # GEOMETRYCOLLECTION (POINT (3 3), POINT (4 4), POINT (5 5))The iteration count can also be specified as a variable:
count = 3;
count >> Point(0 0) # GEOMETRYCOLLECTION(POINT (0 0),POINT (0 0),POINT (0 0))A predicate function can be provided to generate geometries while a condition holds:
a = 0;
((i) => a < 3) >>
(i => (a = a + 1; Point(i i))) # GEOMETRYCOLLECTION (POINT (0 0), POINT (1 1), POINT (2 2))Pipe Transformations
Map, Filter and Reduce operators are natively supported.
Mapping
⚠ experimental feature
The output from any expression can be used as the input to a function with the pipe (|) operator:
Point(1 1) | (x) => LineString(x, 2 2) # LINESTRING (1 1, 2 2)Each item in an array-like geometry can be mapped using a function with the double-pipe (||) operator:
LineString(1 1, 2 2, 3 3) || (x => x * x) # LINESTRING (1 1, 4 4, 9 9)The current index is available in a function parameter:
LineString(1 1, 2 2, 3 3) || ((x, i) => x * i) # LINESTRING (0 0, 2 2, 6 6)Each point in a geometry can be transformed using the pipe-all (|*) operator:
LineString(1.4325 1.5325, 2.23525 2.7453, 3.26474 3.34643) |* Round(1) # LINESTRING (1.4 1.5, 2.2 2.7, 3.3 3.3)Filtering
⚠ experimental feature
Array-like geometries can be filtered using the filter (|~) operator:
LineString(1 1, 2 2, 3 3) |~ (p, i) => (p:x <= 2) # LINESTRING (1 1, 2 2)Reducing
⚠ experimental feature
Array-like geometries can be reduced using the reduce (|>) operator:
LineString(1 1, 2 2, 3 3) |> (total, current, index) => (total + current) # Point(6 6)Module System
⚠ experimental feature
Data and expressions can be encapsulated in modules as a local file, network resource or function.
Data can be imported with Use expressions. For example, if the file etna.wael contains Point(14.99 37.75), it can be used in another script with:
data = Use('etna.wael');
data # POINT (14.99 37.75)Supported data formats include WKT, GeoJSON and WAEL.
To use data or code from a network location, provide the URL path as a string:
Lib = Use("https://gist.githubusercontent.com/anthonydgj/29dd64c93e0656475e01bf228f117144/raw/70f007bc469b07ee7b56f17a8df842e167679cec/ext.wael")By default, the last expression in a WAEL script will be returned from a Use expression. To also provide one or more named variables in a module, the export keyword can be used with variable declarations and accessed using the accessor : operator:
MyLib = () => (
export let p = Point(2 2)
);
Lib = Use(MyLib());
Lib:p # POINT (2 2)The With syntax can be used to bring specific variables into scope:
MyLib = () => (
export let myPoint = Point(2 2);
export let myLine = LineString(1 1, 2 2, 3 3)
);
Use(MyLib()) With (myLine);
myLine # LINESTRING (1 1, 2 2, 3 3)Passing * will bring all variables into scope:
MyLib = () => (
export let myPoint = Point(2 2);
export let myLine = LineString(1 1, 2 2, 3 3)
);
Use(MyLib()) With (*);
myLine ++ myPoint # LINESTRING (1 1, 2 2, 3 3)Standard Library
⚠ experimental feature
Several built-in functions are provided to support common geometry generation and transformation operations. To use the standard library, set the interpreter option useStdLib to true or load the library at runtime with:
Use(StdLib()) With (*)Math properties and methods
All JavaScript Math static properties and static functions are accessible from the Math module:
pi = Math:PI; # 3.141592653589793
Math:round(pi) # 3Flatten
Flatten(g) - flatten all geometries in a GEOMETRYCOLLECTION
Flatten(GeometryCollection(Point(1 1), GeometryCollection(Point(2 2)))) # GEOMETRYCOLLECTION(POINT (1 1),POINT (2 2))PointGrid
PointGrid(x, y, spacing) - create a grid of points with the given X and Y count, and (optional) spacing
PointGrid(20, 10, 2) # GEOMETRYCOLLECTION(POINT (0 0),POINT (0 2), ... POINT (38 18))PointCircle
PointCircle(radius, count) - create a circle of points with a given radius and point count
PointCircle(5, 50) # GEOMETRYCOLLECTION(POINT (5 0),POINT (4.9605735065723895 0.6266661678215213), ... )Rotate
Rotate(angleDegrees, originPoint) - get a function to rotate a geometry by the specified degrees around an origin point
MultiPoint(1 1, 2 2, 3 3) | Rotate(23, Point(0 0)) # MULTIPOINT (1.3112079320509338 0.5297935627181312, ... )_Rotate(angleDegrees, originPoint, geometry) - non-pipe version
_Rotate(23, Point(0 0), MultiPoint(1 1, 2 2, 3 3)) # MULTIPOINT (1.3112079320509338 0.5297935627181312, ... )Round
Round(precision) - get a function to round a number or Point coordinates with a given precision (defaults to 0)
1.255 | Round(1) # 1.3_Round(precision, val) - non-pipe version
_Round(1, 1.255) # 1.3ToX
ToLineString(g), ToMultiPoint(g), ToPolygon(g), ToGeometryCollection(g) - convert an array-like geometry of points to a different geometry type
list = GeometryCollection(Point(1 1), Point(2 2), Point(3 3));
ToLineString(list); # LINESTRING (1 1, 2 2, 3 3)
ToMultiPoint(list); # MULTIPOINT (1 1, 2 2, 3 3)
ToPolygon(list); # POLYGON ((1 1, 2 2, 3 3, 1 1))
ToGeometryCollection(list) # GEOMETRYCOLLECTION(POINT (1 1),POINT (2 2),POINT (3 3))Build Instructions
npm install
npm run buildTo build the CLI binary, run:
npm run build-binaryThe binary will be available at:
dist/bin/waelTesting Instructions
npm testImplementation Details
WAEL is implemented with TypeScript using Ohm. When code is evaluated, geometries are stored in an intermediate representation (IR) as GeoJSON objects, which can then be transformed to either WKT or GeoJSON as output.
License
This project is made publicly available under the MIT license (see the LICENSE file).
