-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathast.py
206 lines (161 loc) · 5.22 KB
/
ast.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
import sys
from exceptions import *
try:
raw_input
except:
raw_input = input
# Root Class
class ASTNode(object):
def eval(self, context):
raise NotImplementedError(self.__class__)
# Types
class Number(ASTNode):
def __init__(self, value):
self.value = value
def eval(self, context):
return self.value
class Boolean(ASTNode):
def __init__(self, value):
self.value = value
def eval(self, context):
return self.value
class String(ASTNode):
def __init__(self, value):
self.value = value
def eval(self, context):
return self.value
# Placeholders
class Identifier(ASTNode):
def __init__(self, value, name=None):
self.name = name
self.value = value
class IdentifierReference(ASTNode):
def __init__(self, name):
self.name = name
def eval(self, context):
return context[self.name]
class Block(ASTNode):
def __init__(self, body):
self.body = body
def eval(self, context):
for instruction in self.body[0]:
instruction.eval(context)
class Assignment(ASTNode):
def __init__(self, identifier, value):
self.identifier = identifier
self.value = value
def eval(self, context):
context[self.identifier.name] = self.value.eval(context)
# Number
class BinaryOperation(ASTNode):
def __init__(self, left, right):
self.left = left
self.right = right
class Add(BinaryOperation):
def eval(self, context):
return self.left.eval(context) + self.right.eval(context)
class Subtract(BinaryOperation):
def eval(self, context):
return self.left.eval(context) - self.right.eval(context)
class Multiply(BinaryOperation):
def eval(self, context):
return self.left.eval(context) * self.right.eval(context)
class Divide(BinaryOperation):
def eval(self, context):
return self.left.eval(context) / self.right.eval(context)
class Modulus(BinaryOperation):
def eval(self, context):
return self.left.eval(context) % self.right.eval(context)
# Boolean
class BooleanOperation(ASTNode):
def __init__(self, left, right):
self.left = left
self.right = right
class And(BooleanOperation):
def eval(self, context):
return self.left.eval(context) and self.right.eval(context)
class Or(BooleanOperation):
def eval(self, context):
return self.left.eval(context) or self.right.eval(context)
# Functions
class FunctionDeclaration(ASTNode):
def __init__(self, function_id, function_param_names, function_body):
self.function_id = function_id
self.function_param_names = function_param_names
self.function_body = function_body
def eval(self, context):
if self.function_id in context:
raise FunctionAlreadyDeclaredException(self.function_id)
context[self.function_id] = [self.function_param_names, self.function_body]
class FunctionBody(ASTNode):
def __init__(self, function_body):
self.function_body = function_body
class FunctionCall(ASTNode):
def __init__(self, function_id, function_param_values):
self.function_id = function_id
self.function_param_values = function_param_values
def eval(self, context):
if self.function_id not in context:
raise FunctionNotDeclaredException(self.function_id)
# Names of the parameters given by function declaration
param_names = context[self.function_id][0]
function_body = context[self.function_id][1].function_body
# Dict to save variables that are overridden by the internal
saved_outside_scope_variables = dict()
# Iterate over param name value pairs
# and if they're to be overridden, save them
# then over write them
for param_name, param_value in zip(param_names, self.function_param_values):
if param_name in context:
saved_outside_scope_variables[param_name] = context[param_name]
context[param_name] = param_value.eval(context)
# Temp var to hold return value
ret = None
# Execute the function body
for instruction in function_body:
if type(instruction).__name__ == "ReturnStatement":
print(instruction)
ret = instruction.eval(context)
print(type(ret), 'lmeo')
break
else:
instruction.eval(context)
# Delete the function parameters from the context
for param_name in param_names:
if param_name in saved_outside_scope_variables:
continue
elif param_name in context:
del context[param_name]
# Add the saved variables back to the context
for saved_var in saved_outside_scope_variables:
context[saved_var] = saved_outside_scope_variables[saved_var]
if not ret:
return ret
class ReturnStatement(ASTNode):
def __init__(self, value):
self.value = value
def eval(self, context):
return self.value.eval(context)
class IndexOperation(ASTNode):
def __init__(self, entity, index):
self.entity = entity
self.index = index
def eval(self, context):
variable = self.entity.eval(context)
if not hasattr(variable, "__getitem__"):
raise MissingAttribute(self.entity, "__getitem__")
return variable[self.index]
class ReadStatement(ASTNode):
def __init__(self, target):
self.target = target
def eval(self, context):
assert self.target.name in context
context[self.target.name] = (
Number(raw_input("Value for %s: " % self.target.name))
.eval(context))
class WriteStatement(ASTNode):
def __init__(self, value, newline=True):
self.value = value
self.newline = newline
def eval(self, context):
sys.stdout.write(str(self.value.eval(context)) + ("\n" if self.newline else ""))