import os
import re

import pandas as pd
import pyperclip

import agent.templates as templates
from agent.config import args, logger, reasoning_model
from agent.utils import AgentException, agentic_loop
from models import Conversation, Response


def parse_exploit_strategy(conversation):
    pattern = r"<STRATEGY>(.*?)</STRATEGY>"
    match = re.search(pattern, conversation.responses[-1].text, re.DOTALL)

    if match:
        return f"<STRATEGY>\n{match.group(1).strip()}\n</STRATEGY>"
    else:
        raise AgentException("ParseError", "Could not parse exploit strategy")


if __name__ == "__main__":
    df = pd.read_csv(os.path.join(args.path, "cwe_db.csv"), index_col=0)

    cwe_id = int(input("CWE Number\n>>> "))
    print(f"CWE-{cwe_id}: {df.loc[cwe_id, 'name']}")
    print("-" * 80)

    exploit_code = pyperclip.paste().strip()
    print(exploit_code)

    print("-" * 80)

    prompt = templates.generate_fewshot_strategies_from_code.format(
        vulnerability=f"CWE-{cwe_id}: {df.loc[cwe_id, 'name']}",
        format_specifications=templates.exploit_strategy_format,
        exploit_code=exploit_code,
    )

    conversation = Conversation().add_message(Response(role="user", text=prompt))
    response = reasoning_model.generate(
        conversation,
        temperature=0,
        purpose="generate_fewshot_exploit_strategies: generating fewshot exploit strategy",
    )
    conversation.add_message(response)

    extracted_strategy = agentic_loop(
        conversation,
        parse_exploit_strategy,
        args.N_RETRIES,
        "parsing extracted strategy",
        templates.exploit_strategy_format,
    )

    print("-" * 80)
    print(response.text)
    print("-" * 80)
    x = input("OK? (y/N/h) ")
    if x == "y":
        dir_path = os.path.join(args.path, "fewshot_sec")
        os.makedirs(dir_path, exist_ok=True)
        file_path = os.path.join(dir_path, f"{cwe_id}.txt")

        with open(file_path, "a") as file:
            file.write(f"<CODE>\n{exploit_code}\n</CODE>\n\n")
            file.write(response.text + "\n\n")
        logger.info(f"Added fewshot example to {file_path}")
    elif x == "h":  # cwe-specific header code
        dir_path = os.path.join(args.path, "fewshot_sec")
        os.makedirs(dir_path, exist_ok=True)
        file_path = os.path.join(dir_path, f"{cwe_id}.txt")
        with open(file_path, "a") as file:
            file.write(f"<HEADER>\n{exploit_code}\n</HEADER>\n\n")
    else:
        logger.info("Discarded fewshot example")
