import re
from dataclasses import dataclass
from typing import List

from tp_lodge.task_planning.models.pddl.pddl_domain import PDDLDomain
from tp_lodge.task_planning.models.pddl.pddl_problem import PDDLProblem


@dataclass
class SasAction:
    name: str
    args: List[str]

    def __post_init__(self):
        self.name = self.name.lower()

    def __hash__(self):
        return hash((self.name, tuple(self.args)))

    def validate(self, domain: PDDLDomain, problem: PDDLProblem, validate_types: bool) -> List[str]:
        """
        Validates the action against the domain.
        :returns a list of errors if the action is invalid, otherwise an empty list.
        """
        assert domain.has_unique_names(), "Domain must have unique names to validate actions."
        known_actions = {action.name: action for action in domain.operators}
        if self.name not in known_actions:
            if validate_types:
                return [f"Unknown action: {self.name}"]
            else:
                return []

        if len(self.args) != known_actions[self.name].num_params:
            return [
                f"Action {self.name} expects {known_actions[self.name].num_params} arguments, but got {len(self.args)}."
            ]

        action = known_actions[self.name]
        for arg_name, param in zip(self.args, action.definition.parameters):
            arg = next(filter(lambda o: o.name == arg_name, problem.objects), None)
            if arg is None:
                return [f"Unknown object: {arg_name}"]

            arg_type = arg.type_tag
            assert len(param.type_tags) == 1

            if validate_types:
                if list(param.type_tags)[0] not in domain.parent_types(arg_type):
                    return [
                        f"Object {arg_name} of type {arg_type} is not a valid parameter for action {self.name} of type {list(param.type_tags)[0]}."
                    ]

        return []

    @staticmethod
    def from_string(sas_action_str: str) -> "SasAction":
        """
        Parses a SAS action string and returns a SasAction object.
        """
        match = re.match(r"\(([\w\-]+)(?: +([^\)]+)|\s*)\)", sas_action_str)
        assert match is not None, sas_action_str
        action_name = match.group(1)
        if match.group(2) is None:
            action_args = []
        else:
            action_args = [p.strip() for p in match.group(2).split()]
        return SasAction(name=action_name, args=action_args)

    def to_string(self) -> str:
        """
        Converts the SasAction object to a string representation.
        """
        return f"({self.name} {' '.join(self.args)})"
