Pantograph/pantograph/expr.py

84 lines
2.2 KiB
Python
Raw Normal View History

2024-04-22 13:00:06 -07:00
"""
2024-04-26 19:09:26 -07:00
Data structures for expressions and goals
2024-04-22 13:00:06 -07:00
"""
from dataclasses import dataclass
2024-04-26 19:15:14 -07:00
from typing import Optional, Union, List
2024-04-22 13:00:06 -07:00
Expr = str
def parse_expr(payload: dict) -> Expr:
return payload["pp"]
@dataclass(frozen=True)
class Variable:
t: Expr
v: Optional[Expr] = None
name: Optional[str] = None
@staticmethod
2024-04-26 19:15:14 -07:00
def parse(payload: dict) -> "Variable": # Replace 'Self' with 'Variable'
2024-04-22 13:00:06 -07:00
name = payload.get("userName")
t = parse_expr(payload["type"])
v = payload.get("value")
if v:
v = parse_expr(v)
return Variable(t, v, name)
def __str__(self):
result = self.name if self.name else "_"
result += f" : {self.t}"
if self.v:
result += f" := {self.v}"
2024-04-22 13:00:06 -07:00
return result
@dataclass(frozen=True)
class Goal:
2024-04-26 19:15:14 -07:00
variables: List[Variable]
2024-04-22 13:00:06 -07:00
target: Expr
name: Optional[str] = None
is_conversion: bool = False
2024-04-22 13:00:06 -07:00
@staticmethod
2024-04-26 19:09:26 -07:00
def sentence(target: Expr) -> "Goal": # Replace 'Self' with 'Goal'
2024-04-22 13:00:06 -07:00
return Goal(variables=[], target=target)
@staticmethod
2024-04-26 19:15:14 -07:00
def parse(payload: dict) -> "Goal": # Replace 'Self' with 'Goal'
2024-04-22 13:00:06 -07:00
name = payload.get("userName")
2024-04-22 22:38:20 -07:00
variables = [Variable.parse(v) for v in payload["vars"]]
2024-04-22 13:00:06 -07:00
target = parse_expr(payload["target"])
2024-04-26 19:15:14 -07:00
is_conversion = payload.get("isConversion", False) # Handle missing keys
return Goal(variables, target, name, is_conversion)
2024-04-22 13:00:06 -07:00
def __str__(self):
front = "|" if self.is_conversion else ""
2024-04-26 19:09:26 -07:00
return "\n".join(str(v) for v in self.variables) + f"\n{front} {self.target}"
2024-04-22 13:00:06 -07:00
@dataclass(frozen=True)
class GoalState:
state_id: int
2024-04-26 19:15:14 -07:00
goals: List[Goal]
2024-04-22 13:00:06 -07:00
@property
def is_solved(self) -> bool:
2024-04-22 22:38:20 -07:00
"""
WARNING: Does not handle dormant goals.
"""
2024-04-22 13:00:06 -07:00
return not self.goals
2024-04-22 13:11:28 -07:00
2024-04-22 22:38:20 -07:00
@staticmethod
2024-04-26 19:15:14 -07:00
def parse(payload: dict) -> "GoalState": # Replace 'Self' with 'GoalState'
state_id = payload.get("nextStateId", 0) # Handle missing keys
goals = [Goal.parse(g) for g in payload.get("goals", [])]
2024-04-22 22:38:20 -07:00
return GoalState(state_id, goals)
2024-04-26 19:09:26 -07:00
2024-04-22 13:11:28 -07:00
@dataclass(frozen=True)
class TacticHave:
branch: str
2024-04-22 22:38:20 -07:00
@dataclass(frozen=True)
class TacticCalc:
step: str
2024-04-22 13:11:28 -07:00
2024-04-22 22:38:20 -07:00
Tactic = Union[str, TacticHave, TacticCalc]