from datetime import datetime
import socket
import subprocess
import time
import json
import os
import argparse
import shutil
import sys
import yaml
from pathlib import Path
from nyuctf_baseline.utils import TeeOutput
import requests


from nyuctf_baseline.challenge import CTFChallenge

from nyuctf_baseline.ctflogging import status
from nyuctf_baseline.backends import Backend, OpenAIBackend, AnthropicBackend, VLLMBackend
from nyuctf_baseline.formatters import Formatter
from nyuctf_baseline.prompts.prompts import PromptManager
from nyuctf_baseline.environment import CTFEnvironment
from nyuctf_baseline.conversation import CTFConversation
from nyuctf_baseline.utils import get_log_filename

def main():
    parser = argparse.ArgumentParser(
        description="Use an LLM to solve a CTF challenge",
        formatter_class=argparse.ArgumentDefaultsHelpFormatter
    )
    model_list = []
    for b in Backend.registry.values():
        model_list += b.get_models()
    model_list = list(set(model_list))

    script_dir = Path(__file__).parent.resolve()

    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")


    parser.add_argument("-s", "--split", default="development", choices=["test", "development"], help="Dataset split to select. Only used when --dataset not provided.")
    parser.add_argument("-c", "--config", help="Config file to run the experiment")

    parser.add_argument("-q", "--quiet", action="store_true", help="don't print messages to the console")
    parser.add_argument("-d", "--debug", action="store_true", help="print debug messages")
    parser.add_argument("-M", "--model", help="the model to use (default is backend-specific)", choices=model_list)
    parser.add_argument("-C", "--container-image", default="ctfenv", help="the Docker image to use for the CTF environment")
    parser.add_argument("-N", "--network", default="ctfnet", help="the Docker network to use for the CTF environment")
    parser.add_argument("--api-key", default=None, help="API key to use when calling the model")
    parser.add_argument("--api-endpoint", default=None, help="API endpoint URL to use when calling the model")
    parser.add_argument("--backend", default="openai", choices=Backend.registry.keys(), help="model backend to use")
    parser.add_argument("--formatter", default="xml", choices=Formatter.registry.keys(), help="prompt formatter to use")
    parser.add_argument("--prompt-set", default="default", help="set of prompts to use")
    # TODO add back hints functionality
    parser.add_argument("--hints", default=[], nargs="+", help="list of hints to provide")
    parser.add_argument("--disable-markdown", default=False, action="store_true", help="don't render Markdown formatting in messages")
    parser.add_argument("-m", "--max-rounds", type=int, default=1000, help="maximum number of rounds to run")
    parser.add_argument("--max-cost", type=float, default=10, help="maximum cost of the conversation to run")

    # Log directory options
    parser.add_argument("--skip-exist", action="store_true", help="Skip existing logs and experiments")
    parser.add_argument("-L", "--logdir", default=str(script_dir / "logs_baseline"), help="log directory to write the log")
    parser.add_argument("-n", "--name", help="Experiment name (creates subdir in logdir)")
    parser.add_argument("-i", "--index", help="Round index of the experiment (creates subdir in logdir)")

    args = parser.parse_args()
    if args.all_unsolved:
        response = requests.get("http://localhost:8000/unsolved")
        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()
        config = None
        if args.config:
            try:
                with open(args.config, "r") as c:
                    config = yaml.safe_load(c)
            except FileNotFoundError:
                pass

        if config:
            config_parameter = config.get("parameter", {})
            config_experiment = config.get("experiment", {})
            config_demostration = config.get("demostration", {})

            args.max_rounds = config_parameter.get("max_rounds", args.max_rounds)
            args.backend = config_parameter.get("backend", args.backend)
            args.model = config_parameter.get("model", args.model)
            args.max_cost = config_parameter.get("max_cost", args.max_cost)
            args.name = config_experiment.get("name", args.name)
            args.debug = config_experiment.get("debug", args.debug)
            args.skip_exist = config_experiment.get("skip_exist", args.skip_exist)
            args.hints = config_demostration.get("hints", [])

        status.set(quiet=args.quiet, debug=args.debug, disable_markdown=args.disable_markdown)

        challenge = CTFChallenge(challenge_id, "http://host.docker.internal:8000/","FLAG{somestring}", args.category)

        logdir = Path(args.logdir)
        
        logdir.mkdir(parents=True, exist_ok=True)
        logfile = logdir / f"{get_log_filename(args, challenge_id)}.json"
        
        if logfile.exists() and args.skip_exist:
            status.print(f"[red bold]Challenge log {logfile} exists; skipping[/red bold]", markup=True)
            exit()
            
        log_filename = f"./baseline-log/agent_communication_{challenge_id}_{datetime.now().strftime('%Y%m%d_%H%M%S')}.log"
        tee =  TeeOutput(log_filename)
        original_stdout = sys.stdout
        sys.stdout = tee
        
        environment = CTFEnvironment(challenge, args.container_image, args.network)
        prompt_manager = PromptManager(prompt_set=args.prompt_set, config=config)

        if args.backend == "openai":
            backend = OpenAIBackend(
                            prompt_manager.system_message(challenge),
                            prompt_manager.hints_message(),
                            environment.available_tools,
                            model=args.model,
                            api_key=args.api_key,
                            args=args
                        )
        elif args.backend ==  "anthropic":
            backend = AnthropicBackend(
                            prompt_manager.system_message(challenge),
                            prompt_manager.hints_message(),
                            environment.available_tools,
                            prompt_manager,
                            model=args.model,
                            api_key=args.api_key,
                            args=args
                        )
        elif args.backend == "vllm":
            backend = VLLMBackend(
                            prompt_manager.system_message(challenge),
                            prompt_manager.hints_message(),
                            environment.available_tools,
                            prompt_manager,
                            model=args.model,
                            api_key=args.api_key,
                            api_endpoint=args.api_endpoint,
                            formatter=args.formatter,
                            args=args
                        )
        
        with CTFConversation(environment, challenge, prompt_manager, backend, logfile, max_rounds=args.max_rounds, max_cost=args.max_cost, args=args) as convo:
            convo.run()

if __name__ == "__main__":
    main()
