import json
import logging
from typing import Optional

from src.api_bank_data_conversion import api_bank_parse_api_request
from src.utility import user_role, ai_role, Record, system_role, tool_role

logger = logging.getLogger(__name__)

prompt_prefix_sum = [
    "Summarize the next action to take based on conversation history.",
    "If the action can be fulfilled with API in API descriptions, "
    "summarized result should contain all necessary information defined in corresponding API descriptions.",
    "Summarize in the format like following: ",
    "If there's no parameter: Call the `API Name` API with no parameter",
    "If there's one or more parameters: Call the `API Name` API with following parameters: `parameter1 name` as `parameter1 value`, ...",
]


def api_bank_parse_conversation_role(prompt: str) -> dict:
    """
    Convert a string based prompt to a dict with role information. e.g. `"User: send a message"`
    will be converted to `{"role": "user", "content": "send a message"}`.
    :param prompt: A string based prompt.
    :return: Dict with role information.
    """
    if prompt.startswith("User: "):
        content = prompt.removeprefix("User: ")
        return {"role": user_role, "content": content}
    elif prompt.startswith("AI: "):
        content = prompt.removeprefix("AI: ")
        return {"role": ai_role, "content": content}
    elif prompt.startswith("API-Request: "):
        return {"role": ai_role, "content": prompt}
    else:
        raise RuntimeError(f"Unexpected line: {prompt}")


def api_bank_remove_extra_instructions(prompts: list[str]) -> list[str]:
    """
    Remove unnecessary lines of prompts.
    :param prompts: Original prompts.
    :return: Updated prompts.
    """
    result = []
    for line in prompts:
        if (
            line.startswith("Input:")
            or line.startswith("User: User's utterance")
            or line.startswith("AI: AI's response")
        ):
            continue
        result.append(line)
    return result


def api_bank_get_prompt_for_strict_api(
    record: Record, prepare_for_chat_template: bool
) -> list[dict]:
    """
    Create prompt for asking LLM to generated schema based strict API strings.
    :param record: Record containing prompts.
    :param prepare_for_chat_template: Process the prompts so it can be used by tokenizer.apply_chat_template.
    :return: List of prompt strings.
    """
    if prepare_for_chat_template:
        pre_api = api_bank_remove_extra_instructions(record.pre_api)
    else:
        pre_api = record.pre_api
    system_start = {
        "role": system_role,
        "content": "\n".join(pre_api + record.api_def),
    }
    system_end = {"role": system_role, "content": "\n".join(record.ending)}
    if prepare_for_chat_template:
        conversation = []
        for line in record.conversation:
            conversation.append(api_bank_parse_conversation_role(line))
    else:
        conversation = [{"content": x} for x in record.conversation]
    prompts = [system_start] + conversation + [system_end]
    return prompts


def api_bank_get_prompt_for_summarize(
    record: Record,
    prepare_for_chat_template: bool,
    extra_instructions: Optional[list[str]] = None,
) -> list[dict]:
    """
    Create prompt for asking LLM to do summarize.
    :param record: Record containing prompts.
    :param extra_instructions: Extract instruction to add to prompt.
    :param prepare_for_chat_template: Process the prompts so it can be used by tokenizer.apply_chat_template.
    :return: List of prompt strings.
    """
    pre_api, api_def, conversation, ending = (
        record.pre_api,
        record.api_def,
        record.conversation,
        record.ending,
    )

    if prepare_for_chat_template:
        pre_api = api_bank_remove_extra_instructions(pre_api)

    assert (
        pre_api[0]
        == "Generate an API request in the format of [ApiName(key1='value1', key2='value2', ...)] based on the previous dialogue context."
    ), f"Unexpected line:{pre_api[0]}"
    new_lines = []
    for line in pre_api[1:]:
        if line == "API-Request: [ApiName(key1='value1', key2='value2', ...)]":
            line = "A sentence describing next action with all necessary information to call API"
        new_lines.append(line)
    if extra_instructions is not None:
        new_pre_api = extra_instructions + new_lines
    else:
        new_pre_api = new_lines
    assert ending[0] == "Generate API Request:", f"Unexpected line:{ending[0]}"
    if prepare_for_chat_template:
        conversation = []
        for line in record.conversation:
            conversation.append(api_bank_parse_conversation_role(line))
    else:
        conversation = [{"content": x} for x in record.conversation]

    system_start = {"role": system_role, "content": "\n".join(new_pre_api + api_def)}
    system_end = {"role": system_role, "content": "Summarize the next action:"}
    prompts = [system_start] + conversation + [system_end]
    return prompts


def api_bank_get_prompt_for_template_summarize(
    record: Record, prepare_for_chat_template: bool
) -> list[dict]:
    """
    Given a Record, generate template based API string.
    :param record: Record containing prompts.
    :param prepare_for_chat_template: Process the prompts so it can be used by tokenizer.apply_chat_template.
    :return: List of prompt strings.
    """
    return api_bank_get_prompt_for_summarize(
        record=record,
        prepare_for_chat_template=prepare_for_chat_template,
        extra_instructions=prompt_prefix_sum,
    )


def api_bank_parse_openai_tool_calling(api_request: str, tool_index: int) -> list[dict]:
    """
    Parse the API call and convert it to OpenAI format.
    :param api_request: String containing API request.
    :param tool_index: Current tool index.
    :return: A tool calling structure compatible to OpenAI format.
    """
    fds = api_request.removeprefix("API-Request: ").split("->")
    assert len(fds) == 2, f"Invalid line:{api_request}"
    api_call = api_bank_parse_api_request(fds[0])
    result = []
    tool_call = {
        "role": ai_role,
        "tool_calls": [
            {
                "id": f"tool_{tool_index}",
                "type": "function",
                "function": {
                    "name": api_call.api_name,
                    "arguments": json.dumps(api_call.params),
                },
            }
        ],
    }
    result.append(tool_call)
    tool_call_result = {
        "role": tool_role,
        "tool_call_id": f"tool_{tool_index}",
        "content": fds[1],
    }
    result.append(tool_call_result)
    return result
