Expressions in flespi are used everywhere — in interval selectors, in interval counters, in REST API selectors, MQTT topic filtering, in webhooks and various validators. Whenever you see a word "expression" in flespi it means flespi expression which format is described below.
Expression operates with a JSON object and the result of an expression is a typed number, string, boolean or null value. In the case of booleantype expressions (various validators and REST API selectors), a nonzero resulting value means the expression is valid and zero, false or null will deny validation of the message or interval.
All numbers are operated as floatingpoint with double precision, thus limiting integers to 53 bits in size.
An expression consists of numbers, operators, parameters references, and functions. If an expression contains a reference to a field that is not present in the object or performs division by zero, its result is undefined or OFF in most boolean expressions.
Important! You can now ask our AI Assistant (codi) to generate correct expression for your task. And we have released a handy expressions testing tool that may greatly simplify debugging of the flespi expressions.
For example, we sequentially analyze the following messages from devices:
{
"device.name": "123456789012345",
"timestamp": 1490347944.893743,
"din": 9,
"channel_id": 123,
"altitude": 568.49,
"direction": 297,
"hdop": 0.9,
"speed": 10,
"lat": 21.328481,
"lng": 47.562136,
"accelerations": [1, 2, 3.3, 0],
...
}
Expressions samples that can be used for such kinds of objects are:
 device.name == "12*45"
 timestamp > 1490347940 && timestamp < 1490347950
 (hdop < 1 && (din & 0x1))  speed == 0
 if(altitude > 500 && altitude < 500, altitude, error())
In an expression, all references to fields that contain a number, boolean, textual, or null value will equal this value. All references to other types of fields (e.g. array, nested object, or binary) will make expression evaluation impossible.
It is possible to use true, false, and null values in the expression. The type of value can be tested with functions: isnull, isnumber, isstring, isboolean, isjson and safely converted to another type with functions tonumber, toboolean, tostring.
In mathematical comparison operators, the maximum level of automatic type casting is used: null and false are cast to zero, true or any valid string — to one.
For textual fields you can use comparison operators like '==' or '!=', for example: "device.name == '123456789012345'" to check the actual value. When comparing two strings, you may use wildcards, for example: "device.name == 'Telto*'". For caseinsensitive comparison, use a special operator "~", e.g. "device.name ~ 'TeLTo??kA'".
If you need to access nested (JSON) objects or arrays, use a notation objectkey['jsonpath'], for example ble.sensors['/0/rssi'] to access BLE sensor with first index and its rssi value. If unable to extract value under path, it will always return null. See more examples and explanations for json function in the table below.
In special sequential operations, like within analytics, to reference the value of the same parameter in the previous message, use previous("parametername") function. For example, you can use "speed  previous('speed')" to calculate the speed change from message to message.
Note(1): reregistered messages cannot be referred to by the previous() function; previous values of parameters are only available for messages that are newer (fresher) than all the existing ones.
Note(2): expressions in plugins define different meaning for previous("parametername") function. It is used to access the current telemetry value of device. If the timestamp of the message that is being registered is earlier than the value in device's telemetry this function will return null.
Note(3): when you use previous("parametername") function in calculator expressions if the system does not know the value of the previous parameter for the very first message, it means that the very first message in such reference will always return an unknown or OFF value. In order to correctly and consistently calculate intervals, we recommend using merge_message_before=true in selector configuration, which expression uses the reference to previous message parameters.
Another form of field specification is by preceding them with a dollar sign ($). In that case, if the field is not present in the object, a null value will be put instead and the expression can be evaluated further. For example, "abs($speed)>=0" will be always valid and return a boolean true.
You can use mathematical operators in expression and add brackets to control the order of operations. The priority of operations is the same as in the C programming language and notation, in general, is also very similar to what the C language defines. You can use the following operators:
operator  explanation 
+  sum 
  diff 
/  divide 
*  multiply 
%  reminder 
  binary OR 
&  binary AND 
&&  logical AND 
  logical OR 
^  binary XOR 
==  equal; for strings will match with wildcards 
!=  notequal; the inverted value of equal operator 
~  case insensitive match of a string with wildcards 
<  less 
<=  less or equal 
>  greater 
>=  greater or equal 
>>  right shift by the specified amount of bits (in the range from 0 to 64) 
<<  left shift by the specified amount of bits (in the range from 0 to 64) 
If you need more operators, please contact us.
Here are some sample expressions and their resulting values:
expression  evaluation 
1 + 2 * 4 / 2  5 
(1 + 2) * 4 / 2  6 
(4 & 1) > 0  false, because 4 AND(binary) 1 is 0 
(4 && 1) > 0  true, because 4 AND(logical) 1 is 1 
It is also possible to use some functions inside expressions, for example, "abs(speedprevious('speed'))>10" will always be true if the instant speed value change between two sequential messages is different by more than 10 km/h in any direction. Here is the full list of functions:
function  explanation 
previous(X)  previous known value of X or null if not yet available 
abs(X)  an absolute value of X 
round(X)  round X to the closest integer 
ceil(X)  round X up to the closest integer 
floor(X)  round X down to the closest integer 
sqrt(X)  the square root of X 
min(X, Y)  return a minimum value between X and Y 
max(X, Y)  return a maximum value between X and Y 
if(X, Y, Z)  if X is nonzero, evaluate to Y otherwise evaluate to Z 
mileage()  mileage in kilometers from the previous message, calculated using position.latitude, position.longitude and position.altitude parameters 
distance(lat1, lon1, lat2, lon2)  distance in kilometers between two points X and Y with coordinates lat1, lon1 and lat2, lon2 where latX is latitude of point X and lonX is longitude of point X 
month(X)  month value for X with ranges 112, where X is UNIX timestamp, e.g. month(timestamp) will return month number for the current message 
weekday(X)  Return day of week value for X with ranges 17, where X is UNIX timestamp. 
day(X)  day of month value for X with ranges 131, where X is UNIX timestamp 
hour(X)  hour for X with ranges 023, where X is UNIX timestamp 
minute(X)  minute value for X with ranges 059, where X is UNIX timestamp 
strftime(X, Y)  format date into text, where X is UNIX timestamp and Y is a format string similar to strftime. 
now()  current time as UNIX timestamp with microseconds granularity 
json(X, Y) or short form: X[Y]  fetch the value from the subelement inside a JSON object or array. Sample JSON: {"x":{"y":1, "z":2}, "a":[3, 5, {"keyA":"valueA"}]} To fetch value under key "y" inside object under key "x" use json(x, 'y') or x['y'] which will return 1. To access arrays use json(a, 0), json(a, '/0') or a[0] which all will return 3. JSON path can be used for as well for fetching values from complex objects/arrays: json(a, '/2/keyA') or a['/2/keyA'] will return "valueA". If unable to extract value under path will return null. 
json_array_count(X)  return number of elements in JSON array X 
hex(X[, Y, Z])  Convert string value X from hexadecimal format into 64bit unsigned integer and take Z bits starting from Y. If Z is not defined, all bits will be taken, if Y is not defined, zero is assumed (first bit). Samples:

strescape(X)  Escape all control and printable characters in string X with backslash. 
urlencode(X)  URLencode string X. Function is useful for generating correct arguments with complex contents in webhook calls. 
exists('X')  return 'true' if the specified parameter with name 'X' exists in the message and 'false' otherwise 
variable('X')  return latest calculated value of variable X or null otherwise. In flespi analytics variables are created using counter of type variable. This function can only be used in calculator counters. 
metadata('X')  return device metadata under key or JSON path 'X'. This function is available in analytics(calculator, selector and counter expressions) and within plugins. 
error()  generate an error and fail expression evaluation. Can be used in conditional evaluation like: if(position.valid, mileage(), error()) 
not(X)  invert argument value  if zero/false, return nonzero/true and vice versa 
tonumber(X), tostring(X), toboolean(X)  convert value to another type 
isnumber(X), isstring(X), isboolean(X), isnull(X), isjson(X)  test if value is of the specified type and return 'true' in such case 
If you need more functions, please contact us.