import argparse
import yaml
import sys
from pathlib import Path
from datetime import datetime
import requests
import json
import os
from dotenv import load_dotenv
# from nyuctf.dataset import CTFDataset
from nyuctf_multiagent.challenge import CTFChallenge

from nyuctf_multiagent.environment import CTFEnvironment
from nyuctf_multiagent.backends import MODELS, Role
from nyuctf_multiagent.prompting import PromptManager
from nyuctf_multiagent.agent import PlannerExecutorSystem, PlannerAgent, ExecutorAgent, AutoPromptAgent
from nyuctf_multiagent.logging import logger
from nyuctf_multiagent.utils import APIKeys, load_common_options, get_log_filename, load_config
from nyuctf_multiagent.config import Config
from nyuctf_multiagent.utils import TeeOutput
parser = argparse.ArgumentParser(description="Multi-agent Planner-Executor LLM for CTF solving")

# Loads the dataset and container related common options into parser
load_common_options(parser)
parser.add_argument("--logdir", default="logs_dcipher", type=str, help="Log directory")
parser.add_argument("--config", default=None, help="YAML config for the planner-executor multiagent. If not provided, it picks one automatically based on challenge cateogory.")
parser.add_argument("--challenge_ids", nargs="+", default=None, help="Challenge id that you should solve.")
parser.add_argument("--category", default=None, help="CTF category of the Challenge. Among rev, pwn, web, crypto, misc, forensics")
parser.add_argument("--all_unsolved", default=False, nargs="?", const=True, help="Try to solve all unsolved problems when this is set")


# Config overriding options
parser.add_argument("--planner-model", default=None, help="Planner model to use (overrides config)")
parser.add_argument("--executor-model", default=None, help="Executor model to use (overrides config)")
parser.add_argument("--autoprompter-model", default=None, help="AutoPrompt model to use (overrides config)")
parser.add_argument("--max-cost", default=0.0, type=float, help="Max cost in $ (overrides config)")
parser.add_argument("--enable-autoprompt", action="store_true", help="Init prompt message auto generated, else use generic base prompt")

args = parser.parse_args()

logger.set(quiet=args.quiet, debug=args.debug)
if args.all_unsolved:
    response = requests.get("http://localhost:8000/challenges")
    data = json.loads(response.text)
    challenge_ids = [item["id"] for item in data]
    print(challenge_ids)
else:
    challenge_ids = args.challenge_ids
original_stdout = sys.stdout
tee =  TeeOutput("./baseline-log/testfile.txt")
for challenge_id in challenge_ids:
    sys.stdout = original_stdout
    tee.close()
    problem_name=''
    response = requests.get("http://localhost:8000/challenges")
    data = json.loads(response.text)
    print(data)
    for item in data:
        if str(item["id"]) == challenge_id:
            problem_name = item["name"]
            print("Challenge Name: ", problem_name)
    
    if problem_name:
        # Remove characters that cannot be used as file names from the problem name
        safe_problem_name = "".join(c for c in problem_name if c.isalnum() or c in ('-', '_')).rstrip()
        base_filename = f"{args.executor_model}_{safe_problem_name}_agent_communication"
    else:
        base_filename = f"{args.executor_model}_agent_communication"
    print(base_filename)
    # If the file already exists, create it by adding a number
    counter = 1
    
    agent_log_dir = Path("./agent-log")
    agent_log_dir.mkdir(parents=True, exist_ok=True)
    
    log_filename = f"./agent-log/"+ f"{base_filename}_{counter}.log"
    
    while Path(log_filename).exists():
        log_filename = f"./agent-log/"+ f"{base_filename}_{counter}.log"
        counter += 1
    load_dotenv(dotenv_path="../CTFd-MCP/.env")
    Flag_format = os.getenv("FLAG_FORMAT")
    print("FLAG FORMAT: ",Flag_format)
    challenge = CTFChallenge(challenge_id, "http://host.docker.internal:8000/",Flag_format, args.category)

    print("Logging to ", log_filename)
    
    sys.stdout = TeeOutput(log_filename)
    original_stdout = sys.stdout
    
    
    logdir = Path(args.logdir)
        
    logdir.mkdir(parents=True, exist_ok=True)
    logfile = get_log_filename(args, challenge_id)
    
    keys = APIKeys(args.keys)
    environment = CTFEnvironment(challenge, args.container_image, args.container_network)

    if args.config:
        config_f = Path(args.config)
    else:
        config_d = Path(sys.argv[0]).parent / "configs" / "dcipher"
        config_f = config_d / f"{args.category}_planner_executor.yaml"
    
    print(f"Using config: {str(config_f)}")
    config = load_config(config_f, args=args)

    config.experiment.enable_autoprompt = True if args.enable_autoprompt else config.experiment.enable_autoprompt

    autoprompter_backend_cls = MODELS[config.autoprompter.model]
    autoprompter_backend = autoprompter_backend_cls(Role.AUTOPROMPTER, config.autoprompter.model,
                                        environment.get_toolset(config.autoprompter.toolset),
                                        keys[autoprompter_backend_cls.NAME.upper()], config)
    autoprompter_prompter = PromptManager(config_f.parent / config.autoprompter.prompt, challenge, environment)
    autoprompter = AutoPromptAgent(environment, challenge, autoprompter_prompter,
                        autoprompter_backend, max_rounds=config.autoprompter.max_rounds)

    if config.experiment.enable_autoprompt:
        autoprompter.enable_autoprompt()

    planner_backend_cls = MODELS[config.planner.model]
    planner_backend = planner_backend_cls(Role.PLANNER, config.planner.model,
                                        environment.get_toolset(config.planner.toolset),
                                        keys[planner_backend_cls.NAME.upper()], config)
    planner_prompter = PromptManager(config_f.parent / config.planner.prompt, challenge, environment)
    planner = PlannerAgent(environment, challenge, planner_prompter,
                        planner_backend, max_rounds=config.planner.max_rounds)

    executor_backend_cls = MODELS[config.executor.model]
    executor_backend = executor_backend_cls(Role.EXECUTOR, config.executor.model,
                                            environment.get_toolset(config.executor.toolset),
                                            keys[executor_backend_cls.NAME.upper()], config)
    executor_prompter = PromptManager(config_f.parent / config.executor.prompt, challenge, environment)
    executor = ExecutorAgent(environment, challenge, executor_prompter,
                            executor_backend, max_rounds=config.executor.max_rounds)
    executor.conversation.len_observations = config.executor.len_observations

    with PlannerExecutorSystem(environment, challenge, autoprompter, planner, executor, max_cost=config.experiment.max_cost, logfile=logfile) as multiagent:
        multiagent.run()
