from emis_funky_funktions import * from typing import Mapping, Sequence, Tuple, TypeAlias Expression: TypeAlias = 'Function | Application | Int | Variable | Builtin | LetBinding | ReplHole' Pattern: TypeAlias = 'NamePattern | IntPattern | SPattern | IgnorePattern' @dataclass(frozen=True) class NamePattern: """ A pattern which always succeeds to match, and binds a whole expression to a name """ name: str def binds(self, var: str) -> bool: """ Test to see if this pattern binds a given variable """ return var == self.name def match(self, e: Expression) -> Option[Sequence[Tuple[str, Expression]]]: """ Match an expression against this pattern >>> NamePattern('my_var').match(Int(1)) Some((('my_var', 1),)) """ return Some(((self.name, e),)) def __repr__(self) -> str: return self.name @dataclass(frozen=True) class IgnorePattern: """ A pattern which always succeeds to match, but binds nothing """ def binds(self, var: str) -> bool: """ Test to see if this pattern binds a given variable For an `IgnorePattern` this is always false """ return False def match(self, e: Expression) -> Option[Sequence[Tuple[str, Expression]]]: """ Match an expression against this pattern >>> IgnorePattern().match(Int(1)) Some(()) """ return Some(tuple()) def __repr__(self) -> str: return '_' @dataclass(frozen=True) class IntPattern: value: int def binds(self, var: str) -> bool: """ Test to see if this pattern binds a given variable For an `IntPattern` this is always false """ return False def match(self, e: Expression) -> Option[Sequence[Tuple[str, Expression]]]: """ Match an expression against this pattern >>> IntPattern(2).match(Int(1)) is None True >>> IntPattern(1).match(Int(1)) Some(()) """ match e: case Int(v) if v == self.value: return Some(tuple()) return None def __repr__(self) -> str: return repr(self.value) @dataclass(frozen=True) class SPattern: pred: Pattern def binds(self, var: str) -> bool: """ Test to see if this pattern binds a given variable """ return self.pred.binds(var) def match(self, e: Expression) -> Option[Sequence[Tuple[str, Expression]]]: """ Match an expression against this pattern >>> SPattern(NamePattern('n')).match(Int(1)) Some((('n', 0),)) >>> SPattern(NamePattern('n')).match(Int(0)) is None True >>> SPattern(SPattern(NamePattern('n'))).match(Int(4)) Some((('n', 2),)) """ match e: case Int(v) if v > 0: return self.pred.match(Int(v - 1)) return None def __repr__(self) -> str: return 'S ' + repr(self.pred) @dataclass(frozen=True) class ReplHole: bindings: Sequence[Tuple[str, Expression]] = tuple() def subst(self, expression: Expression, variable: str) -> Expression: return ReplHole((*self.bindings, (variable, expression))) def is_value(self) -> bool: return True def step(self) -> Option[Expression]: return None def __repr__(self) -> str: return "[]" @dataclass(frozen=True) class Builtin: name: str f: Callable[[Expression], Option[Expression]] def subst(self, expression: Expression, variable: str) -> Expression: return self def is_value(self) -> bool: return True def step(self) -> Option[Expression]: return None def try_apply(self, v: Expression) -> Option[Expression]: return self.f(v) def __repr__(self) -> str: return "'" + repr(self.name)[1:-1] + "'" @cur2 @staticmethod def _PLUS_CONST(i: int, e: Expression) -> Option[Expression]: match e: case Int(v): return Some(Int(i + v)) return None @staticmethod def _PLUS(e: Expression) -> Option[Expression]: match e: case Int(v): return Some(Builtin(f'+{v}', Builtin._PLUS_CONST(v))) return None @staticmethod def PLUS() -> 'Builtin': return Builtin('+', Builtin._PLUS) @staticmethod def S() -> 'Builtin': return Builtin('S', Builtin._PLUS_CONST(1)) @dataclass(frozen=True) class Function: forms: Sequence[Tuple[Pattern, Expression]] def subst(self, expression: Expression, variable: str) -> Expression: return Function([ (p, e if p.binds(variable) else e.subst(expression, variable)) for (p, e) in self.forms ]) def is_value(self) -> bool: return True def step(self) -> Option[Expression]: return None def try_apply(self, v: Expression) -> Option[Expression]: match tuple((bindings.val, body) for (pattern, body) in self.forms for bindings in (pattern.match(v),) if bindings is not None): case []: return None case [(bindings, body), *rest]: return Some(subst_all(bindings, body)) raise Exception('Unreachable') def __repr__(self) -> str: return '{ ' + ', '.join('"' + repr(repr(p))[1:-1] + '" : ' + repr(e) for (p, e) in self.forms) + ' }' @dataclass class LetBinding: lhs: str rhs: Expression body: Expression def subst(self, expression: Expression, variable: str) -> Expression: if self.lhs == variable: return self else: return LetBinding( self.lhs, self.rhs.subst(expression, variable), self.body.subst(expression, variable) ) def is_value(self) -> bool: return False def step(self) -> Option[Expression]: if self.rhs.is_value(): return Some(self.body.subst( self.rhs.subst( LetBinding(self.lhs, self.rhs, Variable(self.lhs)), self.lhs ), self.lhs )) else: return map_opt(lambda rhs_step: LetBinding(self.lhs, rhs_step, self.body), self.rhs.step() ) def __repr__(self) -> str: if isinstance(self.body, LetBinding) or isinstance(self.body, Application): return f'L[ {repr(self.lhs)}, {repr(self.rhs)},{repr(self.body)[1:-1]}]' else: return f'L[ {repr(self.lhs)}, {repr(self.rhs)}, {repr(self.body)}]' @dataclass class Application: first: Expression args: Sequence[Expression] def subst(self, expression: Expression, variable: str) -> Expression: return Application( self.first.subst(expression, variable), [r.subst(expression, variable) for r in self.args] ) def is_value(self) -> bool: return False def step(self) -> Option[Expression]: match self.args: case []: return Some(self.first) case [a, *rest]: match self.first.step(): case Some(first_stepped): return Some(Application(first_stepped, self.args)) case None: match a.step(): case Some(a_stepped): return Some(Application(self.first, [a_stepped, *rest])) case None: if isinstance(self.first, Function) or isinstance(self.first, Builtin): return map_opt( lambda f_sub: Application(f_sub, rest), self.first.try_apply(a) ) elif isinstance(self.first, Variable): lhs = self.first.name rhs = a body = rest match body: case []: return Some(ReplHole()) case [body_first, *body_rest]: return Some(LetBinding(lhs, rhs, Application(body_first, body_rest))) else: return None raise Exception('Unreachable') def __repr__(self) -> str: return f'[ {repr(self.first)}, ' + ', '.join(repr(e) for e in self.args) + ' ]' @dataclass class Int: value: int def subst(self, expression: Expression, variable: str) -> Expression: return self def is_value(self) -> bool: return True def step(self) -> Option[Expression]: return None def __repr__(self) -> str: return str(self.value) @dataclass class Variable: name: str def subst(self, expression: Expression, variable: str) -> Expression: if variable == self.name: return expression else: return self def is_value(self) -> bool: return False def step(self) -> Option[Expression]: match self.name: case '+': return Some(Builtin.PLUS()) case 'S': return Some(Builtin.S()) return None def __repr__(self) -> str: return '"' + repr(self.name)[1:-1] + '"' def subst_all(bindings: Sequence[Tuple[str, Expression]], body: Expression) -> Expression: match bindings: case []: return body case [(var, replacement), *rest]: return subst_all(rest, body.subst(replacement, var)) raise Exception('Unreachable')