Commit d0deda8d authored by Igor Dejanovic's avatar Igor Dejanovic

Reworked robot examples to use visitor

parent 9e09e374
...@@ -23,68 +23,52 @@ from __future__ import print_function, unicode_literals ...@@ -23,68 +23,52 @@ from __future__ import print_function, unicode_literals
from arpeggio import * from arpeggio import *
# Grammar rules # Grammar rules
def robot(): return Kwd('begin'), ZeroOrMore(command), Kwd('end'), EOF def robot(): return 'begin', ZeroOrMore(command), 'end', EOF
def command(): return [up, down, left, right] def command(): return [UP, DOWN, LEFT, RIGHT]
def up(): return 'up' def UP(): return 'up'
def down(): return 'down' def DOWN(): return 'down'
def left(): return 'left' def LEFT(): return 'left'
def right(): return 'right' def RIGHT(): return 'right'
# Semantic actions # Semantic actions visitor
class Up(SemanticAction): class RobotVisitor(PTNodeVisitor):
def first_pass(self, parser, node, children):
if parser.debug: def visit_robot(self, node, children):
if self.debug:
print("Evaluating position")
position = [0, 0]
for move in children:
position[0] += move[0]
position[1] += move[1]
return position
def visit_command(self, node, children):
if self.debug:
print("Command")
return children[0]
def visit_UP(self, node, children):
if self.debug:
print("Going up") print("Going up")
return (0, 1) return (0, 1)
def visit_DOWN(self, node, children):
class Down(SemanticAction): if self.debug:
def first_pass(self, parser, node, children):
if parser.debug:
print("Going down") print("Going down")
return (0, -1) return (0, -1)
def visit_LEFT(self, node, children):
class Left(SemanticAction): if self.debug:
def first_pass(self, parser, node, children):
if parser.debug:
print("Going left") print("Going left")
return (-1, 0) return (-1, 0)
def visit_RIGHT(self, node, children):
class Right(SemanticAction): if self.debug:
def first_pass(self, parser, node, children):
if parser.debug:
print("Going right") print("Going right")
return (1, 0) return (1, 0)
class Command(SemanticAction):
def first_pass(self, parser, node, children):
if parser.debug:
print("Command")
return children[0]
class Robot(SemanticAction):
def first_pass(self, parser, node, children):
if parser.debug:
print("Evaluating position")
position = [0, 0]
for move in children:
position[0] += move[0]
position[1] += move[1]
return position
# Connecting rules with semantic actions
robot.sem = Robot()
command.sem = Command()
up.sem = Up()
down.sem = Down()
left.sem = Left()
right.sem = Right()
def main(debug=False): def main(debug=False):
# Program code # Program code
input_program = ''' input_program = '''
...@@ -105,10 +89,10 @@ def main(debug=False): ...@@ -105,10 +89,10 @@ def main(debug=False):
# We create a parse tree out of textual input # We create a parse tree out of textual input
parse_tree = parser.parse(input_program) parse_tree = parser.parse(input_program)
# getASG will start semantic analysis. # visit_parse_tree will start semantic analysis.
# In this case semantic analysis will evaluate expression and # In this case semantic analysis will evaluate expression and
# returned value will be the final position of the robot. # returned value will be the final position of the robot.
result = parser.getASG() result = visit_parse_tree(parse_tree, RobotVisitor(debug=debug))
if debug: if debug:
print("position = ", result) print("position = ", result)
......
...@@ -33,16 +33,8 @@ LEFT <- 'left'; ...@@ -33,16 +33,8 @@ LEFT <- 'left';
RIGHT <- 'right'; RIGHT <- 'right';
''' '''
# Semantic actions # Semantic actions visitor
from robot import Up, Down, Left, Right, Command, Robot from robot import RobotVisitor
semantic_actions = {
'robot': Robot(),
'command': Command(),
'UP': Up(),
'DOWN': Down(),
'LEFT': Left(),
'RIGHT': Right()
}
def main(debug=False): def main(debug=False):
...@@ -57,7 +49,6 @@ def main(debug=False): ...@@ -57,7 +49,6 @@ def main(debug=False):
end end
''' '''
# First we will make a parser - an instance of the robot parser model. # First we will make a parser - an instance of the robot parser model.
# Parser model is given in the form of PEG specification therefore we # Parser model is given in the form of PEG specification therefore we
# are using ParserPEG class. # are using ParserPEG class.
...@@ -66,10 +57,10 @@ def main(debug=False): ...@@ -66,10 +57,10 @@ def main(debug=False):
# We create a parse tree out of textual input # We create a parse tree out of textual input
parse_tree = parser.parse(input) parse_tree = parser.parse(input)
# getASG will start semantic analysis. # visit_parse_tree will start semantic analysis.
# In this case semantic analysis will evaluate expression and # In this case semantic analysis will evaluate expression and
# returned value will be the final position of the robot. # returned value will be the final position of the robot.
return parser.getASG(sem_actions=semantic_actions) return visit_parse_tree(parse_tree, RobotVisitor(debug=debug))
if __name__ == "__main__": if __name__ == "__main__":
# In debug mode dot (graphviz) files for parser model # In debug mode dot (graphviz) files for parser model
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment