import argparse
import json
from pathlib import Path
import re
from typing import Optional
from llm_utils.textgen_api.textgen_api_connections import TextGenLLMConnections
from pddl.core import Action, Domain, Requirements
from pddl.logic.base import Variable
from tp_lodge.utils.pddl_parse_utils import parse_formula, parse_predicate as pp
from tp_lodge.utils.pddl_domain_syntax import _parentheses_groups


def parse_action(data, action_name: str):
    ps = []
    for parameter in data["parameters"]:
        p = parameter.split(":")[0]
        name, type = p.split(" - ")
        ps.append(Variable(name.replace("?", ""), [type]))

    idx = re.search(r"\(", data["preconditions"]).start()
    precond_contents = list(_parentheses_groups(data["preconditions"][idx:]))
    assert len(precond_contents) == 1
    preconditions_l = parse_formula(precond_contents[0])

    idx = re.search(r"\(", data["effects"]).start()
    eff_contents = list(_parentheses_groups(data["effects"][idx:]))
    assert len(eff_contents) == 1
    # effects_l = []
    # for eff_content in eff_contents:
    #     effects_l.append(parse_formula(eff_content))
    effects_l = parse_formula(eff_contents[0])
    return Action(
        name=action_name.lower().replace(" ", "-"),
        parameters=ps,
        precondition=preconditions_l,
        effect=effects_l,
    )


def main(args, domain: str = "logistics", out_dir: Optional[Path] = None):
    engine = TextGenLLMConnections.default(args.engine).available_connections[0].model  # 'gpt-4' or 'gpt-3.5-turbo'
    curr_dir = Path(__file__).parent

    assert out_dir is not None, "Output directory must be specified."
    result_dir = out_dir
    # result_log_dir = f'results/{domain}/{prompt_version}/corrected'
    expr_result_dir = result_dir / "corrected"

    # actions
    actions_file = expr_result_dir / f"{engine}_final_pddl.json"
    actions_data = json.loads(actions_file.read_text())

    actions = []
    for a_name, action_dict in actions_data.items():
        action = parse_action(action_dict, a_name)
        actions.append(action)

    # predicates
    predicates_file = expr_result_dir / f"{engine}_final_predicates.txt"
    predicates_data = predicates_file.read_text().splitlines()
    predicates = list(map(lambda data: pp(re.match(r".*(\(.*\)).*:(.*)", data).group(1)), predicates_data))

    # types
    types_file = curr_dir / f"prompts/{domain}/hierarchy_requirements.json"
    types_data = json.loads(types_file.read_text())
    types = {n: "object" for n in types_data["hierarchy"].keys()}
    types["smallReceptacle"] = "object"

    pddl_domain = Domain(
        name=domain,
        actions=actions,
        requirements=Requirements.adl_requirements(),
        predicates=predicates,
        types=types,
    )

    (result_dir.parent / f"{engine}_final_domain.pddl").write_text(str(pddl_domain))


if __name__ == "__main__":
    parser = argparse.ArgumentParser(description="Construct action models using LLMs.")
    parser.add_argument(
        "--engine", type=str, help="Specify the LLM engine to use (e.g., gpt4o-mini, gpt-3.5-turbo).", required=True
    )
    parser.add_argument("--idx", type=int, required=True)
    main(parser.parse_args())
