-
Notifications
You must be signed in to change notification settings - Fork 37
/
Copy pathcalc.py
138 lines (102 loc) · 3.51 KB
/
calc.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
################ Calculator: derived from lis.py for Python 3.10
## (c) Peter Norvig, 2010-18; See http://norvig.com/lispy.html
## adapted by Luciano Ramalho with type hints and pattern matching.
################ Imports and Types
import math
import operator as op
import random
from collections import ChainMap
from collections.abc import MutableMapping
from typing import Any, TypeAlias
Symbol: TypeAlias = str
Atom: TypeAlias = float | int | Symbol
Expression: TypeAlias = Atom | list
Environment: TypeAlias = MutableMapping[Symbol, object]
################ Global Environment
def standard_env() -> Environment:
"An environment with some Scheme standard procedures."
env: Environment = {}
env.update(vars(math)) # sin, cos, sqrt, pi, ...
env.update(
{
'+': op.add,
'-': op.sub,
'*': op.mul,
'/': op.truediv,
'>': op.gt,
'<': op.lt,
'>=': op.ge,
'<=': op.le,
'=': op.eq,
'random': random.random,
}
)
return env
################ Parsing: parse, tokenize, and read_from_tokens
def parse(program: str) -> Expression:
"Read a Scheme expression from a string."
return read_from_tokens(tokenize(program))
def tokenize(s: str) -> list[str]:
"Convert a string into a list of tokens."
return s.replace('(', ' ( ').replace(')', ' ) ').split()
def read_from_tokens(tokens: list[str]) -> Expression:
"Read an expression from a sequence of tokens."
if len(tokens) == 0:
raise SyntaxError('unexpected EOF while reading')
token = tokens.pop(0)
if '(' == token:
L = []
while tokens[0] != ')':
L.append(read_from_tokens(tokens))
tokens.pop(0) # discard ')'
return L
elif ')' == token:
raise SyntaxError('unexpected )')
else:
return parse_atom(token)
def parse_atom(token: str) -> Atom:
"Numbers become numbers; every other token is a symbol."
try:
return int(token)
except ValueError:
try:
return float(token)
except ValueError:
return Symbol(token)
################ eval
global_env: Environment = standard_env()
def evaluate(exp: Expression) -> Any:
"Evaluate an expression in an environment."
match exp:
case Symbol(var): # variable reference
return global_env[var]
case literal if not isinstance(exp, list): # constant literal
return literal
case ['define', Symbol(var), value_exp]: # (define var exp)
global_env[var] = evaluate(value_exp)
case [op, *args]: # (proc arg...)
proc = evaluate(op)
values = (evaluate(arg) for arg in args)
return proc(*values)
case _:
raise SyntaxError(exp)
################ High level API
def calc(source: str) -> Any:
return evaluate(parse(source))
################ Interaction: A REPL
def repl(prompt: str = 'calc> ') -> None:
"A prompt-read-evaluate-print loop."
while True:
val = evaluate(parse(input(prompt)))
if val is not None:
print(lispstr(val))
def lispstr(exp: object) -> str:
"Convert a Python object back into a Lisp-readable string."
if isinstance(exp, list):
return '(' + ' '.join(map(lispstr, exp)) + ')'
else:
return str(exp)
if __name__ == '__main__':
# activate line editing and history
import readline
repl()