"""
A dedicated helper to manage templates and prompt building.
"""

from typing import Union

class ZeroPrompter(object):
    __slots__ = ("_verbose")

    def __init__(self, verbose: bool = False):
        self._verbose = verbose

        if self._verbose:
            print(f"Without using prompt template!")

    def generate_prompt(
            self,
            instruction: str,
            input: Union[None, str] = None,
            label: Union[None, str] = None,
    ) -> str:
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        if instruction[-1] == '.':
            instruction = instruction[:-1] + ':'
        if instruction[-1] not in ['.', ':', '?', '!']:
            instruction = instruction + ':'
        instruction += ' '

        if input:
            if input[-1] not in ['.', ':', '?', '!']:
                input = input + '.'
            res = instruction + input
        else:
            res = instruction
        if label:
            res = f"{res} {label}"
        if self._verbose:
            print(res)
        return res

    def get_response(self, output: str) -> str:
        return output.strip()

class OpenBookQAPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by OpenBookQA.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a task and four possible options. Your goal is to select the option that best achieves the given task.\n\nTask: {question}\n\Options:\n{options}\n\nPlease respond with either \"A\" \"B\", \"C\"or \"D\" to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template OpenBookQA: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        question_stem = item["question_stem"]
        choices = item["choices"]["text"]
        answerKey = item["answerKey"]

        label_map = ["A", "B", "C", "D"]
        options = ""
        for i, choice in enumerate(choices):
            options += f"{label_map[i]}. {choice}\n"

        prompt = self.template["prompt_input"].format(
            question=question_stem, options=options, input=""
        )

        if self._verbose:
            print(prompt)
        return {"prompt": prompt, "answer": answerKey}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class GSM8KPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by GSM8K.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a math task, please complete this task.\n\n### Input:\n{question}\n\n### Response:\n",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template GSM8K: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        question = item["question"]
        answer = item["answer"]

        prompt = self.template["prompt_input"].format(
            question=question
        )

        if self._verbose:
            print(prompt)
        return {"prompt": prompt, "answer": answer}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class PIQAPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by PIQA.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a task and two possible solutions. Your goal is to select the solution that best achieves the given task.\n\nTask: {goal}\n\nSolutions:\n  1. {sol1}\n  2. {sol2}\n\nPlease respond with either \"1\" or \"2\" to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template PIQA: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        goal = item["goal"]
        sol1 = item["sol1"]
        sol2 = item["sol2"]
        label = item["label"]

        prompt = self.template["prompt_input"].format(
            goal=goal, sol1=sol1, sol2=sol2, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": str(label + 1)}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class MMLUPrompter(object):

    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by MMLU.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a task and four possible options. Your goal is to select the option that best achieves the given task.\n\nTask: {question}\n\Options:\n{options}\n\nPlease respond with either \"A\" \"B\", \"C\"or \"D\" to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template MMLU: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.

        question = item["question"]
        choices = item["choices"]
        answerKey = item["answer"]

        label_map = ["A", "B", "C", "D"]
        options = ""
        for i, choice in enumerate(choices):
            options += f"{label_map[i]}. {choice}\n"

        prompt = self.template["prompt_input"].format(
            question=question, options=options, input=""
        )

        if self._verbose:
            print(prompt)
        return {"prompt": prompt, "answer": label_map[answerKey]}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class BoolQPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by BoolQ.",
            "prompt_input": "Below is an instruction that describes a yes/no question, paired with an input that provides further context. Write a response that answers the question with either 'yes' or 'no'.\n\n### Instruction:\nYou will be presented with a passage and a boolean question. Your goal is to determine if the answer to the question is 'yes' or 'no'.\n\nQuestion: {question}\n\nPassage:\n{passage}\n\nPlease respond with either 'yes' or 'no'.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template BoolQ: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        question = item["question"]
        passage = item["passage"]
        label = "yes" if item["label"] == 1 else "no"

        prompt = self.template["prompt_input"].format(
            question=question, passage=passage, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": label}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class SIQAPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by SIQA.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a social scenario and a related question. Your goal is to select the option that best answers the question based on the given context.\n\nContext: {context}\n\nQuestion: {question}\n\nOptions:\nA.{answerA}\nB.{answerB}\nC.{answerC}\n\nPlease respond with either \"A\", \"B\", or \"C\", to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template SIQA: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        context = item["context"]
        question = item["question"]
        answerA = item["answerA"]
        answerB = item["answerB"]
        answerC = item["answerC"]
        label = item["label"]

        label_map = ["A", "B", "C"]
        prompt = self.template["prompt_input"].format(
            context=context, question=question, answerA=answerA, answerB=answerB, answerC=answerC, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": label_map[int(label) - 1]}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class HellaswagPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by Hellaswag.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a context and a question. Your goal is to select the option that best completes the context based on common sense and reasoning.\n\nContext: {context}\n\nOptions:\n{options}\n\nPlease respond with either \"A\", \"B\", \"C\", or \"D\" to indicate the most appropriate completion.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template Hellaswag: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        activity_label = item["activity_label"]
        ctx = item["ctx"]
        endings = item["endings"]
        label = item["label"]

        label_map = ["A", "B", "C", "D"]
        options = ""
        for i in range(len(endings)):
            options += f"{label_map[i]}. {endings[i]}\n"

        prompt = self.template["prompt_input"].format(
            context=activity_label + ctx, options=options, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": label_map[int(label)]}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class MedQAPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by MedQA.",
            "prompt_input": "Below is an instruction that describes a medical-related task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a medical question and multiple possible answers. Your goal is to select the option that best answers the question based on medical knowledge and best practices.\n\nQuestion: {question}\n\nOptions:\n{options}\n\nPlease respond with either \"A\", \"B\", \"C\", or \"D\" to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }

        if self._verbose:
            print(f"Using prompt template MedQA: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        sent1 = item["sent1"]
        sent2 = item["sent2"]
        ending0 = item["ending0"]
        ending1 = item["ending1"]
        ending2 = item["ending2"]
        ending3 = item["ending3"]
        label = item["label"]

        label_map = ["A", "B", "C", "D"]
        options = f"A.{ending0}\nB.{ending1}\nC.{ending2}\nD.{ending3}"

        prompt = self.template["prompt_input"].format(
            question=sent1 + sent2, options=options, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": label_map[label]}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class AlpacaPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by Alpaca-LoRA.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\n{instruction}\n\n### Input:\n{input}\n\n### Response:\n",
            "response_split": "### Response:"
        }

        if self._verbose:
            print(f"Using prompt template Alpaca: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        instruction = item["instruction"]
        input = item["input"]
        output = item["output"]

        prompt = self.template["prompt_input"].format(
            instruction=instruction, input=input
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": output}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class SamSumPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by Alpaca-LoRA.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nSummarize the following conversation.\n\n### Input:\n{input}\n\n### Response:\n",
            "response_split": "### Response:"
        }

        if self._verbose:
            print(f"Using prompt template SamSum: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        dialogue = item["dialogue"]
        output = item["summary"]
        prompt = self.template["prompt_input"].format(
            input=dialogue
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": output}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class ArcPrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by ARC.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further question. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a question. Your goal is to select the option that best completes the context based on common sense and reasoning.\n\nQuestion: {question}\n\nOptions:\n{options}\n\nPlease respond with either \"A\", \"B\", \"C\", or \"D\" to indicate the most appropriate completion.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template Hellaswag: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        question = item["question"]
        answerKey = item["answerKey"]
        choices = item["choices"]["text"]

        label_map = item["choices"]["label"]
        options = ""
        for i in range(len(label_map)):
            options += f"{label_map[i]}. {choices[i]}\n"

        prompt = self.template["prompt_input"].format(
            question=question, options=options, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": answerKey}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class WinograndePrompter(object):
    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "Template used by Winogrande.",
            "prompt_input": "Below is an instruction that describes a task, paired with an input that provides further context. Write a response that appropriately completes the request.\n\n### Instruction:\nYou will be presented with a sentence with a blank and two possible solutions. Your goal is to select the solution that best fills blank.\n\nSentence: {sentence}\n\nSolutions:\n  1. {option1}\n  2. {option2}\n\nPlease respond with either \"1\" or \"2\" to indicate the most appropriate solution.\n\n### Input:\n{input}\n\n### Response:",
            "response_split": "### Response:"
        }
        if self._verbose:
            print(f"Using prompt template Winogrande: {self.template['description']}")

    def generate_prompt(self, item):
        # returns the full prompt from instruction and optional input
        # if a label (=response, =output) is provided, it's also appended.
        sentence = item["sentence"]
        option1 = item["option1"]
        option2 = item["option2"]
        answer = item["answer"]

        prompt = self.template["prompt_input"].format(
            sentence=sentence, option1=option1, option2=option2, input=""
        )

        if self._verbose:
            print(prompt)

        return {"prompt": prompt, "answer": answer}

    def get_response(self, output: str) -> str:
        return output.split(self.template["response_split"])[1].strip()

class Prompter(object):
    __slots__ = ("_verbose")

    def __init__(self, verbose: bool = False):
        self._verbose = verbose
        self.template = {
            "description": "",
            "prompt_input": "",
            "prompt_no_input": "",
            "response_split": ""
        }
        if self._verbose:
            print(f"Using prompt template alpaca: {self.template['description']}")
        if self._verbose:
            print(f"Without using prompt template!")

    def generate_prompt(self, *args, **kwargs):
        return self.prompter.generate_prompt(*args, **kwargs)

    def get_response(self, *args, **kwargs):
        return self.prompter.get_response(*args, **kwargs)
