import unittest

from src.tool_ace_prompt_functions import (
    tool_ace_parse_conversation_role,
    tool_ace_get_prompt_for_strict_api,
    tool_ace_get_prompt_for_summarize,
    tool_ace_get_prompt_for_template_summarize,
)
from src.utility import user_role, Record

record_dict = {
    "id": "5a925fb7-42c7-40c2-9e37-2e76bc596d5f",
    "data_set": "tool_ace",
    "pre_api": [
        "You are an expert in composing functions. You are given a question and a set of possible functions.",
        "Based on the question, you will need to make one or more function/tool calls to achieve the purpose.",
        "If none of the function can be used, point it out. If the given question lacks the parameters required by the function,",
        "also point it out.",
        "Here is a list of functions in JSON format that you can invoke:",
    ],
    "api_def": [
        '[{"name": "Get All Servers", "description": "Retrieve a list of all Minecraft servers", "parameters": {"type": "dict", "properties": {"limit": {"description": "The maximum number of servers to return", "type": "int"}}, "required": ["limit"]}, "required": null}, {"name": "Get Spell", "description": "Retrieve information about a D&D 5E spell, including its school, level, and description.", "parameters": {"type": "dict", "properties": {"spell_name": {"description": "The name of the spell to retrieve information about.", "type": "string"}}, "required": ["spell_name"]}, "required": null}, {"name": "Full Item Details", "description": "Retrieves detailed information about a game, including prices, discounts, descriptions, PEGI/ESRB ratings, DLCs, system requirements, screenshots, and more.", "parameters": {"type": "dict", "properties": {"language": {"description": "The language to retrieve the game data in.", "type": "string"}, "countryCode": {"description": "The country code to retrieve the game data for.", "type": "string"}, "appid": {"description": "The App ID of the game to retrieve details for.", "type": "float"}}, "required": ["language", "countryCode", "appid"]}, "required": null}, {"name": "Heroes List API", "description": "This API provides a list of heroes from the Dota2 game, including their IDs, names, images, complexity levels, and attributes.", "parameters": {"type": "dict", "properties": {}, "required": []}, "required": null}].'
    ],
    "conversation": [
        "user: Could you help me find some popular Minecraft servers to join? Let's set the limit to five servers."
    ],
    "ending": [],
    "output": "[Get All Servers(limit=5)]",
    "post_api": [
        "Should you decide to return the function call(s).",
        "Put it in the format of [func1(params_name=params_value, params_name2=params_value2...), func2(params)]",
        "NO other text MUST be included.",
    ],
    "api_call": {
        "api_call_status": "tool_call",
        "api_calls": [
            {"api_name": "Get All Servers", "params": {"limit": 5}, "responses": None}
        ],
    },
    "template_output": [
        "Call the `Get All Servers` API with following parameters: `limit` as `5`"
    ],
}
content_1_1 = 'You are an expert in composing functions. You are given a question and a set of possible functions.\nBased on the question, you will need to make one or more function/tool calls to achieve the purpose.\nIf none of the function can be used, point it out. If the given question lacks the parameters required by the function,\nalso point it out.\nHere is a list of functions in JSON format that you can invoke:\n[{"name": "Get All Servers", "description": "Retrieve a list of all Minecraft servers", "parameters": {"type": "dict", "properties": {"limit": {"description": "The maximum number of servers to return", "type": "int"}}, "required": ["limit"]}, "required": null}, {"name": "Get Spell", "description": "Retrieve information about a D&D 5E spell, including its school, level, and description.", "parameters": {"type": "dict", "properties": {"spell_name": {"description": "The name of the spell to retrieve information about.", "type": "string"}}, "required": ["spell_name"]}, "required": null}, {"name": "Full Item Details", "description": "Retrieves detailed information about a game, including prices, discounts, descriptions, PEGI/ESRB ratings, DLCs, system requirements, screenshots, and more.", "parameters": {"type": "dict", "properties": {"language": {"description": "The language to retrieve the game data in.", "type": "string"}, "countryCode": {"description": "The country code to retrieve the game data for.", "type": "string"}, "appid": {"description": "The App ID of the game to retrieve details for.", "type": "float"}}, "required": ["language", "countryCode", "appid"]}, "required": null}, {"name": "Heroes List API", "description": "This API provides a list of heroes from the Dota2 game, including their IDs, names, images, complexity levels, and attributes.", "parameters": {"type": "dict", "properties": {}, "required": []}, "required": null}].\nShould you decide to return the function call(s).\nPut it in the format of [func1(params_name=params_value, params_name2=params_value2...), func2(params)]\nNO other text MUST be included.'
content_2_1 = 'You are an expert in composing functions. You are given a question and a set of possible functions.\nDo sum\nIf none of the function can be used, point it out. If the given question lacks the parameters required by the function,\nalso point it out.\nHere is a list of functions in JSON format that you can invoke:\n[{"name": "Get All Servers", "description": "Retrieve a list of all Minecraft servers", "parameters": {"type": "dict", "properties": {"limit": {"description": "The maximum number of servers to return", "type": "int"}}, "required": ["limit"]}, "required": null}, {"name": "Get Spell", "description": "Retrieve information about a D&D 5E spell, including its school, level, and description.", "parameters": {"type": "dict", "properties": {"spell_name": {"description": "The name of the spell to retrieve information about.", "type": "string"}}, "required": ["spell_name"]}, "required": null}, {"name": "Full Item Details", "description": "Retrieves detailed information about a game, including prices, discounts, descriptions, PEGI/ESRB ratings, DLCs, system requirements, screenshots, and more.", "parameters": {"type": "dict", "properties": {"language": {"description": "The language to retrieve the game data in.", "type": "string"}, "countryCode": {"description": "The country code to retrieve the game data for.", "type": "string"}, "appid": {"description": "The App ID of the game to retrieve details for.", "type": "float"}}, "required": ["language", "countryCode", "appid"]}, "required": null}, {"name": "Heroes List API", "description": "This API provides a list of heroes from the Dota2 game, including their IDs, names, images, complexity levels, and attributes.", "parameters": {"type": "dict", "properties": {}, "required": []}, "required": null}].\nPost sum'
content_3_1 = 'You are an expert in composing functions. You are given a question and a set of possible functions.\nBased on the question, summarize the next action to take.\nSummarized result should contain all necessary information defined in corresponding functions.\nSummarize in the format like following: \nIf there\'s no parameter: Call the `function name` API with no parameter\nIf there\'s one or more parameters: Call the `function name` API with following parameters: `parameter1 name` as `parameter1 value`, ...\nIf there\'s multiple function call, use \'\n\' to connect the results, for example: \'Call the `function name` API with ...\nCall the `function name` API with ...\'\nIf none of the function can be used, point it out. If the given question lacks the parameters required by the function,\nalso point it out.\nHere is a list of functions in JSON format that you can invoke:\n[{"name": "Get All Servers", "description": "Retrieve a list of all Minecraft servers", "parameters": {"type": "dict", "properties": {"limit": {"description": "The maximum number of servers to return", "type": "int"}}, "required": ["limit"]}, "required": null}, {"name": "Get Spell", "description": "Retrieve information about a D&D 5E spell, including its school, level, and description.", "parameters": {"type": "dict", "properties": {"spell_name": {"description": "The name of the spell to retrieve information about.", "type": "string"}}, "required": ["spell_name"]}, "required": null}, {"name": "Full Item Details", "description": "Retrieves detailed information about a game, including prices, discounts, descriptions, PEGI/ESRB ratings, DLCs, system requirements, screenshots, and more.", "parameters": {"type": "dict", "properties": {"language": {"description": "The language to retrieve the game data in.", "type": "string"}, "countryCode": {"description": "The country code to retrieve the game data for.", "type": "string"}, "appid": {"description": "The App ID of the game to retrieve details for.", "type": "float"}}, "required": ["language", "countryCode", "appid"]}, "required": null}, {"name": "Heroes List API", "description": "This API provides a list of heroes from the Dota2 game, including their IDs, names, images, complexity levels, and attributes.", "parameters": {"type": "dict", "properties": {}, "required": []}, "required": null}].'
utterance = "Could you help me find some popular Minecraft servers to join? Let's set the limit to five servers."


class ToolAcePromptFunctionsTestCases(unittest.TestCase):
    def test_tool_ace_parse_conversation_role(self):
        input_str = "user: hello"
        result = tool_ace_parse_conversation_role(input_str)
        self.assertEqual(result, {"role": user_role, "content": "hello"})

        input_str = "hello"
        self.assertRaises(RuntimeError, tool_ace_parse_conversation_role, input_str)

    def test_tool_ace_get_prompt_for_strict_api(self):
        record = Record.from_dict(record_dict)
        prompt = tool_ace_get_prompt_for_strict_api(record, True)
        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_1_1)

        self.assertEqual(prompt[1]["role"], "user")
        self.assertEqual(prompt[1]["content"], utterance)

        prompt = tool_ace_get_prompt_for_strict_api(record, False)
        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_1_1)

        self.assertEqual(prompt[1]["content"], f"user: {utterance}")

    def test_tool_ace_get_prompt_for_summarize(self):
        record = Record.from_dict(record_dict)
        prompt = tool_ace_get_prompt_for_summarize(
            record, True, ["Do sum"], ["Post sum"]
        )
        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_2_1)

        self.assertEqual(prompt[1]["role"], "user")
        self.assertEqual(prompt[1]["content"], utterance)

        prompt = tool_ace_get_prompt_for_summarize(
            record, False, ["Do sum"], ["Post sum"]
        )

        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_2_1)

        self.assertEqual(prompt[1]["content"], f"user: {utterance}")

    def test_tool_ace_get_prompt_for_template_summarize(self):
        record = Record.from_dict(record_dict)
        prompt = tool_ace_get_prompt_for_template_summarize(record, True)
        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_3_1)

        self.assertEqual(prompt[1]["role"], "user")
        self.assertEqual(prompt[1]["content"], utterance)

        prompt = tool_ace_get_prompt_for_template_summarize(record, False)

        self.assertEqual(len(prompt), 2)

        self.assertEqual(prompt[0]["role"], "system")
        self.assertEqual(prompt[0]["content"], content_3_1)

        self.assertEqual(prompt[1]["content"], f"user: {utterance}")


if __name__ == "__main__":
    unittest.main()
