In JSLT you write expressions. The expression evaluates to JSON. So the result can be an object, a number, a string, null, etc.
JSON is a subset of JSLT, so anywhere you can write an expression you can write JSON. JSON evaluates to itself.
This is valid JSLT, and will produce itself:
{"foo" : {"bar" : [1,2,3,4,5]}}
JSLT is always evaluated against an input, which is called "the
context node." If the JSON above is the input you can do simple
navigation with dot expressions. .
always returns the context node,
so that would give us back the same JSON.
You can access object keys with .key
, so that .foo
(on the same
input) would give us:
{"bar" : [1,2,3,4,5]}
These can be chained, so .foo.bar
would produce:
[1,2,3,4,5]
If your key contains special characters you can put it in quotes, so
while .Key - Type Strange
would not work ."Key - Type Strange"
would.
You can also do array indexing with [ index ]
, so that .foo.bar[0]
would give you 1
.
Array slicing is supported, so you can write [1 : 3]
and get [2, 3]
. The first index is inclusive, and the last index is exclusive.
You can use negative indexes to refer to elements starting from the
end of the array, so that [-1]
would return the last element of the
array, 5
. This also works with slicing, so to remove the first and
the last element, write [1 : -1]
.
Inside JSON you can write expressions anywhere you can write a JSON literal, so (still working on same input) you could make a new object like this:
{
"array" : .foo.bar
}
which would then produce:
{
"array" : [1,2,3,4,5]
}
Note that if the expression evaluates to null
, {}
, or []
then
the entire key is omitted from the constructed object. This is to
avoid producing objects with lots of no-value keys. If you don't like
this you can change it using the API.
JSLT has a number of built-in functions, like size
, which returns
the number of elements in an object or array, or the number of
characters in a string. The function syntax is the common one, so we
could also write:
{
"array" : .foo.bar,
"size" : size(.foo.bar)
}
which would then produce:
{
"array" : [1,2,3,4,5],
"size" : 5
}
The full set of functions is documented on another page.
As a rule, parameters are quietly converted to the expected type,
using the built-in type conversion functions. In general, if a
parameter is null
(because of missing data in the source JSON) then
the function will just return null
.
We also have if
tests, which use the syntax:
if (<expr>) <expr> else <expr>
if
always returns a value and does nothing else. The else
part can
be left out, in which case the if
will evaluate to null
if the
condition is false.
Boolean false
, null
, empty objects, and empty arrays are all
considered to be false
. That means we could write our transform as
follows:
if (.foo.bar)
{
"array" : [for (.foo.bar) string(.)],
"size" : size(.foo.bar)
}
else
"No array today"
In this case, if there is nothing (or an empty array) in the
.foo.bar
key you will get the string instead. If you want to
distinguish between an empty array and no array at all you can do an
explicit comparison:
if (.foo.bar != null)
{
"array" : [for (.foo.bar) string(.)],
"size" : size(.foo.bar)
}
else
"No array today"
The for
expression lets you take an array and transform it into a
new array, by evaluating an expression on each of the array elements.
The syntax is
[for (<expr>)
<expr>]
The first expression, inside the parenthesis, evaluates to an array,
which we loop over. For each element in it the second expression is
evaluated (with .
now referring to the current array element).
So if we want an array of strings instead, we can say:
{
"array" : [for (.foo.bar) string(.)],
"size" : size(.foo.bar)
}
This will produce:
{
"array" : ["1","2","3","4","5"],
"size" : 5
}
You can also bind variables inside the for
. The example below
produces the same output as the first example.
{
"array" : [for (.foo.bar)
let s = string(.)
$s],
"size" : size(.foo.bar)
}
The for expression also supports filtering the array using an if
expression at the end. With that we could write:
let filtered = [for (.foo.bar) string(.) if (. > 3)]
{
"array" : $filtered,
"size" : size($filtered)
}
Now the output would be:
{
"array" : ["4","5"],
"size" : 2
}
It's also possible to use for
expressions to produce objects. As
before, we process each element in the input array separately, but in
this case we make a key in the object for each element in the array.
When the key and value are evaluated, .
refers to the current array
element.
We can iterate over the same array we've been working on, and turn the array into an object as follows:
{for (.foo.bar) string(.) : .}
This would produce the following object:
{
"1" : 1,
"2" : 2,
"3" : 3,
"4" : 4,
"5" : 5
}
That's a rather strange object, of course, but we can also loop over
objects (with both types of for
loop). Looping over an object turns
the object into an array, where each key/value pair in the object has
become an object of the form:
{"key" : the key, "value" : the value}
So let's say we want to process an object, and add "custom"
as a
prefix to each of the existing keys. Let's say our input was:
{
"foo" : 1,
"bar" : 2
}
We can process this with:
{for (.) "custom_" + .key : .value}
and get:
{
"custom_foo" : 1,
"custom_bar" : 2
}
Just as with object constructors, if the expression evaluates to
null
, {}
, or []
then the entire key is omitted from the
constructed object.
You can bind variables inside the object for, and filter with an if
expression, in the same way as in the array for. Thus we could write:
{for (.)
"custom_" + .key : .value
if (.key != "bar")}
and get:
{
"custom_foo" : 1
}
The usual operators are supported. For arithmetic you can use +
,
-
, *
, and /
as expected.
+
also works on strings, arrays, and objects. For objects, if the
same key appears on both sides, the value on the left will "win".
The usual comparison operators are also supported, so you could process the 1-5 array with
[for (.foo.bar) . > 2]
and get
[false, false, true, true, true]
You can also combine boolean values with and
and or
. not
is a
function. So you can write a boolean expression as:
.foo.bar > 2 and not(contains(.baz, ["george", "harry"]))
Strings can be concatenated with +
, and any object can be turned
into a string with string( ... )
. Strings can also be sliced the
same way as arrays: .name[ : 5]
will give the first five characters
of the string. The size( .name )
function gives you the length of
the string.
For more ways to process strings, see the string functions.
In some cases, you want to insert or override some values in an
object, but leave the rest untouched. For example, let's say we want
to multiply "foo"
by 10, but keep the rest of the object. This can
be done with object matching, where we write an object like this:
{
"foo" : .foo * 10,
* : .
}
This is a normal JSON object, except for the last part, * : .
. The
*
matches all keys in the input object, except for those that are
already specified ("foo"
in this case) and copies them into the
output object. The .
is a normal expression, evaluated with the
value of each key in turn.
So the output is:
{
"foo" : 10,
"bar" : 2
}
Since .
is a normal expression, you could actually write:
{ * : . * 10 }
That would match and copy all keys, but all values would be multiplied by 10, so the output would be:
{
"foo" : 10,
"bar" : 20
}
If there are some keys you don't want to copy you can explicitly omit them:
{
"foo" : .foo * 10,
* - bar, baz, quux : .
}
This would produce:
{"foo" : 10}
If your input is a nested object you can still use the matching. So if you write:
{
"foo" : {
"baz" : .hey.ho,
* : .
},
"bar" : 24
}
Then this *
will match the object inside the "foo"
key in the
input object.
You can set variables to break up complex computations, or to avoid computing things more than once. The syntax is:
let foo = .foo
From that point on, you can refer to the value with $foo
.
Let statements are allowed at the top level, at the start of objects,
inside for
(before the expression), and inside if
. Variables
defined inside an object, for
, or if
are only visible inside those
expressions.
It's also possible to set variables from the code running a JSLT transform, so that the value is "injected" into the transform. This can be useful for configuration, passwords, and similar values.
You can declare your own functions at the top level in expressions and templates like so:
def sum(array)
if ($array)
$array[0] + sum($array[1 : ])
else
0
This function sums the numbers in an array, returning the total.
The syntax is simple: first declare the name and the parameters, then follow that with an expression that evaluates to the results. You can define variables inside the function. Functions can call themselves, and they can call other functions you define.
Here is a function that counts the number of keys in an object,
including objects contained within it, building on the sum
function
above.
def count(node)
if (is-object($node))
size($node) + sum([for ($node) count(.value)])
else if (is-array($node))
sum([for ($node) count(.)])
else
0
Functions can be declared in any order. You can declare a function with the same name as a built-in or extension function, and your declaration will shadow the original.
In order to make it possible to modularize transforms by reusing code JSLT allows you to import JSLT modules from files. These are exactly like ordinary JSLT templates, except that the final expression after the variable and function declarations is not required.
If the two functions above were saved in a file named utilities.jstl
then we could use them in another transform as follows:
import "utilities.jstl" as utils
{
"type" : "object",
"size" : size(.),
"keys" : utils:count(.)
}
The import
statement must appear before any variable or function
declarations. You can have any number of them, and a module can import
other modules. Cyclic imports are not allowed.
For now, modules are imported from the classpath, and there is no resolution of relative paths.