postfixcalc 0.8.4

Last updated:

0 purchases

postfixcalc 0.8.4 Image
postfixcalc 0.8.4 Images
Add to Cart

Description:

postfixcalc 0.8.4

postfixcalc
Simple, stupid but easy and powerful infix expression evaluator using postfix notation.
User Guide
How to use?
from postfixcalc import Calc

expr = "(-1) ^ 2 + 3 - 4 * 8 - 2 ^ 3"
calc = Calc(expr)

print(calc.answer)
print(type(calc.answer))

-36
<class 'int'>

Explanation
expression: (-1) ^ 2 + 3 - 4 * 8 - 2 ^ 3

which with the math operator precedence is:
expression: ((-1) ^ 2) + 3 - (4 * 8) - (2 ^ 3)
= (1) + 3 - (32) - (8)
= 4 - 32 - 8
= -28 - 8
= -36

Initialization
calc = Calc(
'(2 ^ 32) ^ (2 ^ 15) + -1',
calc_timeout=1, # timeout for the calculation (in seconds)
str_repr_timeout=1.5 # timeout to generate the string representation (in seconds)
)

print(f"'(2 ^ 32) ^ (2 ^ 15) + -1's answer has '{len(calc.stranswer())}' digits")

'(2 ^ 32) ^ (2 ^ 15) + -1's answer has '315653' digits

print(f'answer is: {calc.stranswer(15, 15)}')

answer is: 674114012549907...068940335579135

Other Attributes
from rich.pretty import Pretty
from rich import print as rprint

rprint(calc.parsed)

<ast.BinOp object at 0x7fb65c3917b0>

rprint(calc.extracted)

[
(
[
(
[([2], <ast.Pow object at 0x7fb661991120>, [32])],
<ast.Pow object at 0x7fb661991120>,
[([2], <ast.Pow object at 0x7fb661991120>, [15])]
)
],
<ast.Add object at 0x7fb661990ee0>,
[(<ast.USub object at 0x7fb661991540>, [1])]
)
]

rprint(calc.flattened)

(
(
(([2], <ast.Pow object at 0x7fb661991120>, [32]),),
<ast.Pow object at 0x7fb661991120>,
(([2], <ast.Pow object at 0x7fb661991120>, [15]),)
),
<ast.Add object at 0x7fb661990ee0>,
(<ast.USub object at 0x7fb661991540>, [1])
)

rprint(calc.strparenthesized)

(((2 ^ 32)) ^ ((2 ^ 15))) + (-1)

rprint(calc.listparenthesized)

['(', '(', '(', 2, '^', 32, ')', ')', '^', '(', '(', 2, '^', 15, ')', ')', ')', '+', '(', '-1', ')']

rprint(calc.numerized)

['(', '(', '(', 2, '^', 32, ')', ')', '^', '(', '(', 2, '^', 15, ')', ')', ')', '+', '(', -1, ')']

rprint(calc.postfix)

[2, 32, '^', 2, 15, '^', '^', -1, '+']

rprint(f'{calc.stranswer(15, 15)}')

674114012549907...068940335579135

Calc Documentation
class Calc (expr: str, calc_timeout: int | float = 0.1, str_repr_timeout: int | float = 0.2)


expr: infix math expression


calc_timeout: the timeout of the math calculation, if a expression's calculation took longer than this time, it would be killed and a TimeoutError will be raised.


str_repr_timeout: Calculating a expression like: (2 ^ 32) ^ (2 ^ 15) takes about a seconds, but the result has 315653 digits; so printing (getting the string representation) takes some time, this timeout is controlled by this parameter; and a TimeoutError will be raised.


Properties of Calc
Important Note
All the properties of Calc type (except stranswer) are cached properties. It means that they are calculated once and stored in the object and are not calculated every time you need them (if other properties and attributes are remained unchanged)

Calc.parsed

Parse the object with ast.parse function and return the parsed expression.
Because ast.parse uses the grammar of Python any syntax error will be raised here!
# underlying function

from postfixcalc.parser import parse

expr = '-1 ^ 2'
parsed = parse(expr)

rprint(parsed)

<ast.UnaryOp object at 0x7fb6603566b0>


Calc.extracted

Return a list of extracted numbers and operators from the parsed object
# underlying function
from postfixcalc.parser import extract_nums_and_ops

extracted = extract_nums_and_ops(parsed)
rprint(extracted)

[(<ast.USub object at 0x7fb661991540>, [([1], <ast.Pow object at 0x7fb661991120>, [2])])]


Calc.flattened

Flatten the numbers and operators list, this will reduce the nested lists and tuples
# underlying function
from postfixcalc.parser import flatten_nodes

flattened = flatten_nodes(extracted)
rprint(flattened)

(<ast.USub object at 0x7fb661991540>, ([1], <ast.Pow object at 0x7fb661991120>, [2]))


Calc.strparenthesized

Generate a parenthesized version of the expression passed in initialization according to the math operator precedence
# underlying function
from postfixcalc.parser import restrexpression

rprint(restrexpression(flattened))

-(1 ^ 2)


Calc.listparenthesized

Return the digits and parenthesis and operators in a list that will be used to generate the postfix list
# underlying function
from postfixcalc.parser import relistexpression

listed = relistexpression(flattened)
rprint(listed)

['-', '(', 1, '^', 2, ')']


Calc.numerized

Numerize the string numbers returned by listparenthesized method. In some cased like: (-1) ^ 2) the listparenthesized looks like this: ['(', '-1', ')', '^', 2], so we have to make those strings numbers
# underlying function
from postfixcalc.parser import make_num

numerized = make_num(listed)
rprint(numerized)

['-', '(', 1, '^', 2, ')']


Calc.postfix

Return a list with the postfix notation of the expression
# underlying function
from postfixcalc.parser import infix_to_postfix

postfixed = infix_to_postfix(numerized)
rprint(postfixed)

[1, 2, '^', '-']


Calc.answer

Calculate the answer respecting the calc_timeout
IMPORTANT NOTE: DON'T CALL print ON THE RESULT OF THIS METHOD.
This is because for instance calculating (2 ^ 32) ^ (2 ^ 15) is done under calc_timeout BUT generating the string
representation WILL TAKE MUCH LONGER!!! (it has 315653 digits)
If you want to print the result, use stranswer method
method
@cached_property
def answer(self):
process = multiprocessing.Process(target=evaluate, args=(self.postfix,))
process.start()
process.join(timeout=self.calc_timeout)
if process.is_alive():
process.terminate()
raise TimeoutError(
f"Calculations of {self.strparenthesized!r} took longer than {self.calc_timeout} seconds",
) from None
return evaluate(self.postfix)

# underlying function
from postfixcalc.pyeval import evaluate

rprint(evaluate(postfixed))

-1


Calc.stranswer

Return the string representation of the Calc.answer respecting the str_repr_timeout
method
def stranswer(
self,
beginning: Optional[int] = None,
ending: Optional[int] = None,
) -> str:
"""Return the string representation of the answer with the respect to the `str_repr_timeout`

beginning: see the first n digits of the answer: '982734...'
ending: see the last n digits of the answer: '...982734'

if both are specified: '987234...873242'
"""
process = multiprocessing.Process(target=str, args=(self.answer,))
process.start()
process.join(self.str_repr_timeout)
if process.is_alive():
process.terminate()
raise TimeoutError(
f"Generating a string representation of {self.expr!r} took longer than {self.str_repr_timeout} seconds",
) from None
try:
answer = str(self.answer)
match (beginning, ending):
case (None, None):
return answer
case (x, None):
return f"{answer[:x]}..."
case (None, x):
return f"...{answer[-x:]}"
case (x, y):
return f"{answer[:x]}...{answer[-y:]}"
case _:
raise ValueError("Confusing beginning and ending")
except ValueError:
raise TimeoutError(
f"Generating a string representation of {self.expr!r} took longer than {self.str_repr_timeout} seconds",
) from None

How the answer is calculated
Calc type has cached properties which rely on each others returned values, it means that when we access the Calc.answer property this sequential method and function calls, happen:
print(calc.answer)

calc.answer needs the postfix: calc.postfix gets called
calc.postfix needs the numerized list of the digits and operators: calc.numerized gets called
calc.numerized needs the parenthesized list of the digits and operators: calc.listparenthsized gets called
calc.listprenthesized needs the flattened list of extracted nums and ops: calc.flattened gets called.
calc.flattened needs the extracted nums and ops: calc.extracted gets called.
calc.extracted needs the parsed expression of the input expression: calc.parsed gets called.

Call graph for getting the answer

License:

For personal and professional use. You cannot resell or redistribute these repositories in their original state.

Customer Reviews

There are no reviews.