from dataclasses import dataclass, field
from time import time
from typing import Dict, List, Optional
from uuid import uuid4

from pddl.core import Action, Formula
from python_utils.string_utils import xml_escape

from tp_lodge.task_planning.models.pddl.pddl_object import PDDLObject
from tp_lodge.task_planning.models.pddl.pddl_operation import PDDLOperation
from tp_lodge.utils.pddl_domain_syntax import parse_action
from tp_lodge.utils.pddl_lib_utils import copy_action_w_args


@dataclass
class PDDLOperator(PDDLObject):
    definition: Action
    description: str
    id: str = field(default_factory=lambda: str(uuid4()))
    last_operation: PDDLOperation = PDDLOperation.ADD

    parent_operator_id: str = "root"
    verified: bool = False
    mapped_skill_sequence: Optional[List[str]] = None

    def __post_init__(self):
        definition = parse_action(str(self.definition).lower())
        # so we for sure have a difference between skills and operators
        self.definition = copy_action_w_args(definition, name=definition.name.replace("_", "-"))
        self.description = xml_escape(self.description)

    @property
    def name(self) -> str:
        return self.definition.name

    def to_json(self) -> Dict:
        return {
            "id": self.id,
            "definition": str(self.definition),
            "description": self.description,
            "last_operation": self.last_operation.value,
            "parent_operator_name": self.parent_operator_id,
            "verified": self.verified,
            "mapped_skill_sequence": self.mapped_skill_sequence,
        }

    @classmethod
    def from_json(cls, data: Dict) -> "PDDLOperator":
        definition = parse_action(data["definition"])
        description = data["description"]
        last_operation = PDDLOperation(data["last_operation"])
        parent_operator_name = data.get("parent_operator_name", "root")
        verified = data.get("verified", False)
        mapped_skill_sequence = data.get("mapped_skill_sequence", None)

        return PDDLOperator(
            id=data.get("id", definition.name),
            definition=definition,
            description=description,
            last_operation=last_operation,
            parent_operator_id=parent_operator_name,
            verified=verified,
            mapped_skill_sequence=mapped_skill_sequence,
        )

    def __eq__(self, value):
        if not isinstance(value, PDDLOperator):
            return False
        return self.definition == value.definition

    def param_names(self) -> List[str]:
        return ["%s_%s" % (p.name.replace("-", "_"), list(p.type_tags)[0]) for p in self.definition.parameters]

    def update_inplace(self, definition: Optional[Action] = None, description: Optional[str] = None, parent_operator_id: Optional[str] = None):
        changed = False
        if definition is not None and self.definition != definition:
            changed = True
            self.definition = definition
        if description is not None and self.description != description:
            changed = True
            self.description = description
        if parent_operator_id is not None and self.parent_operator_id != parent_operator_id:
            self.parent_operator_id = parent_operator_id

        if changed:
            self.last_operation = PDDLOperation.EDIT
            self.verified = False

    def copy_with(
        self,
        parent_operator_id: Optional[str] = None,
        precondition: Optional[Formula] = None,
        effect: Optional[Formula] = None,
        last_operation: Optional[PDDLOperation] = None,
    ) -> "PDDLOperator":
        return PDDLOperator(
            id=self.id,
            parent_operator_id=(
                parent_operator_id if parent_operator_id is not None else self.parent_operator_id
            ),
            definition=copy_action_w_args(self.definition, precondition=precondition, effect=effect),
            description=self.description,
            last_operation=last_operation if last_operation is not None else self.last_operation,
            verified=self.verified,
            mapped_skill_sequence=self.mapped_skill_sequence,
        )

    @property
    def num_params(self) -> int:
        return len(self.definition.parameters)
