This library provides Python implementation of Simple String Expression Evaluator a.k.a. s2e2. The Evaluator returns value of an input expression. Unlike commonly known mathematical expression evaluators this one treats all parts of the expression as a strings and its output value is also a string.
For example:
- the value of the expression
A + BisAB - the value of
REPLACE("The cat is black", cat, dog)isThe dog is black
This is how one can use Evaluator to get value of some expression:
>>> import s2e2 >>> >>> evaluator = s2e2.Evaluator() >>> >>> evaluator.add_standard_functions() >>> evaluator.add_standard_operators() >>> >>> expression = 'A + B' >>> result = evaluator.evaluate(expression)
Supported expressions consist of the following tokens: string literals, operators (unary and binary), functions, predefined constants, round brackets for function's arguments denoting, commas for function's arguments separation and double quotes for characters escaping.
The difference between a function and an operator is that a function is always followed by a pair of round brackets with a list of function's arguments (probably empty) in between, while an operator does not use brackets and, if it is a binary operator, sticks between its operands. Also operators can have different priorities a.k.a. precedence.
For example:
- this is a function of 2 arguments:
FUNC(Arg1, Arg2) - and this is a binary operator:
Arg1 OP Arg2
There is only one predefined constant -- NULL -- which corresponds to an None value in Python. It can be used to check if some sub-expression is evaluated into some result: IF(SUBEXPR(Arg1, Arg2) == NULL, NULL, Value)
s2e2 provides a small set of predefined functions. They are:
Function
IF(Condition, Value1, Value2)Returns
Value1ifConditionis true, andValue2otherwise.Conditionmust be a boolean value.Function
REPLACE(Source, Regex, Replacement)Returns copy of
Sourcewith all matches ofRegexreplaced byReplacement. All three arguments are strings,Regexcannot beNULLor an empty string, Replacement cannot be NULL.Function
NOW()Returns current UTC datetime. The result is of
datetime.datetimetype.Function
ADD_DAYS(Datetime, NumberOfDays)Adds days to the provided datetime.
Datetimemust be ofdatetime.datetimetype and not NULL.NumberOfDaysis a notNULLobject which can be converter into any integer. The result is ofdatetime.datetimetype.Function
FORMAT_DATE(Datetime, Format)Converts
Datetimeinto a string according toFormat.Datetimemust be ofdatetime.datetimetype and notNULL.Formatis a notNULLstring.
It is possible to create and use any custom function. Here is a simple example:
import s2e2
class CustomFunction(s2e2.functions.Function):
def __init__(self, some_set):
super().__init__('CONTAINS', 1)
self.__set = some_set
def _check_arguments(self):
return isinstance(self._arguments[0], str)
def _result(self):
return self._arguments[0] in self.__set
evaluator = s2e2.Evaluator()
evaluator.add_standard_functions()
evaluator.add_standard_operators()
some_set = set(['key1', 'key2'])
custom_function = CustomFunction(some_set)
evaluator.add_function(custom_function)
expression = 'IF(CONTAINS(key1), YES, NO)'
result = evaluator.evaluate(expression)
As it was mentioned before, every operator has a priority. Within s2e2 the range of priorities is from 1 to 999. A set of predefined operators is provided. They are:
Binary operator
+, priority500Concatenates two strings. Every operand can be either a
NULLor a string. The result is a string.Binary operator
==, priority300Compares any two objects, including
NULL. If both operands areNULLthe result isTrue. The type of the result is boolean.Binary operator
!=, priority300The same as
==, but checks objects for inequality.Binary operator
>, priority400Compares any two comparable objects. None of the operands can be
NULL. The result is a boolean.Binary operator
>=, priority400Compares any two comparable objects. Both operands must be not
NULLor both must beNULL. In the latter case the result isTrue.Binary operator
<, priority400Same as
>, but checks if first operand is less that the second one.Binary operator
<=, priority400Same as
>=, but checks if first operand is less or equal that the second one.Binary operator
&&, priority200Computes logical conjunction of two boolean values. Both arguments are boolean, not
NULLvalue. The result is a boolean.Binary operator
||, priority100Computes logical disjunction of two boolean values. Both arguments are boolean, not
NULLvalue. The result is a boolean.Unary operator
!, priority600Negates boolean value. Operand cannot be
NULL. The result is a boolean.
It is possible to create and use any custom operator. Here is a simple example:
import s2e2
class CustomOperator(s2e2.operators.Operator):
def __init__(self):
super().__init__('~', 600, 1)
def _check_arguments(self):
return isinstance(self._arguments[0], str)
def _result(self):
return self._arguments[0][::-1]
evaluator = s2e2.Evaluator()
evaluator.add_standard_functions()
evaluator.add_standard_operators()
some_set = set(['key1', 'key2'])
custom_operator = CustomOperator()
evaluator.add_operator(custom_operator)
expression = '~Foo'
result = evaluator.evaluate(expression)
To use this project one would need:
- Python >= 3.4
- setuptools >= 3.4
To develop and/or change:
This will build the library into a egg file:
./setup.py bdist_egg
The output egg file can be found in the created dist folder.
The easiest way to install the library is to use easy_install (which is part of setuputils3):
sudo easy_install ./s2e2-0.0.0-py3.7.egg
Use one simple command:
./setup.py test
This project is licensed under the MIT License.