import logging
from pathlib import Path
from typing import Optional, Tuple

from llm_utils.openai_api.chat import Chat

from tp_lodge.task_planning.models.pddl.pddl_domain import PDDLDomain
from tp_lodge.task_planning.models.pddl.pddl_problem import PDDLProblem
from tp_lodge.task_planning.models.sas.sas_plan import SasPlan
from tp_lodge.task_planning.pddl_planner.hi_planner.mixins.gen_definitions_mixin import (
    GenDefinitionsMixin,
    verify_retries,
)
from tp_lodge.utils.pddl_parse_utils import get_domain_text, get_problem_text, inject_definitions_into_chat, inject_defs_into_text
from tp_lodge.utils.pddl_utils import (
    get_valid_predicates,
)

logger = logging.getLogger(__name__)


class PlannerMixin(GenDefinitionsMixin):

    @verify_retries
    def plan_with_llm(
        self,
        domain: PDDLDomain,
        domain_knowledge: Optional[str],
        problem: PDDLProblem,
        demo_plan: Optional[str],
        domain_changed: bool,
        last_plan: Optional[SasPlan],
        last_exception: Optional[str],
        out_dir: Path,
        use_cache: bool = True,
        exception: Optional[str] = None,
    ):
        plan_file = out_dir / "llm-plan.cache.plan"
        if use_cache and plan_file.is_file():
            logger.info("Using cached plan")
            sas_plan = SasPlan.from_string(plan_file.read_text())
            used_cached = True
        else:
            used_cached = False

            # init_consts = get_constants_in_term(simple_problem.initial_state)
            # goal_consts = get_constants_in_term(simple_problem.initial_state)

            simple_problem = problem.copy_with(
                # objects=list(set(init_consts + goal_consts)),
                initial_state=get_valid_predicates(problem.initial_state),
                # goal_state=problem.goal_state,
            )

        errors = []
        try:
            # if not domain_changed and last_plan is not None:
            # last_plan = None
            if last_plan is not None:
                if last_exception is None:
                    sas_plan_str = last_plan.to_string()

                else:
                    assert last_exception is not None
                    sas_plan_str = self.llm_interface.try_correct_plan(
                        domain=get_domain_text(domain),
                        domain_knowledge=domain_knowledge,
                        problem=get_problem_text(simple_problem),
                        demo_plan=demo_plan if self.storage.use_ai_plan_for_llm_planning else None,
                        last_exception=last_exception,
                        last_plan=last_plan.to_string(),
                        out_dir=out_dir,
                    )

            else:
                sas_plan_str = self.llm_interface.try_gen_plan(
                    domain=get_domain_text(domain),
                    domain_knowledge=domain_knowledge,
                    problem=get_problem_text(simple_problem),
                    demo_plan=demo_plan if self.storage.use_ai_plan_for_llm_planning else None,
                    out_dir=out_dir,
                    exception=exception or last_exception,
                )
            sas_plan = SasPlan.from_string(sas_plan_str)
        except ValueError as e:
            errors.append(str(e))
            sas_plan = None

        if len(errors) == 0:
            assert sas_plan is not None, "SAS plan should not be None after generation"
            if len(sas_plan.actions) == 0:
                errors.append(ValueError("LLM generated empty plan"))
            else:
                errors = sas_plan.validate(domain=domain, problem=problem, validate_types=False)

        if len(errors) > 0:
            plan_file.unlink(missing_ok=True)
            if used_cached:
                logger.info("Cached plan was not valid. Regenerate plan")
                exception = None
            else:
                logger.info("LLM plan was not valid. Regenerate plan")
                exception = "Following LLM-plan is not valid:\n %s\n\nErrors:\n%s" % (sas_plan.to_string(), errors)
            self.storage.add_val_retry()
            return self.plan_with_llm(
                domain=domain,
                domain_knowledge=domain_knowledge,
                problem=problem,
                out_dir=out_dir,
                demo_plan=demo_plan,
                domain_changed=False,
                # last_exception=last_exception,
                # last_plan=last_plan,
                last_exception=exception,
                last_plan=sas_plan,
                use_cache=False,
                exception=exception,
            )
        assert sas_plan is not None

        # check plan with VAL
        exception, val_success = self.ai_validator.validate(
            domain=str(domain.to_pddl()), problem=str(problem.to_pddl()), plan=sas_plan.to_string(), options="-v"
        )

        if not val_success and used_cached:
            plan_file.unlink()
            logger.info("Cached plan was not valid. Regenerate plan")
            return self.plan_with_llm(
                domain=domain,
                domain_knowledge=domain_knowledge,
                problem=problem,
                out_dir=out_dir,
                demo_plan=demo_plan,
                domain_changed=False,
                last_exception=exception,
                last_plan=sas_plan,
                use_cache=False,
                exception=exception,
            )
        elif val_success and use_cache:
            plan_file.write_text(sas_plan.to_string())

        assert len(sas_plan.actions) > 0, "SAS plan should not be empty after generation"

        return val_success, exception, sas_plan

    def plan_with_ai_planner(self, domain: PDDLDomain, problem: PDDLProblem):
        exception, success, sas_plan = self.ai_planner.plan(domain=domain.to_pddl(), problem=problem.to_pddl())

        if success:
            assert sas_plan is not None
            exception, val_success = self.ai_validator.validate(
                domain=str(domain.to_pddl()), problem=str(problem.to_pddl()), plan=sas_plan.to_string(), options="-v"
            )
        else:
            val_success = success
        logger.info("AI planner success: %s" % str(success))
        return val_success, exception, sas_plan

    @verify_retries
    def retry_ai_planning(
        self,
        domain: PDDLDomain,
        problem: PDDLProblem,
        chat: Chat,
        out_dir: Path,
        domain_knowledge: str,
        chat_file: Path,
        last_plan: Optional[SasPlan] = None,
        last_exception: Optional[str] = None,
        replan_plan: bool = True,
        domain_changed: bool = True,
    ) -> Tuple[SasPlan, Chat, PDDLDomain, PDDLProblem]:
        assert domain is not None and problem is not None
        assert domain.fully_defined and problem.fully_defined

        # if problem is not None:
        # fix the init state
        # problem = copy_problem_w_args(problem=problem, init=prev_problem.init)
        ai_plan_file = out_dir / "ai-plan.cache.plan"
        sas_plan = None
        if ai_plan_file.is_file():
            sas_plan = SasPlan.from_string(ai_plan_file.read_text())
            if len(sas_plan.validate(domain=domain, problem=problem)) == 0:
                successful, repair_advice = self.ai_validator.validate_plan_executes_successfully(
                    domain=str(domain.to_pddl()), problem=str(problem.to_pddl()), plan=sas_plan.actions[0].to_string()
                )
                if not successful:
                    sas_plan = None
                    val_success = False
                else:
                    exception, val_success = self.ai_validator.validate(
                        domain=str(domain.to_pddl()),
                        problem=str(problem.to_pddl()),
                        plan=sas_plan.to_string(),
                        options="-v",
                    )
                if (out_dir / "should-replan").is_file():
                    # if replan is forced, save current plan as sas_plan and reset current plan
                    last_plan = sas_plan
                    sas_plan = None
                    val_success = False
                    (out_dir / "should-replan").unlink()

                # if not val_success or last_exception is not None:
                if last_exception is not None:
                    # we only retain the cached plan if we are currently not retrying (last_exception is None)
                    # last_plan = sas_plan
                    sas_plan = None
                    val_success = False

            else:
                ai_plan_file.unlink()
                sas_plan = None
                val_success = False
        else:
            val_success = False

        if not val_success:
            ai_val_success, _, ai_sas_plan = self.plan_with_ai_planner(domain=domain, problem=problem)

            if self.storage.use_ai_plan and ai_val_success:
                val_success = ai_val_success
                sas_plan = ai_sas_plan

        if not val_success:
            chat_file = out_dir / "chat-reprompt.chat"
            if sas_plan is None:
                # we did load a sas_plan from cache
                val_success, exception, sas_plan = self.plan_with_llm(
                    domain=domain,
                    domain_knowledge=domain_knowledge,
                    problem=problem,
                    demo_plan=ai_sas_plan.to_string() if ai_sas_plan is not None else None,
                    out_dir=out_dir,
                    domain_changed=domain_changed,
                    last_exception=last_exception if replan_plan else None,
                    last_plan=last_plan,
                    use_cache=False,
                )

            # check types
            errors = sas_plan.validate(domain=domain, problem=problem, validate_types=True)
            successful = True
            exception = None
            if len(errors) > 0:
                errors_str = "\n".join(map(lambda x: "- %s" % x, errors))
                logger.info("LLM plan was not valid:\n%s\n Regenerate plan" % errors_str)
                exception = "LLM-plan and domain don't align. If the planner uses unknown actions, check whether they must be defined in the domain.\nErrors:\n%s\n" % (errors_str)
                # exception = "LLM-plan and domain don't align:\n%s\n\nErrors:\n%s\n" % (sas_plan.to_string(), errors_str)
                successful = False

            # check first action is ok
            # if successful:
            #     successful, repair_advice = self.ai_validator.validate_plan_executes_successfully(
            #         domain=str(domain.to_pddl()), problem=str(problem.to_pddl()), plan=sas_plan.actions[0].to_string()
            #     )
            #     if not successful:
            #         logger.info(
            #             "First action is not valid. Regenerating pddl files with chat and exception: %s" % repair_advice
            #         )
            #         # exception = "The first action of following plan is invalid:\nException: %s" % repair_advice
            #         exception = "The first action of following plan is invalid:\nPlan:\n%s\n\nException: %s" % (
            #             sas_plan.to_string(),
            #             repair_advice,
            #         )

            if not successful:
                use_history = True
                new_problem, new_domain, added_chat = self.regenerate_pddl_definitions(
                    domain=domain,
                    problem=problem,
                    domain_knowledge=domain_knowledge,
                    exception=exception,
                    chat_file=chat_file,
                    chat=chat if use_history else None,
                    out_dir=out_dir,
                )

                # TODO: must be validated, not sure whether domain as previous domain for `get_change_to` is correct
                chat = added_chat
                # op_change, pred_change = domain.get_change_to(new_domain)
                # chat = inject_definitions_into_chat(
                #     chat=chat,
                #     operator_changes=op_change,
                #     predicate_changes=pred_change,
                #     domain=new_domain,
                #     problem=new_problem,
                #     prev_problem=problem,
                # )

                self.storage.add_val_retry()
                return self.retry_ai_planning(
                    domain=new_domain,
                    problem=new_problem,
                    chat=chat,
                    chat_file=chat_file,
                    out_dir=out_dir,
                    domain_knowledge=domain_knowledge,
                    last_plan=sas_plan,
                    last_exception=exception,
                    domain_changed=new_domain != domain,
                )
        else:
            assert sas_plan is not None
            ai_plan_file.write_text(sas_plan.to_string())

        ai_plan_file.write_text(sas_plan.to_string())

        logger.info("Successfully found plan (VAL: %s)" % str(val_success))
        logger.info("\n" + "#" * 100 + ("\n%s\n" % sas_plan.to_string()) + "#" * 100)

        return sas_plan, chat, domain, problem
