openexpressions


Nameopenexpressions JSON
Version 1.4 PyPI version JSON
download
home_pagehttps://github.com/AaryamanBhute/OpenExpressions
SummaryA easy to use and expandable expression parser
upload_time2023-06-29 08:22:06
maintainer
docs_urlNone
authorAaryaman Bhute
requires_python
licenseMIT
keywords expression parser expandable customizable
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # OpenExpressions
## _The Most Simple And Expandable Python Expression Library_

OpenExpressions is a lightweight, open-source, and easily integrated expression library.
## Features

- Parse and Evaluate Expressions With Built-In and Self-Implemented Functionality 
- Inspect Abstract Syntax Tree generated by the Parser
- Quickly Create New Operators and Operands by Inheriting from Predefined Abstract Classes
- Default Modes Available for Math or Boolean Expressions
- Empty Mode Available for Clean Slate Start for New Expression Parser

OpenExpressions generates a context-free grammar from the given operators and operands and then
creates an LR(1) Automata and Parse Table to quickly generate and Abstract Syntax Tree which represents
the expression provided. The expression can then be evaluated from the leaf nodes up and the generated
structure of the nodes determines the order of operations.

> The goal of this project is to provide a better alternative
> to the native eval function that python provides for evaluating
> expressions. Although it can provide the functionality, the
> security risks can quickly become an issue if not used carefully.
> Additionally, using custom operations in eval must be done with
> function calls which can quickly become verbose in longer expressions
> OpenExpressions' options to create new Unary and Binary operators
> allows for simpler expression syntax and more readable expressions.
> It also resolves the security risks associated with eval as all
> the expression is not executed as code but instead interpreted as
> the available operators and operands.

## Installation

OpenExpressions is available on PyPi and easily installable on Python 3.3+ using pip

```sh
pip install openexpressions
```

Source code is available at this [Github Repository](https://github.com/AaryamanBhute/OpenExpressions)

Module info is available at [PyPi Page](https://pypi.org/project/openexpressions/)

## Documentation

### Basic Usage
The basic usage of this library consists of using the preset of the Parser.
Currently, the 2 presets are: __math__ (_default_) and __boolean__
```python
from openexpressions.Parser import Parser

if __name__ == "__main__":
    # create the parser
    math_parser = Parser() # or explicitly with math_parser = Parser(mode="math")
    # using the parser, store the parsed expression
    expression = math_parser.parse("1 + (2 ** a ** b - 4 / -c)")
    # expression is now a Tree of ExpressionNodes, each one is an Operator or Operand
    
    # evaluate the expression by passing in a context we would like for variables
    value_0 = expression.eval({'a': 1, 'b': 3, 'c': -5})
    # we can reuse the same expression to evaluate with a different context
    value_1 = expression.eval({'a': 0.1, 'b': 9, 'c': 18.2})
    
    # you can also reuse the same parser for multiple expression strings
    other_expression = math_parser.parse("i * like * math")
    value_2 = other_expression.eval({'i': 5, 'like': 32, 'math': 64})
```

Here's an equivalent usage for the __boolean__ mode

```python
from openexpressions.Parser import Parser

if __name__ == "__main__":
    # create the parser
    bool_parser = Parser(mode="boolean")
    # using the parser, store the parsed expression
    expression = bool_parser.parse("1 & a | 0 | (b ^ c)")
    # expression is now a Tree of ExpressionNodes, each one is an Operator or Operand
    
    # evaluate the expression by passing in a context we would like for variables
    value_0 = expression.eval({'a': 1, 'b': 0, 'c': 1})
    # we can reuse the same expression to evaluate with a different context
    value_1 = expression.eval({'a': 0, 'b': 1, 'c': 0})
    
    # you can also reuse the same parser for multiple expression strings
    other_expression = bool_parser.parse("i | like | boolean")
    value_2 = other_expression.eval({'i': 1, 'like': 1, 'boolean': 1})
```

With the preset modes, you can use the following default Operators and Operands

## Built-Ins
Source code for built-in operators and operands found [here](https://github.com/AaryamanBhute/OpenExpressions/blob/main/openexpressions/ExpressionNodes.py)

This provides an overview of each of the default values, to better understand the significance
of each field, taking a look at the [Expression Node Declarations](#Expression-Node-Declarations)
#### Math Mode Built-Ins

##### Operands
- Int - _An Integer Literal_ - __identifier regex:__ \d+(?!\\.)
- Float - _An Floating Point Literal_ - __identifier regex:__ \d*\\.\d+
- Var - _A Variable Usage_ - __identifier regex:__ [a-zA-Z]\w*
##### Unary Operators (_UnOp_)
- Neg - _Arithmetic Negation_ -  __identifier regex:__ -
__order value__: 70000
##### Binary Operators (_BinOp_)
- Add - _Arithmetic Addition_ -  __identifier regex:__ \\+
__order value__: 100000
- Sub - _Arithmetic Subtraction_ -  __identifier regex:__ -
__order value__: 100000
- Mult - _Arithmetic Multiplication_ -  __identifier regex:__ \\*
__order value__: 90000
- Div - _Arithmetic Division -  __identifier regex:__ /
__order value__: 90000
- IntDiv - _Integer Division_ -  __identifier regex:__ //
__order value__: 90000
- Mod - _Arithmetic Modulo_ -  __identifier regex:__ %
__order value__: 90000
- Pow - _Arithmetic Exponentiation_ -  __identifier regex:__ \\*\\\*
__order value__: 80000
##### Poly Operators (_PolyOp_)
- Sum - _Summation Operator_ -  __identifier regex:__ SUM
> - __number of sub-expressions__: 4
> - _first sub-expression_: __counter variable__
> - _second sub-expression_: __lower bound count__
> - _third sub-expression_: __upper bound count__
> - _fourth sub-expression_: __evaluated expression__
- Prod - _Product Notation Operator_ -  __identifier regex:__ PROD
> - __number of sub-expressions__: 4
> - _first sub-expression_: __counter variable__
> - _second sub-expression_: __lower bound count__
> - _third sub-expression_: __upper bound count__
> - _fourth sub-expression_: __evaluated expression__
##### Wrap Operators (_WrapOp_)
- Paren - _Parenthetical Expression_ - __left identifier regex:__ \\( - __right identifier regex:__ \\)
- Abs - _Absolute Value Expression_ - __left identifier regex:__ \\| - __right identifier regex:__ \\|
#### Boolean Mode Built-Ins

##### Operands
- BoolVal - _A Boolean Literal_ - __identifier regex:__ (0 | 1)
- BoolVar - _A Boolean Variable_ - __identifier regex:__ [a-zA-Z]\w*
##### Unary Operators (_UnOp_)
- Not - _Arithmetic Negation_ -  __identifier regex:__ ~
__order value__: 70000
##### Binary Operators (_BinOp_)
- BitAnd - _Logical And_ -  __identifier regex:__ &
__order value__: 90000
- BitOr - _Logical Or_ -  __identifier regex:__ |
__order value__: 100000
- BitXOr - _Logical XOr_ -  __identifier regex:__ \\^
__order value__: 80000
##### Wrap Operators (_WrapOp_)
- Paren - _Parenthetical Expression_ - __left identifier regex:__ \\( - __right identifier regex:__ \\)
### Advanced Usage
Advanced usage of this library consists of creating custom operators and operands, here we'll go over templates and an example usage of this feature. The current options are to create new __Operands__, __UnOps__, __BinOps__, __PolyOps__, and __WrapOps__ by inheriting from their respective abstract classes.
#### Custom Operand
A custom operand can be made by using the following template
```python
from openexpressions.ExpressionNodes import Operand

class NEW_OPERAND(Operand):
    #REQUIRED - must be a python re library compatible regex
    identifier = r"0"
    #OPTIONAL - used by some operators to determine behavior.
    negation_inclusive = False # used by Pow to determine if result should be negative
    
    def __init__(self, image): # will be passed the raw string image of the operand match
        super().__init__(image) # super will store passed value in self.val
    #REQUIRED - must accept a context object
    def eval(context=None):
        #RETURN WHAT THIS OPERAND SHOULD EVALUATE TO
        return(self.val)
```

The parser will scan for a regex match of the given identifier and the matched string will be passed to the constructor of this class as "image". Since operands are the leaves of the expression tree, there will be no children of this ExpressionNode.

#### Custom Unary Operator
A custom unary operator can be made by using the following template
```python
from openexpressions.ExpressionNodes import UnOp

class NEW_UNOP(UnOp):
    #REQUIRED - must be a python re library compatible regex
        #Ensure that operators match only 1 non-zero length substring to prevent issues
        #In other words, only use characters and not any other regex functionality
    identifier = r"UNOP"
    #OPTIONAL - used by some operators to determine behavior.
    negation_inclusive = False # used by Pow to determine if result should be negative
    
    def __init__(self, expression) -> None:
        super().__init__(expression) # super will store passed expression in self.expr
    def eval(self, context=None):
        #PERFORM OPERATION ON self.expr EVALUATION AND RETURN NEW VALUE
        return(do_something(self.expr.eval(context)))
```

The parser will scan for a regex match of the given identifier and the appropriate following expression will be passed into the constructor as "expression."

#### Custom Binary Operator
A custom binary operator can be made by using the following template

```python
from openexpressions.ExpressionNodes import BinOp

class NEW_BINOP(BinOp):
    #REQUIRED - must be a python re library compatible regex
        #Ensure that operators match only 1 non-zero length substring to prevent issues
        #In other words, only use characters and not any other regex functionality
    identifier = r"BINOP"
    #OPTIONAL - used by some operators to determine behavior.
    negation_inclusive = False # used by Pow to determine if result should be negative
    
    def __init__(self, left_expression, right_expression) -> None:
        # store expressions in self.left and self.right
        super().__init__(left_expression, right_expression) 
    def eval(self, context=None):
        #PERFORM OPERATION ON self.left and self.right, EVALUATE AND RETURN NEW VALUE
        return(do_something(self.left.eval(context), self.right.eval(context)))
```
The parser will scan for a regex match of the given identifier and the appropriate preceding and following expressions will be passed into the constructor as "left_expression" and "right_expression" respectively.

#### Custom Poly Operator
A custom poly operator can be made by using the following template
```python
from openexpressions.ExpressionNodes import PolyOp

class NEW_POLYOP(PolyOp):
    #REQUIRED - must be a python re library compatible regex
        #Ensure that operators match only 1 non-zero length substring to prevent issues
        #In other words, only use characters and not any other regex functionality
    identifier = r"POLYOP"
    #REQUIRED - must be an integer which represents the number of required fields
    num_fields = 3
    #OPTIONAL - used by some operators to determine behavior.
    negation_inclusive = False # used by Pow to determine if result should be negative
    
    def __init__(self, op_1, op_2, op_3) -> None:
         # super will store passed parameters in self.ops with the leftmost being index 0
         # and each subsequence paramter being at the index after the previous one
        super().__init__(op_1, op_2, op_3)
        # op_1 stored in self.op[0]
        # op_2 stored in self.op[1]
        # op_3 stored in self.op[2]
    def eval(self, context=None):
        #PERFORM OPERATION ON all OPERANDS EVALUATION AND RETURN NEW VALUE
        return(do_something(self.op[0].eval(context), self.op[1].eval(context), self.op[2].eval(context)))
```

The parser will scan for a regex match of the given identifier and followed by a parenthetical expression with "num_field" sub-expressions which are delimited by commas.

#### Custom Wrap Operator
A custom wrap operator can be made by using the following template
```python
from openexpressions.ExpressionNodes import WrapOp

class NEW_WRAPOP(WrapOp):
    #REQUIRED - must be a python re library compatible regex
        #Ensure that operators match only 1 non-zero length substring to prevent issues
        #In other words, only use characters and not any other regex functionality
    left_identifier = r"LEFT_WRAPOP_BOUND"
    right_identifier = r"RIGHT_WRAPOP_BOUND"
    #OPTIONAL - used by some operators to determine behavior.
    negation_inclusive = False # used by Pow to determine if result should be negative
    
    def __init__(self, expression) -> None:
        super().__init__(expression) # super will store passed expression in self.expr
    def eval(self, context=None):
        #PERFORM OPERATION ON self.expr EVALUATION AND RETURN NEW VALUE
        return(do_something(self.expr.eval(context)))
```

The parser will scan for a regex match of the given left_identifier followed by the right_identifier and the expression between them will be passed into the constructor as "expression."

#### Create a Parser With Custom Operands and Operators

After creating your operands, you need to create a parser that uses them by passing them
in the constructor of the parser. This can be done as shown in the following example.

__Note__: The parser creates an expression tree that evaluates the given expression
from lowest to highest _order value_ operators.

```python
from openexpressions.Parser import Parser
from openexpressions.ExpressionNodes import UnOp, BinOp, PolyOp, WrapOp

# CREATE DECLARATIONS OF CUSTOM OPERANDS AND OPERATORS HERE

if __name__ == "__main__":
    # when a mode is selected, custom operators and operands are appended 
    # on top of the provided built-in ones for that mode. If you want a empty
    # start, use mode = "empty".
    
    # UnOps must be passed in as a tuple of 2 elements
        # index 0: Class
        # index 1: order value
        
    # BinOps must be passed in as a tuple of 3 elements
        # index 0: Class
        # index 1: order value
        # index 2: reversed boolean (order of evaluation for equal priority operations)
            # when reversed = False -> left to right
            # when reverse = True -> right to left
            
    # PolyOps and WrapOps can simply be passed in as a Class
    
    # Operands can simply be passed in as a Class
    
    parser = Parser(mode="math"
            custom_operators=[
                    (UnOp1, 100), (UnOp2, 99)...,
                    (BinOp1, 50, True), (BinOp2, 49, False)...
                    PolyOp1, PolyOp2..., WrapOp1, WrapOp2...
                ]
            custom_operands=[
                    Operand1, Operand2...
                ]
            )
```

#### Warnings About Custom Operators and Operands
Since this library relies on LR(1) parsing, there are restrictions on the types of context-free
grammars that can be parsed using an LR(1) parser. To ensure that the parser works as expected,
make sure to follow the following rules.
> - Priority Values __Must__ be Non-Negative
> - Each Priority Value __Must__ contain __exclusively__ UnOps __OR__ BinOps
> - Each Priority Value __Must__ contain __exclusively__ Reversed __OR__ Non-Reversed BinOps
> - Each Priority Value __Must__ contain __exclusively__ Operators with unique identifiers


#### Full-Scale Example
To get a better idea of the capabilities of the customization for this library, lets take a look 
at an example setup for 3 dimensional vector expressions.

```python
from openexpressions.Parser import Parser
from openexpressions.ExpressionNodes import UnOp,BinOp,PolyOp,WrapOp,Operand,Var,Paren
import math
import re

class Vector(Operand):
    identifier = r"<-?(\d+.\d+|\d+), -?(\d+.\d+|\d+), -?(\d+.\d+|\d+)>"
    def __init__(self, image) -> None:
        super().__init__(tuple(float(i) for i in re.findall(r"-?(\d+.\d+|\d+)", image)))
    def eval(self, context=None):
        return(self.val)

class Negate(UnOp):
    identifier = r"-"
    def __init__(self, expression) -> None:
        super().__init__(expression)
    def eval(self, context=None):
        return(tuple(-i for i in self.expr.eval(context)))

class CrossProduct(BinOp):
    identifier = r"X"
    def __init__(self, left_expression, right_expression) -> None:
        super().__init__(left_expression, right_expression)
    def eval(self, context=None):
        lv = self.left.eval(context)
        rv = self.right.eval(context)
        return((lv[1] * rv[2] - lv[2] * rv[1], lv[2] * rv[0] - lv[0] * rv[2], lv[0] * rv[1] - lv[1] * rv[0]))

class DotProduct(BinOp):
    identifier = r"\."
    def __init__(self, left_expression, right_expression) -> None:
        super().__init__(left_expression, right_expression)
    def eval(self, context=None):
        lv = self.left.eval(context)
        rv = self.right.eval(context)
        s = 0
        for i in range(3):
            s += lv[i] * rv[i]
        return(s)

class Magnitude(WrapOp):
    left_identifier = r"\|"
    right_identifier = r"\|"
    def __init__(self, expression) -> None:
        super().__init__(expression)
    def eval(self, context=None):
        v = self.expr.eval(context)
        s = 0
        for i in range(3):
            s += abs(v[i]) ** 2
        return(math.sqrt(s))

class Select(PolyOp):
    identifier = r"SELECT"
    num_fields = 3

    def __init__(self, one, two, three) -> None:
        super().__init__(one, two, three)
    def eval(self, context=None):
        one = self.ops[0].eval(context)
        two = self.ops[1].eval(context)
        three = self.ops[2].eval(context)
        return((one[0], two[1], three[2]))


if __name__ == "__main__":
    vector_parser = Parser(mode="empty",
                           custom_operands=(Vector, Var),
                           custom_operators=((Negate, 10), (CrossProduct, 20, False), (DotProduct, 20, False), Magnitude, Select, Paren))
    print(vector_parser.parse("-(SELECT(<0, -991, -992>, <-990, 1, -992>, <-990, -991, 2>) X <3, 4, 5>) . a").eval({'a': (-6, 7, -8)}))
```

Using this example, we can see the power of this library. It's capabilities extend beyond just
numbers and it's generic nature can quickly be used to create entire new expression systems to
help simplify representing them in your code.

__Note__: Notice that Var and Paren we imported from _openexpressions.ExpressionNodes_ and
then passed as an extra operand and operator respectively. Since we were using the "empty"
mode, we can reuse these operators as they fulfilled the purpose we wanted.

#### License
This library operates under the __MIT License__

#### Contact
If discover some issue within this library or would like to collaborate on it's development
or some other project, please reach out to me at __aryamanbhute@gmail.com__. I am always
happy to chat!

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/AaryamanBhute/OpenExpressions",
    "name": "openexpressions",
    "maintainer": "",
    "docs_url": null,
    "requires_python": "",
    "maintainer_email": "",
    "keywords": "EXPRESSION,PARSER,EXPANDABLE,CUSTOMIZABLE",
    "author": "Aaryaman Bhute",
    "author_email": "aryamanbhute@gmail.com",
    "download_url": "https://files.pythonhosted.org/packages/33/32/d94b1a159cb8463c340612b760d830c3bde59179b350f31e8f5a52c8a018/openexpressions-1.4.tar.gz",
    "platform": null,
    "description": "# OpenExpressions\r\n## _The Most Simple And Expandable Python Expression Library_\r\n\r\nOpenExpressions is a lightweight, open-source, and easily integrated expression library.\r\n## Features\r\n\r\n- Parse and Evaluate Expressions With Built-In and Self-Implemented Functionality \r\n- Inspect Abstract Syntax Tree generated by the Parser\r\n- Quickly Create New Operators and Operands by Inheriting from Predefined Abstract Classes\r\n- Default Modes Available for Math or Boolean Expressions\r\n- Empty Mode Available for Clean Slate Start for New Expression Parser\r\n\r\nOpenExpressions generates a context-free grammar from the given operators and operands and then\r\ncreates an LR(1) Automata and Parse Table to quickly generate and Abstract Syntax Tree which represents\r\nthe expression provided. The expression can then be evaluated from the leaf nodes up and the generated\r\nstructure of the nodes determines the order of operations.\r\n\r\n> The goal of this project is to provide a better alternative\r\n> to the native eval function that python provides for evaluating\r\n> expressions. Although it can provide the functionality, the\r\n> security risks can quickly become an issue if not used carefully.\r\n> Additionally, using custom operations in eval must be done with\r\n> function calls which can quickly become verbose in longer expressions\r\n> OpenExpressions' options to create new Unary and Binary operators\r\n> allows for simpler expression syntax and more readable expressions.\r\n> It also resolves the security risks associated with eval as all\r\n> the expression is not executed as code but instead interpreted as\r\n> the available operators and operands.\r\n\r\n## Installation\r\n\r\nOpenExpressions is available on PyPi and easily installable on Python 3.3+ using pip\r\n\r\n```sh\r\npip install openexpressions\r\n```\r\n\r\nSource code is available at this [Github Repository](https://github.com/AaryamanBhute/OpenExpressions)\r\n\r\nModule info is available at [PyPi Page](https://pypi.org/project/openexpressions/)\r\n\r\n## Documentation\r\n\r\n### Basic Usage\r\nThe basic usage of this library consists of using the preset of the Parser.\r\nCurrently, the 2 presets are: __math__ (_default_) and __boolean__\r\n```python\r\nfrom openexpressions.Parser import Parser\r\n\r\nif __name__ == \"__main__\":\r\n    # create the parser\r\n    math_parser = Parser() # or explicitly with math_parser = Parser(mode=\"math\")\r\n    # using the parser, store the parsed expression\r\n    expression = math_parser.parse(\"1 + (2 ** a ** b - 4 / -c)\")\r\n    # expression is now a Tree of ExpressionNodes, each one is an Operator or Operand\r\n    \r\n    # evaluate the expression by passing in a context we would like for variables\r\n    value_0 = expression.eval({'a': 1, 'b': 3, 'c': -5})\r\n    # we can reuse the same expression to evaluate with a different context\r\n    value_1 = expression.eval({'a': 0.1, 'b': 9, 'c': 18.2})\r\n    \r\n    # you can also reuse the same parser for multiple expression strings\r\n    other_expression = math_parser.parse(\"i * like * math\")\r\n    value_2 = other_expression.eval({'i': 5, 'like': 32, 'math': 64})\r\n```\r\n\r\nHere's an equivalent usage for the __boolean__ mode\r\n\r\n```python\r\nfrom openexpressions.Parser import Parser\r\n\r\nif __name__ == \"__main__\":\r\n    # create the parser\r\n    bool_parser = Parser(mode=\"boolean\")\r\n    # using the parser, store the parsed expression\r\n    expression = bool_parser.parse(\"1 & a | 0 | (b ^ c)\")\r\n    # expression is now a Tree of ExpressionNodes, each one is an Operator or Operand\r\n    \r\n    # evaluate the expression by passing in a context we would like for variables\r\n    value_0 = expression.eval({'a': 1, 'b': 0, 'c': 1})\r\n    # we can reuse the same expression to evaluate with a different context\r\n    value_1 = expression.eval({'a': 0, 'b': 1, 'c': 0})\r\n    \r\n    # you can also reuse the same parser for multiple expression strings\r\n    other_expression = bool_parser.parse(\"i | like | boolean\")\r\n    value_2 = other_expression.eval({'i': 1, 'like': 1, 'boolean': 1})\r\n```\r\n\r\nWith the preset modes, you can use the following default Operators and Operands\r\n\r\n## Built-Ins\r\nSource code for built-in operators and operands found [here](https://github.com/AaryamanBhute/OpenExpressions/blob/main/openexpressions/ExpressionNodes.py)\r\n\r\nThis provides an overview of each of the default values, to better understand the significance\r\nof each field, taking a look at the [Expression Node Declarations](#Expression-Node-Declarations)\r\n#### Math Mode Built-Ins\r\n\r\n##### Operands\r\n- Int - _An Integer Literal_ - __identifier regex:__ \\d+(?!\\\\.)\r\n- Float - _An Floating Point Literal_ - __identifier regex:__ \\d*\\\\.\\d+\r\n- Var - _A Variable Usage_ - __identifier regex:__ [a-zA-Z]\\w*\r\n##### Unary Operators (_UnOp_)\r\n- Neg - _Arithmetic Negation_ -  __identifier regex:__ -\r\n__order value__: 70000\r\n##### Binary Operators (_BinOp_)\r\n- Add - _Arithmetic Addition_ -  __identifier regex:__ \\\\+\r\n__order value__: 100000\r\n- Sub - _Arithmetic Subtraction_ -  __identifier regex:__ -\r\n__order value__: 100000\r\n- Mult - _Arithmetic Multiplication_ -  __identifier regex:__ \\\\*\r\n__order value__: 90000\r\n- Div - _Arithmetic Division -  __identifier regex:__ /\r\n__order value__: 90000\r\n- IntDiv - _Integer Division_ -  __identifier regex:__ //\r\n__order value__: 90000\r\n- Mod - _Arithmetic Modulo_ -  __identifier regex:__ %\r\n__order value__: 90000\r\n- Pow - _Arithmetic Exponentiation_ -  __identifier regex:__ \\\\*\\\\\\*\r\n__order value__: 80000\r\n##### Poly Operators (_PolyOp_)\r\n- Sum - _Summation Operator_ -  __identifier regex:__ SUM\r\n> - __number of sub-expressions__: 4\r\n> - _first sub-expression_: __counter variable__\r\n> - _second sub-expression_: __lower bound count__\r\n> - _third sub-expression_: __upper bound count__\r\n> - _fourth sub-expression_: __evaluated expression__\r\n- Prod - _Product Notation Operator_ -  __identifier regex:__ PROD\r\n> - __number of sub-expressions__: 4\r\n> - _first sub-expression_: __counter variable__\r\n> - _second sub-expression_: __lower bound count__\r\n> - _third sub-expression_: __upper bound count__\r\n> - _fourth sub-expression_: __evaluated expression__\r\n##### Wrap Operators (_WrapOp_)\r\n- Paren - _Parenthetical Expression_ - __left identifier regex:__ \\\\( - __right identifier regex:__ \\\\)\r\n- Abs - _Absolute Value Expression_ - __left identifier regex:__ \\\\| - __right identifier regex:__ \\\\|\r\n#### Boolean Mode Built-Ins\r\n\r\n##### Operands\r\n- BoolVal - _A Boolean Literal_ - __identifier regex:__ (0 | 1)\r\n- BoolVar - _A Boolean Variable_ - __identifier regex:__ [a-zA-Z]\\w*\r\n##### Unary Operators (_UnOp_)\r\n- Not - _Arithmetic Negation_ -  __identifier regex:__ ~\r\n__order value__: 70000\r\n##### Binary Operators (_BinOp_)\r\n- BitAnd - _Logical And_ -  __identifier regex:__ &\r\n__order value__: 90000\r\n- BitOr - _Logical Or_ -  __identifier regex:__ |\r\n__order value__: 100000\r\n- BitXOr - _Logical XOr_ -  __identifier regex:__ \\\\^\r\n__order value__: 80000\r\n##### Wrap Operators (_WrapOp_)\r\n- Paren - _Parenthetical Expression_ - __left identifier regex:__ \\\\( - __right identifier regex:__ \\\\)\r\n### Advanced Usage\r\nAdvanced usage of this library consists of creating custom operators and operands, here we'll go over templates and an example usage of this feature. The current options are to create new __Operands__, __UnOps__, __BinOps__, __PolyOps__, and __WrapOps__ by inheriting from their respective abstract classes.\r\n#### Custom Operand\r\nA custom operand can be made by using the following template\r\n```python\r\nfrom openexpressions.ExpressionNodes import Operand\r\n\r\nclass NEW_OPERAND(Operand):\r\n    #REQUIRED - must be a python re library compatible regex\r\n    identifier = r\"0\"\r\n    #OPTIONAL - used by some operators to determine behavior.\r\n    negation_inclusive = False # used by Pow to determine if result should be negative\r\n    \r\n    def __init__(self, image): # will be passed the raw string image of the operand match\r\n        super().__init__(image) # super will store passed value in self.val\r\n    #REQUIRED - must accept a context object\r\n    def eval(context=None):\r\n        #RETURN WHAT THIS OPERAND SHOULD EVALUATE TO\r\n        return(self.val)\r\n```\r\n\r\nThe parser will scan for a regex match of the given identifier and the matched string will be passed to the constructor of this class as \"image\". Since operands are the leaves of the expression tree, there will be no children of this ExpressionNode.\r\n\r\n#### Custom Unary Operator\r\nA custom unary operator can be made by using the following template\r\n```python\r\nfrom openexpressions.ExpressionNodes import UnOp\r\n\r\nclass NEW_UNOP(UnOp):\r\n    #REQUIRED - must be a python re library compatible regex\r\n        #Ensure that operators match only 1 non-zero length substring to prevent issues\r\n        #In other words, only use characters and not any other regex functionality\r\n    identifier = r\"UNOP\"\r\n    #OPTIONAL - used by some operators to determine behavior.\r\n    negation_inclusive = False # used by Pow to determine if result should be negative\r\n    \r\n    def __init__(self, expression) -> None:\r\n        super().__init__(expression) # super will store passed expression in self.expr\r\n    def eval(self, context=None):\r\n        #PERFORM OPERATION ON self.expr EVALUATION AND RETURN NEW VALUE\r\n        return(do_something(self.expr.eval(context)))\r\n```\r\n\r\nThe parser will scan for a regex match of the given identifier and the appropriate following expression will be passed into the constructor as \"expression.\"\r\n\r\n#### Custom Binary Operator\r\nA custom binary operator can be made by using the following template\r\n\r\n```python\r\nfrom openexpressions.ExpressionNodes import BinOp\r\n\r\nclass NEW_BINOP(BinOp):\r\n    #REQUIRED - must be a python re library compatible regex\r\n        #Ensure that operators match only 1 non-zero length substring to prevent issues\r\n        #In other words, only use characters and not any other regex functionality\r\n    identifier = r\"BINOP\"\r\n    #OPTIONAL - used by some operators to determine behavior.\r\n    negation_inclusive = False # used by Pow to determine if result should be negative\r\n    \r\n    def __init__(self, left_expression, right_expression) -> None:\r\n        # store expressions in self.left and self.right\r\n        super().__init__(left_expression, right_expression) \r\n    def eval(self, context=None):\r\n        #PERFORM OPERATION ON self.left and self.right, EVALUATE AND RETURN NEW VALUE\r\n        return(do_something(self.left.eval(context), self.right.eval(context)))\r\n```\r\nThe parser will scan for a regex match of the given identifier and the appropriate preceding and following expressions will be passed into the constructor as \"left_expression\" and \"right_expression\" respectively.\r\n\r\n#### Custom Poly Operator\r\nA custom poly operator can be made by using the following template\r\n```python\r\nfrom openexpressions.ExpressionNodes import PolyOp\r\n\r\nclass NEW_POLYOP(PolyOp):\r\n    #REQUIRED - must be a python re library compatible regex\r\n        #Ensure that operators match only 1 non-zero length substring to prevent issues\r\n        #In other words, only use characters and not any other regex functionality\r\n    identifier = r\"POLYOP\"\r\n    #REQUIRED - must be an integer which represents the number of required fields\r\n    num_fields = 3\r\n    #OPTIONAL - used by some operators to determine behavior.\r\n    negation_inclusive = False # used by Pow to determine if result should be negative\r\n    \r\n    def __init__(self, op_1, op_2, op_3) -> None:\r\n         # super will store passed parameters in self.ops with the leftmost being index 0\r\n         # and each subsequence paramter being at the index after the previous one\r\n        super().__init__(op_1, op_2, op_3)\r\n        # op_1 stored in self.op[0]\r\n        # op_2 stored in self.op[1]\r\n        # op_3 stored in self.op[2]\r\n    def eval(self, context=None):\r\n        #PERFORM OPERATION ON all OPERANDS EVALUATION AND RETURN NEW VALUE\r\n        return(do_something(self.op[0].eval(context), self.op[1].eval(context), self.op[2].eval(context)))\r\n```\r\n\r\nThe parser will scan for a regex match of the given identifier and followed by a parenthetical expression with \"num_field\" sub-expressions which are delimited by commas.\r\n\r\n#### Custom Wrap Operator\r\nA custom wrap operator can be made by using the following template\r\n```python\r\nfrom openexpressions.ExpressionNodes import WrapOp\r\n\r\nclass NEW_WRAPOP(WrapOp):\r\n    #REQUIRED - must be a python re library compatible regex\r\n        #Ensure that operators match only 1 non-zero length substring to prevent issues\r\n        #In other words, only use characters and not any other regex functionality\r\n    left_identifier = r\"LEFT_WRAPOP_BOUND\"\r\n    right_identifier = r\"RIGHT_WRAPOP_BOUND\"\r\n    #OPTIONAL - used by some operators to determine behavior.\r\n    negation_inclusive = False # used by Pow to determine if result should be negative\r\n    \r\n    def __init__(self, expression) -> None:\r\n        super().__init__(expression) # super will store passed expression in self.expr\r\n    def eval(self, context=None):\r\n        #PERFORM OPERATION ON self.expr EVALUATION AND RETURN NEW VALUE\r\n        return(do_something(self.expr.eval(context)))\r\n```\r\n\r\nThe parser will scan for a regex match of the given left_identifier followed by the right_identifier and the expression between them will be passed into the constructor as \"expression.\"\r\n\r\n#### Create a Parser With Custom Operands and Operators\r\n\r\nAfter creating your operands, you need to create a parser that uses them by passing them\r\nin the constructor of the parser. This can be done as shown in the following example.\r\n\r\n__Note__: The parser creates an expression tree that evaluates the given expression\r\nfrom lowest to highest _order value_ operators.\r\n\r\n```python\r\nfrom openexpressions.Parser import Parser\r\nfrom openexpressions.ExpressionNodes import UnOp, BinOp, PolyOp, WrapOp\r\n\r\n# CREATE DECLARATIONS OF CUSTOM OPERANDS AND OPERATORS HERE\r\n\r\nif __name__ == \"__main__\":\r\n    # when a mode is selected, custom operators and operands are appended \r\n    # on top of the provided built-in ones for that mode. If you want a empty\r\n    # start, use mode = \"empty\".\r\n    \r\n    # UnOps must be passed in as a tuple of 2 elements\r\n        # index 0: Class\r\n        # index 1: order value\r\n        \r\n    # BinOps must be passed in as a tuple of 3 elements\r\n        # index 0: Class\r\n        # index 1: order value\r\n        # index 2: reversed boolean (order of evaluation for equal priority operations)\r\n            # when reversed = False -> left to right\r\n            # when reverse = True -> right to left\r\n            \r\n    # PolyOps and WrapOps can simply be passed in as a Class\r\n    \r\n    # Operands can simply be passed in as a Class\r\n    \r\n    parser = Parser(mode=\"math\"\r\n            custom_operators=[\r\n                    (UnOp1, 100), (UnOp2, 99)...,\r\n                    (BinOp1, 50, True), (BinOp2, 49, False)...\r\n                    PolyOp1, PolyOp2..., WrapOp1, WrapOp2...\r\n                ]\r\n            custom_operands=[\r\n                    Operand1, Operand2...\r\n                ]\r\n            )\r\n```\r\n\r\n#### Warnings About Custom Operators and Operands\r\nSince this library relies on LR(1) parsing, there are restrictions on the types of context-free\r\ngrammars that can be parsed using an LR(1) parser. To ensure that the parser works as expected,\r\nmake sure to follow the following rules.\r\n> - Priority Values __Must__ be Non-Negative\r\n> - Each Priority Value __Must__ contain __exclusively__ UnOps __OR__ BinOps\r\n> - Each Priority Value __Must__ contain __exclusively__ Reversed __OR__ Non-Reversed BinOps\r\n> - Each Priority Value __Must__ contain __exclusively__ Operators with unique identifiers\r\n\r\n\r\n#### Full-Scale Example\r\nTo get a better idea of the capabilities of the customization for this library, lets take a look \r\nat an example setup for 3 dimensional vector expressions.\r\n\r\n```python\r\nfrom openexpressions.Parser import Parser\r\nfrom openexpressions.ExpressionNodes import UnOp,BinOp,PolyOp,WrapOp,Operand,Var,Paren\r\nimport math\r\nimport re\r\n\r\nclass Vector(Operand):\r\n    identifier = r\"<-?(\\d+.\\d+|\\d+), -?(\\d+.\\d+|\\d+), -?(\\d+.\\d+|\\d+)>\"\r\n    def __init__(self, image) -> None:\r\n        super().__init__(tuple(float(i) for i in re.findall(r\"-?(\\d+.\\d+|\\d+)\", image)))\r\n    def eval(self, context=None):\r\n        return(self.val)\r\n\r\nclass Negate(UnOp):\r\n    identifier = r\"-\"\r\n    def __init__(self, expression) -> None:\r\n        super().__init__(expression)\r\n    def eval(self, context=None):\r\n        return(tuple(-i for i in self.expr.eval(context)))\r\n\r\nclass CrossProduct(BinOp):\r\n    identifier = r\"X\"\r\n    def __init__(self, left_expression, right_expression) -> None:\r\n        super().__init__(left_expression, right_expression)\r\n    def eval(self, context=None):\r\n        lv = self.left.eval(context)\r\n        rv = self.right.eval(context)\r\n        return((lv[1] * rv[2] - lv[2] * rv[1], lv[2] * rv[0] - lv[0] * rv[2], lv[0] * rv[1] - lv[1] * rv[0]))\r\n\r\nclass DotProduct(BinOp):\r\n    identifier = r\"\\.\"\r\n    def __init__(self, left_expression, right_expression) -> None:\r\n        super().__init__(left_expression, right_expression)\r\n    def eval(self, context=None):\r\n        lv = self.left.eval(context)\r\n        rv = self.right.eval(context)\r\n        s = 0\r\n        for i in range(3):\r\n            s += lv[i] * rv[i]\r\n        return(s)\r\n\r\nclass Magnitude(WrapOp):\r\n    left_identifier = r\"\\|\"\r\n    right_identifier = r\"\\|\"\r\n    def __init__(self, expression) -> None:\r\n        super().__init__(expression)\r\n    def eval(self, context=None):\r\n        v = self.expr.eval(context)\r\n        s = 0\r\n        for i in range(3):\r\n            s += abs(v[i]) ** 2\r\n        return(math.sqrt(s))\r\n\r\nclass Select(PolyOp):\r\n    identifier = r\"SELECT\"\r\n    num_fields = 3\r\n\r\n    def __init__(self, one, two, three) -> None:\r\n        super().__init__(one, two, three)\r\n    def eval(self, context=None):\r\n        one = self.ops[0].eval(context)\r\n        two = self.ops[1].eval(context)\r\n        three = self.ops[2].eval(context)\r\n        return((one[0], two[1], three[2]))\r\n\r\n\r\nif __name__ == \"__main__\":\r\n    vector_parser = Parser(mode=\"empty\",\r\n                           custom_operands=(Vector, Var),\r\n                           custom_operators=((Negate, 10), (CrossProduct, 20, False), (DotProduct, 20, False), Magnitude, Select, Paren))\r\n    print(vector_parser.parse(\"-(SELECT(<0, -991, -992>, <-990, 1, -992>, <-990, -991, 2>) X <3, 4, 5>) . a\").eval({'a': (-6, 7, -8)}))\r\n```\r\n\r\nUsing this example, we can see the power of this library. It's capabilities extend beyond just\r\nnumbers and it's generic nature can quickly be used to create entire new expression systems to\r\nhelp simplify representing them in your code.\r\n\r\n__Note__: Notice that Var and Paren we imported from _openexpressions.ExpressionNodes_ and\r\nthen passed as an extra operand and operator respectively. Since we were using the \"empty\"\r\nmode, we can reuse these operators as they fulfilled the purpose we wanted.\r\n\r\n#### License\r\nThis library operates under the __MIT License__\r\n\r\n#### Contact\r\nIf discover some issue within this library or would like to collaborate on it's development\r\nor some other project, please reach out to me at __aryamanbhute@gmail.com__. I am always\r\nhappy to chat!\r\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "A easy to use and expandable expression parser",
    "version": "1.4",
    "project_urls": {
        "Download": "https://github.com/AaryamanBhute/OpenExpressions/v_10.tar.gz",
        "Homepage": "https://github.com/AaryamanBhute/OpenExpressions"
    },
    "split_keywords": [
        "expression",
        "parser",
        "expandable",
        "customizable"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "3332d94b1a159cb8463c340612b760d830c3bde59179b350f31e8f5a52c8a018",
                "md5": "e2c3f0609f758199b0c17dad1ddbc73f",
                "sha256": "d4addd48312e01094186918479e3158551663fbf78d8edb4a5833423f2cd299b"
            },
            "downloads": -1,
            "filename": "openexpressions-1.4.tar.gz",
            "has_sig": false,
            "md5_digest": "e2c3f0609f758199b0c17dad1ddbc73f",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": null,
            "size": 19721,
            "upload_time": "2023-06-29T08:22:06",
            "upload_time_iso_8601": "2023-06-29T08:22:06.958682Z",
            "url": "https://files.pythonhosted.org/packages/33/32/d94b1a159cb8463c340612b760d830c3bde59179b350f31e8f5a52c8a018/openexpressions-1.4.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-06-29 08:22:06",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "AaryamanBhute",
    "github_project": "OpenExpressions",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": false,
    "lcname": "openexpressions"
}
        
Elapsed time: 0.18843s