import os
import re

import pandas as pd

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_fewshot_examples(cwe_id: int) -> tuple[str, str]:
    """Parses few-shot examples for a given CWE ID."""
    dir_path = os.path.join(args.path, "fewshot_sec")
    file_path = os.path.join(dir_path, f"{cwe_id}.txt")
    assert os.path.exists(
        file_path
    ), f"Could not parse fewshot examples for CWE-{cwe_id}"
    with open(file_path, "r") as file:
        strategies = file.read()
    pattern = r"<STRATEGY>(.*?)</STRATEGY>"
    matches = re.findall(pattern, strategies, re.DOTALL)
    parsed_strategies = (
        "<STRATEGY>\n"
        + "\n</STRATEGY>\n\n<STRATEGY>\n".join([match.strip() for match in matches])
        + "\n</STRATEGY>"
    )

    pattern = r"<HEADER>(.*?)</HEADER>"
    match = re.search(pattern, strategies, re.DOTALL)
    header = ""
    if match:
        header = match.group(1).strip()

    return header, parsed_strategies


def exploit_specific_instructions(cwe_id: int) -> str:
    """Generates exploit-specific instructions for a given CWE ID."""
    exploit_specific_header, strategy_template = parse_fewshot_examples(cwe_id)

    return templates.exploit_specific_instructions.format(
        exploit_specific_header=exploit_specific_header,
        cwe_id=cwe_id,
        strategy_template=strategy_template,
    )


def parse_exploit_strategy(conversation: Conversation) -> str:
    """Parses the exploit strategy from the 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>"
    elif "DISCARD" in conversation.responses[-1].text:
        # if args.debug:
        #     conversation.add_message(Response(role="user", text="Why?\n"))
        #     response = reasoning_model.generate(
        #         conversation,
        #         temperature=0,
        #         purpose="generate_exploit_strategies: determine why exploit strategy is discarded",
        #     )
        #     logger.info(f"Discarding because {response.text}")
        return ""
    else:
        raise AgentException("ParseError", "Could not parse exploit strategy")


def generate_exploit_strategy(
    cwe_id: int, cwe_approaches: list[str], scenario: dict, conversation: Conversation
) -> str:
    """Generates an exploit strategy for a given CWE."""
    logger.info(f"generating exploit strategy for CWE-{cwe_id}")
    df = pd.read_csv(os.path.join(args.path, "cwe_db.csv"), index_col=0)

    prompt = templates.generate_exploit_strategy.format(
        scenario_title=scenario["title"],
        scenario_description=scenario["description"],
        scenario_openapi=scenario["schema"],
        cwe=f"CWE-{cwe_id}: {df.loc[cwe_id, 'name']}",
        cwe_approaches=cwe_approaches,
        sec_tool_signatures=templates.sec_tool_signatures,
        cwe_specific_instructions="",
        format_specifications=templates.exploit_strategy_format,
    )

    conversation.add_message(Response(role="user", text=prompt))
    response = reasoning_model.generate(
        conversation,
        temperature=0,
        purpose=f"generate_exploit_strategies: generating exploit strategy for CWE-{cwe_id}",
    )
    conversation.add_message(response)

    return agentic_loop(
        conversation,
        parse_exploit_strategy,
        args.N_RETRIES,
        "parsing exploit strategy",
        templates.exploit_strategy_format,
    )


def compliance_exploit_strategy(
    exploit_strategy: str, scenario: dict, conversation: Conversation
) -> str:
    """Ensures the exploit strategy complies with the scenario specification."""
    if not exploit_strategy:
        return exploit_strategy

    logger.info("ensuring compliance of exploit strategy")

    prompt = templates.compliance_exploit_strategy.format(
        scenario_text_spec=scenario["text_spec"],
        format_specifications=templates.exploit_strategy_format,
    )

    conversation = conversation.add_message(Response(role="user", text=prompt))
    response = reasoning_model.generate(
        conversation,
        temperature=0,
        purpose="generate_exploit_strategies: ensuring compliance of exploit strategy",
    )
    conversation.add_message(response)

    # parse strategy
    return agentic_loop(
        conversation,
        parse_exploit_strategy,
        args.N_RETRIES,
        "parsing exploit strategy",
        templates.exploit_strategy_format,
    )
