import logging
import re
from pathlib import Path
from tempfile import TemporaryDirectory
from typing import Optional, Tuple

import docker
from docker.errors import DockerException

logger = logging.getLogger(__name__)


class AIValidator:

    def __init__(self):
        self.client = docker.from_env()

    def _is_docker_running(self) -> bool:
        try:
            self.client.ping()  # Sends a request to Docker to check if it's alive
            return True
        except DockerException:
            return False

    def validate_plan_executes_successfully(self, domain: str, problem: str, plan: str):
        response, success = self.validate(domain=domain, problem=problem, plan=plan, options="-v")
        successful = "Plan executed successfully" in response
        if not successful:
            assert not success
            if "Bad plan description" in response:
                match = re.search("Type problem in action([\w\W]+)Bad plan", response)
            else:
                match = re.search("Plan Repair Advice:([\w\W]+)Failed plans:", response)
            assert match is not None
            repair_advice = match.group(1)
        else:
            repair_advice = None
        return successful, repair_advice

    def validate(
        self, domain: str, problem: Optional[str], plan: Optional[str], options: Optional[str]
    ) -> Tuple[str, bool]:
        with TemporaryDirectory() as dir:
            dir = Path(dir)
            domain_file = dir / "domain.pddl"
            domain_file.write_text(domain)
            options = options or ""
            cmd = options + " /pddls/domain.pddl "
            if problem is not None:
                problem_file = dir / "problem.pddl"
                problem_file.write_text(problem)
                cmd += "/pddls/problem.pddl "
            else:
                problem_file = ""

            if plan is not None:
                assert problem is not None
                actions_file = dir / "actions"
                plan = "\n".join([l for l in plan.splitlines() if not l.strip().startswith(";")]).strip()
                actions_file.write_text(plan)
                cmd += "/pddls/actions"
            else:
                actions_file = ""

            assert self._is_docker_running()
            container = self.client.containers.run(
                image="val",
                command=cmd,
                volumes={dir.absolute().as_posix(): {"bind": "/pddls", "mode": "rw"}},
                detach=True,
            )
            result = container.wait()
            response = container.logs().decode().strip()
            if plan is None:
                success = result["StatusCode"] == 0
            else:
                success = "Failed plans:" not in response
                if success:
                    # can have StatusCode 0 if syntax of plan is invalid
                    assert result["StatusCode"] == 0, response
            container.remove()

        if not success and "Plan Repair Advice" in response:
            # repair_advice = re.findall(r".*Plan Repair Advice:(.*)Failed plans:.*", response)
            repair_advice = re.findall(r"Plan Repair Advice:\n([\w\W]*?)\nFailed plans:", response)[0]
            lines = repair_advice.strip().splitlines()
            assert len(lines) >= 2

            if "has an unsatisfied precondition" in lines[0] or "The goal is not satisfied" in lines[0]:
                predicates = [re.findall(r"(\([\w\-]+[\w ]*\))", line) for line in lines[1:]]
                predicates_new = [re.findall(r"(\([\w\-]+[\w ]*\)) to (false|true)", line) for line in lines[1:]]
                assert len(predicates) == len(predicates_new)
                predicates = [
                    p[0][0] if p[0][1] == "true" else ("(not %s)" % p[0][0]) for p in predicates_new if len(p) > 0
                ]
                predicate = ", ".join(predicates)

                response = response.replace(
                    repair_advice, ("Predicate that leads to unsatisfied precondition: %s\n" % predicate)
                )

        return response, success
