import argparse
from pathlib import Path
import argparse
import os
from pathlib import Path
import shutil
import subprocess

from dotenv import load_dotenv
from llm_utils.textgen_api.textgen_api import TextGenApi
from natsort import natsorted
from tp_lodge.utils.pddl_lib_utils import copy_domain_w_args


def _out_dir(args):
    ipc_root_dir = Path(__file__).parent.parent
    connection = TextGenApi.default(args.llm).connections.connections[0]
    return (
        ipc_root_dir
        / "results"
        / args.domain
        / connection.model_dir
        / "lionel"
        / "no-docstrings"
    )


def _lionel_dir():
    ipc_root_dir = Path(__file__).parent.parent
    return ipc_root_dir.parent.parent / "3rdparty/llm-operators"


def run_lionel(args):
    load_dotenv()

    exp_dir = _out_dir(args)
    data_dir = Path(__file__).parent.parent / "data" / args.domain

    if exp_dir.is_dir():
        print(f"Experiment directory already exists: {exp_dir}")
        return

    assert not args.no_docstrings

    env = os.environ.copy()

    env = {
        **env,
        "OPENAI_MODEL": args.llm,
    }
    lionel_root = _lionel_dir()
    lionel_python_path = lionel_root / ".pixi/envs/default/bin/python"

    local_data_dir = lionel_root / "data/dataset" / args.domain
    if local_data_dir.is_dir():
        shutil.rmtree(local_data_dir, ignore_errors=True)
    shutil.copytree(data_dir, local_data_dir)


    response = subprocess.run(
        [
            lionel_python_path,
            lionel_root / "main.py",
            "--dataset_name",
            args.domain,
            "--pddl_domain_name",
            args.domain,
            "--dataset_pddl_directory",
            data_dir,
            "--experiment_name",
            args.domain,
            "--supervision_name",
            "supervision",
            "--dataset_fraction",
            "1.0",
            "--verbose",
            "--train_iterations",
            "2",
            "--output_directory",
            exp_dir,
            "--external_plan_supervision",
            f"{lionel_root}/data/dataset/supervision-NLgoals-operators.json",
            "--external_operator_supervision",
            f"{lionel_root}/data/dataset/crafting-world-crafting-only-operator-supervision_",
            "--external_operator_sample_with_prompt",
            # "--resume"
        ],
        env=env,
        cwd=lionel_root,
    )
    if response.returncode not in [0, 120]:
        print("Error running lionel main.py:")
        print(response.returncode)
        if response.stdout is not None:
            print(response.stdout.decode("utf-8"))
        if response.stderr is not None:
            print(response.stderr.decode("utf-8"))
        raise RuntimeError("Failed to run lionel main.py")

def post_process_lionel(args):
    exp_dir = _out_dir(args)

    out_dir = exp_dir / args.domain
    
    last_iteration = natsorted([d for d in out_dir.iterdir() if d.is_dir()])[-1]
    print(last_iteration)
    assert isinstance(last_iteration, Path)

    shutil.copy2(last_iteration / "textgen-usage.json", exp_dir / "textgen-usage.json")

    from pddl.parser.domain import DomainParser
    gt_domain = DomainParser()((last_iteration / "domain.pddl").read_text())

    pred_domain = copy_domain_w_args(gt_domain, predicates=set(), actions=[])

    (exp_dir / "domain.pddl").write_text(str(pred_domain))


def main(args):
    run_lionel(args)
    post_process_lionel(args)


if __name__ == "__main__":
    argparser = argparse.ArgumentParser()
    argparser.add_argument("--domain", type=str, required=True)
    argparser.add_argument("--llm", type=str, required=True)
    argparser.add_argument("--no-docstrings", action="store_true", help="Use domain without docstrings")
    main(argparser.parse_args())
