import logging
import tempfile
from pathlib import Path
from typing import Optional

from requests.exceptions import ConnectionError

from tp_lodge.task_planning.models.sas.sas_plan import SasPlan
from tp_lodge.task_planning.pddl_planner.ai_planner.ai_planner import AIPlanner

logger = logging.getLogger(__name__)


class FastDownwardAIPlanner(AIPlanner):

    def __init__(self, alias: Optional[str] = None, search_time_limit: Optional[int] = None):
        super().__init__()
        self.alias = alias
        self.search_time_limit = search_time_limit

    def _run_planner(self, docker_host_dir: Path, domain_file: str, problem_file: str):
        assert self._is_docker_running()
        # heuristics can be found at https://www.fast-downward.org/Doc/Evaluator

        cmd = ""

        if self.alias:
            cmd += f" --alias {self.alias} "
        else:
            cmd += " --alias lama-first "

        if self.search_time_limit:
            cmd += f" --search-time-limit {self.search_time_limit} "
        else:
            cmd += " --search-time-limit 60 "

        cmd += " /pddls/%s /pddls/%s" % (domain_file, problem_file)
        # cmd += " --search lazy_greedy([ff()],preferred=[ff()])"

        container = self.client.containers.run(
            image="aibasel/downward",
            command=cmd,
            # command='/pddls/%s /pddls/%s --search "astar(blind())"' % (domain_file, problem_file),
            # command="/pddls/%s /pddls/%s --alias seq-sad-fd-autotune" % (domain_file, problem_file),
            # command='/pddls/%s /pddls/%s --search "lazy_greedy([ff()], preferred=[ff()])"' % (domain_file, problem_file),
            mem_limit="16g",
            working_dir="/pddls",
            entrypoint="/workspace/downward/fast-downward.py",
            volumes={docker_host_dir.absolute().as_posix(): {"bind": "/pddls", "mode": "rw"}},
            detach=True,
        )
        try:
            result = container.wait()
            success = 0 <= result["StatusCode"] <= 3  # https://www.fast-downward.org/latest/documentation/exit-codes/
            response = container.logs().decode().strip()
        except ConnectionError:
            logger.warning("Connection error while waiting for the planner to finish (timeout=%f)." % timeout)
            success = False
            response = "Timeout while waiting for the planner to finish."

        container.remove(force=True)

        sas_plan_file = next(docker_host_dir.glob("sas_plan*"), None)
        if success:
            assert sas_plan_file is not None
            sas_plan = SasPlan.from_string(sas_plan_file.read_text())
        else:
            sas_plan = None

        return sas_plan, response

    def translate(self, domain: str, problem: str) -> str:
        """
        Translates the PDDL domain and problem files into a SAS plan using Fast Downward.
        """
        with tempfile.TemporaryDirectory() as temp_dir:
            temp_dir = Path(temp_dir)
            domain_file = temp_dir / "domain.pddl"
            problem_file = temp_dir / "problem.pddl"

            domain_file.write_text(domain)
            problem_file.write_text(problem)

            container = self.client.containers.run(
                image="aibasel/downward",
                command="--sas-file /pddls/sas-file /pddls/domain.pddl /pddls/problem.pddl",
                working_dir="/pddls",
                # entrypoint="/workspace/downward/fast-downward.py",
                volumes={temp_dir.absolute().as_posix(): {"bind": "/pddls", "mode": "rw"}},
                detach=True,
            )
            result = container.wait()
            assert result["StatusCode"] == 36
            sas = (temp_dir / "sas-file").read_text()

        return sas
