This repository was archived by the owner on Jul 22, 2020. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathchapter3.py
229 lines (191 loc) · 4.13 KB
/
chapter3.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
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
#!/usr/bin/python3
from __future__ import division
line = ""
cursor = 0 # Index of current reading position
token = None # The last token matched, if any
def skip_whitespace():
global cursor
while cursor < len(line) and line[cursor].isspace():
cursor += 1
def match_keyword():
global cursor, token
skip_whitespace()
if cursor >= len(line) or not line[cursor].isalpha():
return False
mark = cursor
while cursor < len(line) and line[cursor].isalpha():
cursor += 1
token = line[mark:cursor]
return True
def match_number():
global cursor, token
skip_whitespace()
if cursor >= len(line) or not line[cursor].isdigit():
return False
mark = cursor
while cursor < len(line) and line[cursor].isdigit():
cursor += 1
if cursor < len(line) and line[cursor] == ".":
cursor += 1
while cursor < len(line) and line[cursor].isdigit():
cursor += 1
token = line[mark:cursor]
return True
def match_varname():
global cursor, token
skip_whitespace()
if cursor >= len(line) or not line[cursor].isalpha():
return False
mark = cursor
while cursor < len(line) and line[cursor].isalnum():
cursor += 1
token = line[mark:cursor]
return True
def match_string():
global cursor, token
skip_whitespace()
if cursor >= len(line) or line[cursor] != '"':
return False
mark = cursor
cursor += 1 # Skip the opening double quote.
while line[cursor] != '"':
cursor += 1
if cursor >= len(line):
raise IndexError("Unclosed string")
cursor += 1 # Skip the closing double quote.
# Save string value without the double quotes.
token = line[mark + 1:cursor - 1]
return True
def parse_statement():
if not match_keyword():
raise SyntaxError("Statement expected")
stmt = token.lower()
if stmt == "let":
parse_let()
elif stmt == "print":
parse_print()
else:
raise SyntaxError("Unknown statement")
variables = {}
def parse_let():
if not match_varname():
raise SyntaxError("Variable expected")
var_name = token.lower()
if not match("="):
raise SyntaxError("'=' expected")
variables[var_name] = parse_expression()
if not match_eol():
raise SyntaxError("End of line expected")
def match(text):
global cursor
skip_whitespace()
if line.startswith(text, cursor):
cursor += len(text)
return True
else:
return False
def match_eol():
skip_whitespace()
return cursor >= len(line)
def parse_print():
if match_eol():
print()
return
value = str(parse_value())
while match(","):
value += str(parse_value())
print(value)
if not match_eol():
raise SyntaxError("End of line expected")
def parse_value():
global token
if match_string():
return token
else:
return parse_expression()
def parse_expression():
t1 = parse_term()
while match_add_sub():
op = token
t2 = parse_term()
if op == "+":
t1 += t2
elif op == "-":
t1 -= t2
else:
raise SyntaxError(op)
return t1
def parse_term():
t1 = parse_factor()
while match_mul_div():
op = token
t2 = parse_factor()
if op == "*":
t1 *= t2
elif op == "/":
t1 /= t2
else:
raise SyntaxError(op)
return t1
def parse_factor():
global token
if match("-"):
signum = -1
else:
signum = 1
if match_number():
return float(token) * signum
elif match_varname():
token = token.lower()
if token in variables:
return variables[token] * signum
else:
raise NameError("Var not found")
elif match("("):
value = parse_expression()
if match(")"):
return value * signum
else:
raise SyntaxError("Missing ')'")
else:
raise SyntaxError("Expression expected")
def match_add_sub():
global token
if match("+"):
token = "+"
return True
elif match("-"):
token = "-"
return True
else:
return False
def match_mul_div():
global token
if match("*"):
token = "*"
return True
elif match("/"):
token = "/"
return True
else:
return False
#line = "let c = 37"
#cursor = 0
#parse_statement()
#line = 'print "f = ", 32 + c * (9 / 5)'
#cursor = 0
#parse_statement()
print("Basic v0.2 READY\n")
done = False
while not done:
line = input("> ")
if line.lower() == "bye":
done = True
else:
cursor = 0
try:
parse_statement()
except Exception as e:
print(str(e) +
" in column " +
str(cursor))