from enum import Enum class NodeType(Enum): INTEGER = 1 STRING = 2 NOTE = 3 BLOCK = 4 ARGUMENTS = 5 IDENTIFIER = 6 ASSIGN = 7 PROGRAM = 8 ASTERISK = 9 COLON = 10 FUNCTION_CALL = 11 COMMA = 12 PERCENT = 13 class Node: def __init__(self): self.children = [] def __repr__(self): return self.__str__() def __len__(self): return len(self.children) def __getitem__(self, index): return self.children[index] def append(self, node): self.children.append(node) def pop(self, index): return self.children.pop(index) class Program(Node): def __init__(self): Node.__init__(self) self.type = NodeType.PROGRAM def __str__(self): return "Program:\n" + "\n".join([str(e) for e in self.children]) class BlockNode(Node): def __init__(self): Node.__init__(self) self.type = NodeType.BLOCK def __str__(self): return "B{\n" + "\n".join([str(e) for e in self.children]) + "\n}" class ArgumentsNode(Node): def __init__(self): Node.__init__(self) self.type = NodeType.ARGUMENTS def __str__(self): return "@(" + ", ".join([str(e) for e in self.children]) + ")" class IdentifierNode(Node): def __init__(self, identifier): self.type = NodeType.IDENTIFIER self.identifier = identifier def __str__(self): return f"L'{self.identifier}'" class AssignExpression(Node): def __init__(self, target, value): self.type = NodeType.ASSIGN self.target = target self.value = value def __str__(self): return f"A[{self.target} = {self.value}]" class AsteriskStatementNode(Node): def __init__(self, iterator, statement): self.type = NodeType.ASTERISK self.iterator = iterator self.statement = statement def __str__(self): return f"*({self.iterator}: {self.statement})" class ColonNode(Node): def __init__(self, a, b): self.type = NodeType.COLON self.a = a self.b = b def __str__(self): return f":({self.a}, {self.b})" class ExpressionNode(Node): def __str__(self): return f"{self.__class__.__name__}('{self.value}')" class IntegerLiteralNode(ExpressionNode): def __init__(self, value): self.type = NodeType.INTEGER self.value = value def __str__(self): return f"i'{self.value}'" class StringLiteralNode(ExpressionNode): def __init__(self, value): self.type = NodeType.STRING self.value = value def __str__(self): return f"s'{self.value}'" class NoteLiteralNode(ExpressionNode): def __init__(self, value): self.type = NodeType.NOTE self.value = value def __str__(self): return f"n'{self.value}'" class FunctionCallNode(Node): def __init__(self, identifier, arguments): self.type = NodeType.FUNCTION_CALL self.identifier = identifier self.arguments = arguments def __str__(self): return f"F({self.identifier}: {self.arguments})" class CommaNode(Node): def __init__(self): self.type = NodeType.COMMA def __str__(self): return "[,]" class PercentNode(Node): def __init__(self, value): self.type = NodeType.PERCENT self.value = value def __str__(self): return f"%'{self.value}'"