Piet_interpreter/interpreter/executionFunctions.py
2020-05-06 22:42:04 +02:00

111 lines
3.7 KiB
Python

import copy
from typing import Union, List, Callable
import sys
import numpy as np
# sys.path.insert(0, "../")
from interpreter import imageWrapper as imageWrapper
from interpreter import lexer as lexer
from interpreter import tokens as tokens
from interpreter import movement as movement
from interpreter import colors as colors
from interpreter import tokenFunctions as runner
from interpreter import errors as errors
from interpreter.dataStructures import programState, position, direction
def interpret(image: np.ndarray) -> Union[programState, List[BaseException]]:
"""
Interprets and executes a Piet image
:param image: Input image
:return: Either the final state of the program, or a list of exceptions
"""
graph = lexer.graphImage(image)
if len(graph[1]) > 0:
print("The following exceptions occured while making the graph:\n{}".format("".join(list(map(lambda x: "\t{}\n".format(x), graph[1])))))
return graph[1]
startPosition = position((0, 0))
pointers = direction((0, 0))
PS = programState(graph[0], startPosition, pointers)
result = runProgram(image, PS)
if isinstance(result, BaseException):
print("The following exceptions occured while executing the next step:\n{}".format(result))
return [result]
return result
def runProgram(image: np.ndarray, PS: programState) -> Union[programState, BaseException]:
"""
Executes all steps from the image
:param image: input image
:param PS: current program state with which to make the next step
:return: Either the last program state, or a runtime exception
"""
newState = copy.deepcopy(PS)
if colors.isBlack(imageWrapper.getPixel(image, newState.position)):
return errors.inBlackPixelError("Programstate starts in black pixel at {}".format(newState.position))
currentCodel = imageWrapper.getCodel(image, newState.position)
newGraph = newState.graph.graph
graphNode = newGraph[currentCodel]
newToken = graphNode.graphNode[newState.direction][0]
if isinstance(newToken, tokens.terminateToken):
return newState
newState = takeStep(image, newState)
if isinstance(newState, BaseException):
return newState
return runProgram(image, newState)
def countSteps(f: Callable):
def inner(image: np.ndarray, PS: programState):
inner.counter += 1
return f(image, PS)
inner.counter = 0
return inner
@countSteps
def takeStep(image: np.ndarray, PS: programState) -> Union[programState, BaseException]:
"""
Takes a single step from the programstate
:param image: input image
:param PS: input programstate
:return: Returns either the resulting programstate, or an exception that occurred
"""
newState = copy.deepcopy(PS)
currentCodel = imageWrapper.getCodel(image, newState.position)
newGraph = newState.graph.graph
graphNode = newGraph[currentCodel]
newToken = graphNode.graphNode[newState.direction][0]
edgePosition = graphNode.graphNode[newState.direction][1]
result = runner.executeToken(newToken, newState.direction, newState.dataStack)
if isinstance(result, BaseException):
return result
# If the next token is either white or color, just move along. If the token was black (or terminate), the direction
# is already changed
if isinstance(newToken, (tokens.toWhiteToken, tokens.toColorToken)):
newState.position = movement.getNextPosition(edgePosition, newState.direction.pointers[0])
newState.direction = result[0]
newState.dataStack = result[1]
return newState
if __name__ == "__main__":
sys.setrecursionlimit(1000000)
im = imageWrapper.getImage("../Piet_hello.png")
interpret(im)