import os
import zipfile
from tqdm import tqdm
import time

import re
from typing import List, Tuple
import json

import subprocess
from pathlib import Path
import sys
from time import sleep
import shutil

from start_service import start_services, kill_service_on_port
from sandbox import create_docker_compose_file, start_docker_containers, stop_docker_containers, free_docker_port
from eval_db import eval_db
from compute_acc import compute_acc
from eval_backend import run_info_gathering_agents, get_test_data, run_backend_testing_agents
from webvoyager import run_single_task
from backend_compute_acc import backend_compute_acc

from initialize_db import initialize_db, find_first_sql

import sys
sys.path.append(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))

from dotenv import load_dotenv
load_dotenv()

from utils import dump_database, to_jsonable

DB_PORT = 5432
DB_CONFIG = dict(
    db_host="localhost",
    db_port=DB_PORT,
    db_username="myappuser",
    db_password="myapppassword",
    db_name="myapp",
)

def load_json(in_file):
    with open(in_file, "r", encoding="utf-8") as f:
        data = json.load(f)
    return data


def save_json(data, out_file):
    with open(out_file, "w", encoding="utf-8") as f:
        json.dump(data, f)


def load_jsonl(in_file):
    datas = []
    with open(in_file, "r", encoding="utf-8") as f:
        for line in tqdm(f):
            datas.append(json.loads(line))
    return datas


def save_jsonl(datas, out_file, mode="w"):
    with open(out_file, mode, encoding="utf-8") as f:
        for data in tqdm(datas):
            f.write(json.dumps(data, ensure_ascii=False) + "\n")


def get_shell_start(app_paths, output_root):
    commands = {}
    for app_path in tqdm(app_paths):
        commands[os.path.basename(app_path)] = {"shell_actions": ["npm run install:all", "npm install"], "last_start_action": "npm run dev"}
        if "frontend" in os.listdir(app_path):
            commands[os.path.basename(app_path)]["shell_actions"].append("cd frontend && npm install")
        if "backend" in os.listdir(app_path):
            commands[os.path.basename(app_path)]["shell_actions"].append("cd backend && npm install")
    save_json(commands, os.path.join(output_root, "commands.json"))
    return commands


ui_prompt_template = """

Task: {task}

Expected Result: {expected_result}

Instructions:
- Attempt the task as a user would, using the UI elements available.
- Make multiple attempts if needed to try and achieve the expected result.
- Observe whether the expected result is fully, partially, or not at all achieved.
- IMPORTANT: You can at most interact with the website 15 times. If the limit is reached, directly output your answer.
- If prompted for a username, password, or email in the process of testing, enter "superadmin@example.com", "admin123", and "superadmin@example.com", respectively.
- If login is unsuccessful, try to register a new account first.

At the end of your testing, answer only with one of the following:
- YES: if the expected result was fully achieved.
- NO: if the expected result could not be achieved at all.
- PARTIAL: if only some aspects of the expected result were achieved.

"""


def create_tasks_test(test_file, ports, tasks_file):
    datas = load_json(test_file)
    tasks = []
    for idx, data in tqdm(enumerate(datas)):
        app = data["id"]
        if app not in ports.keys():
            continue
        for ui_idx, ui_instruct in enumerate(data["ui_instruct"]):
            instruction = ui_prompt_template.format(task=ui_instruct["task"], expected_result=ui_instruct["expected_result"])
            tasks.append({
                "web_name": data["id"],
                "id": f"{app}_{ui_idx}",
                "ques": instruction,
                "web": f"http://localhost:{ports[app]}/",
                "expected_result": ui_instruct["expected_result"],
                "task": ui_instruct["task"]
            })
    save_jsonl(tasks, tasks_file)


def run_webvoyager(input_dir, db_dir=None):
    from webvoyager.run_webvoyager import run_single_task
    import concurrent.futures
    import json
    from pathlib import Path

    input_dir = Path(input_dir)
    tasks_file = input_dir / "tasks_test_with_answer.jsonl"
    output_dir = input_dir / "results"
    download_dir = input_dir / "downloads"
    
    # Create output and download directories
    output_dir.mkdir(exist_ok=True)
    download_dir.mkdir(exist_ok=True)
    
    # Load tasks
    tasks = []
    with open(tasks_file, "r", encoding="utf-8") as f:
        for line in f:
            tasks.append(json.loads(line))
    
    # Define arguments for run_single_task
    args_dict = {
        "test_file": str(tasks_file),
        "max_iter": 15,
        "api_key": "sk-mah6FUel7jrB3lNj8c3cnqUGeKy1ovL5DAD1GFge92C7Fe864c8646B1B9DaB6C20a10A896",
        "api_model": os.environ["VLM_MODEL"],
        "output_dir": str(output_dir),
        "seed": 42,
        "max_attached_imgs": 3,
        "temperature": 1.0,
        "download_dir": str(download_dir),
        "text_only": False,
        "headless": True,
        "save_accessibility_tree": False,
        "force_device_scale": False,
        "window_width": 1600,
        "window_height": 1200,
        "fix_box_color": True,
        "num_workers": 8
    }
    
    # Run tasks in parallel using ProcessPoolExecutor as in the original
    with concurrent.futures.ProcessPoolExecutor(max_workers=args_dict["num_workers"]) as executor:
        futures = [executor.submit(run_single_task, task, args_dict, db_dir) for task in tasks]
        for future in concurrent.futures.as_completed(futures):
            try:
                future.result()  # This will raise an exception if the task failed
            except Exception as exc:
                print(f"Task generated an exception: {exc}")


def copy_db(orig_db_dir, db_tmp):
    if os.path.exists(db_tmp):
        shutil.rmtree(db_tmp)          # old leftovers
    if os.path.exists(orig_db_dir):
        # Copy as fast as possible: use reflink on capable filesystems,
        # fallback to regular copy otherwise.
        try:
            subprocess.run(
                ["cp", "-a", "--reflink=auto", orig_db_dir, db_tmp],
                check=True
            )
        except subprocess.CalledProcessError:
            shutil.copytree(orig_db_dir, db_tmp)
    else:
        os.makedirs(db_tmp, exist_ok=True)  # empty cluster
 

def has_db(db_config):
    if db_config is None:
        return False
    if not isinstance(db_config, dict):
        return False
    for k, v in db_config.items():
        if v is not None:
            return True


def main():
    from argparse import ArgumentParser
    parser = ArgumentParser()
    parser.add_argument("--in_dir", type=str)
    parser.add_argument("--log_dir", type=str)
    args = parser.parse_args()
    in_dir = args.in_dir
    test_file = r"data\WebGen-Bench_test-db-backend.json"
    test_datas = load_json(test_file)
    app_paths = [os.path.join(in_dir, data["id"]) for data in test_datas]

    output_root = args.log_dir
    tasks_file = os.path.join(output_root, "tasks_test_with_answer.jsonl")
    log_file = os.path.join(output_root, "log.jsonl")
    log_datas = []
    if os.path.isfile(log_file):
        log_datas = load_jsonl(log_file)

    processed_app_paths = [data["app_path"] for data in log_datas]

    filtered_app_paths = []
    for app_path in tqdm(app_paths):
        if app_path not in processed_app_paths:
            filtered_app_paths.append(app_path)
    app_paths = filtered_app_paths

    # gather info for backend testing
    run_info_gathering_agents(args.in_dir, args.log_dir, test_datas)

    batch_size = 1
    compose_path = os.path.join(args.log_dir, "docker-compose.yml")
    for i in range(0, len(app_paths), batch_size):
        batch_app_paths = app_paths[i:i + batch_size]
        working_dir = batch_app_paths[0]
        log_dir = os.path.join(args.log_dir, os.path.basename(working_dir))

        info_path = os.path.join(args.log_dir, "info_gathering", os.path.basename(log_dir), "info_result.json")
        info_data = {}
        if os.path.isfile(info_path):
            info_data = load_json(info_path)

        db_dir = os.path.join(log_dir, "db")
        kill_service_on_port(DB_PORT)
        create_docker_compose_file(working_dir, log_dir, compose_path, db_dir, db_port=DB_PORT)
        stop_docker_containers(compose_path)
        free_docker_port(DB_PORT)
        start_docker_containers(compose_path)
        
        sleep(15)
        sql_file = find_first_sql(working_dir)
        if sql_file is not None:
            print(f"found sql file: {sql_file}")
            initialize_db(sql_file)

        commands = get_shell_start(batch_app_paths, args.in_dir)
        ports = start_services(args.in_dir, commands)
        print(ports)

        db_exists = True

        create_tasks_test(test_file, ports, tasks_file)
        sleep(30)
        
        test_data = get_test_data(test_datas, os.path.basename(working_dir))
        run_backend_testing_agents(args.in_dir, args.log_dir, test_data, db_dir, db_exists=db_exists)

        # test frontend
        run_webvoyager(output_root, db_dir)
        
        try:
            dump = dump_database(DB_CONFIG, limit=5, connect_timeout=60)
            dump = to_jsonable(dump)
            save_json(dump, os.path.join(log_dir, "db_dump.json"))
        except Exception as e:
            print(f"Failed to dump database for {working_dir}: {e}")
        
        subprocess.run("pm2 delete all", shell=True)
        # if db_dir is not None:
            # shutil.rmtree(db_dir, ignore_errors=True)
        
        curr_log_datas = [{"app_path": app_path} for app_path in batch_app_paths]
        save_jsonl(curr_log_datas, log_file, mode="a")

    compute_acc(args.log_dir)
    backend_compute_acc(args.log_dir)
    eval_db(args.in_dir, args.log_dir)


if __name__ == "__main__":
    main()