from pathlib import Path
from pddl.parser.domain import DomainParser
from pddl.parser.problem import ProblemParser


env_dir_1 = Path(__file__).parent.parent.parent / "data/domains" / "household"
env_dir = Path(__file__).parent.parent.parent / "data/domains" / "household-simplified"

simplified_domain_file = env_dir / "domain.pddl"
domain = DomainParser()(simplified_domain_file.read_text())
orig_domain = DomainParser()((simplified_domain_file.parent.parent / "household/domain.pddl").read_text())
predicate_names = {p.name: p for p in domain.predicates}

from tp_lodge.task_planning.models.pddl.pddl_domain import PDDLDomain
orig_domain = PDDLDomain(types=orig_domain.types, actions=[], predicates=[])

for problem_file in env_dir_1.glob("p*.pddl"):
    problem = ProblemParser()(problem_file.read_text())
    out_file = env_dir / problem_file.name

    upd_objects = []
    for obj in problem.objects:
        p_t = [t for t in orig_domain.parent_types(obj.type_tag) if t in domain.types.keys()]
        from pddl.core import Constant
        if len(p_t) == 0:
            print(obj, "has no parent type in the simplified domain")
            continue
        upd_objects.append(Constant(name=obj.name, type_tag=p_t[0]))

    from pddl.core import Predicate, And
    from pddl.logic.base import Not

    def update_preds(preds):
        if isinstance(preds, And):
            return And(*update_preds(preds.operands))
        if not isinstance(preds, (list, frozenset, tuple, set)):
            return And(*update_preds([preds]))

        updated_init = []
        for pred in preds:
            is_not = isinstance(pred, Not)
            if is_not:
                pred = pred.argument

            assert isinstance(pred, Predicate)
            orig_pred = predicate_names.get(pred.name)
            assert orig_pred is not None, f"Predicate {pred.name} not found in original domain"
            if orig_pred.arity != pred.arity:
                if pred.name == "at-agent":
                    up_p = Predicate(pred.name, pred.terms[1])
                elif pred.name == "holding":
                    up_p = Predicate(pred.name, pred.terms[1])
                else:
                    print(f"Predicate {pred.name} has arity mismatch: {orig_pred.arity} vs {pred.arity}")
                    continue
            else:
                up_p = pred
            updated_init.append(up_p if not is_not else Not(up_p))
        return updated_init

    upd_init = update_preds(problem.init)
    for obj in problem.objects:
        if obj.type_tag.replace("_", "-") in predicate_names:
            n_pred = Predicate(obj.type_tag.replace("_", "-"), Constant(obj.name))
            print(n_pred)
            upd_init.append(n_pred)

    upd_goal = update_preds(problem.goal)
    
    from pddl.core import Problem
    upd_problem =  Problem(
        name=problem.name,
        domain_name=domain.name,
        objects=upd_objects,
        init=upd_init,
        goal=upd_goal
    )


    out_file = env_dir / problem_file.name
    out_file.write_text(str(upd_problem))