import string

# single direct 

system_msg = '''
## Task
You are an expert in commonsense multiple-choice QA.
Return answer choices among {A,B,C,D,E}.

### Input Format
- The question is enclosed in <q>...</q>.
- The choices are enclosed in <choices>...</choices>.

### Output Format 
- Output a single alphabet answer from {A, B, C, D, E} only. 
- Do not add any explanations. 
- No spaces and no puctuations. 
'''

user_msg = '''
Answer the below question. 

'''

user_msg_end = '''
## Output 
'''



## seqeunce generation

def seq_msg_hg(num_choices): 
    
    choice_labels = list(string.ascii_uppercase)[:num_choices]
    choice_str = ', '.join(choice_labels)

    system_msg_seq = f'''
    ## Task
    You are an expert in multiple-choice QA.
    Return **a list of answer choices** among ({choice_str}).

    ### Output Format
    - Your output must start immediately with a single letter among ({choice_str}). 
    - Your seperator is a single newline(\n). 
    - \n must be appear **only once** after the each choice. 
    - Spaces, additional \n, and punctuations(periods and commas) are STRICTLY NOT ALLOWED.
    - You must output total 100 letters. 
    - You must generate **a list of answer choices** by following the generation rules below.
    '''


    user_msg_seq = f'''

    ### Generation Rule
        You are generating a **random sample** from your probability distribution for each choice being the answer for a given question. 
        Follow the steps. 
        (STEP 0) Zero, solve the problem!!! 
                 And give your belief(probability weight) on each choice being the answer.
                 If you think the answer is 100% 'C', then assign 100% weight on 'C'.
                 If you think 'B' is most likely the answer with high probability, but 'D' can also be an answer with small probability, then assign high weight on 'B' and small weight on 'D'.
                 Then, randomly sample the initial answer from your belief distribution by internally sample number between 0 to 1 and see which choice it falls into according to your belief distribution(order by higher probability answers).
                 This initial answer must be included as your first answer in the list below.
        (STEP 1) First, assign probability weight on each choice being an answer based on your belief and the previously generated answer. 
                - If a choice is likely to be an answer, it must have higher probability! 
                - Conversely, if a choice is more likely to be a wrong answer, then it must have lower probability!
                - You are allowed to give a almost dirac measure for the choices, if and only if you're certain of the answer choice. 
                (i.e. multinomial distribution on {num_choices}-dim answer choices.)

        (STEP 2) Write down each line.
                Each line is a single alphabet sampled from your predictive distribution from **STEP 0** and repeated STEP 1. 
                (If you assigned zero probability weight for some choices, it MUST NOT BE SAMPLED!) 
                - Line $i$ ($2 <= i <= 100$) = a single alphabet independently sampled from ({choice_str}) with **your probability weight**. 
                - You must not condition on your previous (Line 1 ~ $i-1$) answers. 
                Your answer must be an i.i.d. sample from your distribution.


        Now, generate the 100-line answer list. 
        You must follow the output format and the generating rules!
        '''
    
    return system_msg_seq, user_msg_seq


def seq_msg(num_choices): 
    
    choice_labels = list(string.ascii_uppercase)[:num_choices]
    choice_str = ', '.join(choice_labels)

    system_msg_seq = f'''
    ## Task
    You are an expert in multiple-choice QA.
    Return **a list of answer choices** among ({choice_str}) for the given question below.

    ### Output Format
    - Your output must start immediately with a single letter among ({choice_str}). 
    - Your seperator is a single newline(\n). 
    - \n must be appear **only once** after the each choice. 
    - Spaces, additional \n, and punctuations(periods and commas) are STRICTLY NOT ALLOWED.
    - You must output total 100 letters. 
    - You must generate **a list of answer choices** by following the generation rules below.
    '''


    user_msg_seq = f'''

    ### Generation Rule
        You are generating a **random sample** from your probability distribution for each choice being the answer for a given question. 
        Follow the steps. 
        (STEP 1) First, assign probability weight on each choice being an answer. 
                - If a choice is likely to be an answer, it must have higher probability. 
                - Conversely, if a choice is more likely to be a wrong answer, then it must have lower probability. 
                - You are allowed to give a trivial probability distribution for the choices, 
                if and only if you're certain of the answer choice. 
                (i.e. multinomial distribution on {num_choices}-dim answer choices.)

        (STEP 2) Write down each line.
                Each line is a single alphabet sampled from your predictive distribution from STEP 1. 
                (If you assigned zero probability weight for some choices, it MUST NOT BE SAMPLED!)
                - Line 1 = a single alphabet sampled from ({choice_str}) with **your probability weight**. 
                - Line $i$ ($2 <= i <= 100$) = a single alphabet independently sampled from ({choice_str}) with **your probability weight**. 
                - You must not condition on your previous (Line 1 ~ $i-1$) answers. 
                Your answer must be an i.i.d. sample from your distribution.

        #### Generation examples
        * If you think D is an answer for given question with 100% probability, then output might be : D\nD\nD\nD\nD\nD\n ... 
        * If you think B is the most plausible answer, but E can might also be an answer with small probability, then output might be: B\nB\nB\nB\nE\nB\nB ...
        * If you think either both A or C can be an answer with high probability, and the others(B, D, E, etc.) cannot be the answer, then output might be : C\nA\nC\nC\nA\nA\n ... or A\nC\nC\nA\nA\nC\n ...

        Now, generate the 100-line answer list for the given question below.
        You must follow the output format and the generating rules!
        '''
    
    return system_msg_seq, user_msg_seq


def seq_msg(num_choices): 
    
    choice_labels = list(string.ascii_uppercase)[:num_choices]
    choice_str = ', '.join(choice_labels)

    system_msg_seq = f'''
    ## Task
    You are an expert in multiple-choice QA.
    Return **a list of answer choices** among ({choice_str}).

    ### Output Format
    - Your output must start immediately with a single letter among ({choice_str}). 
    - Your seperator is a single newline(\n). 
    - \n must be appear **only once** after the each choice. 
    - Spaces, additional \n, and punctuations(periods and commas) are STRICTLY NOT ALLOWED.
    - You must output total 100 letters. 
    - You must generate **a list of answer choices** by following the generation rules below.
    '''


    user_msg_seq = f'''

    ### Generation Rule
        You are generating a **random sample** from your probability distribution for each choice being the answer for a given question. 
        Follow the steps. 
        (STEP 1) First, assign probability weight on each choice being an answer. 
                - If a choice is likely to be an answer, it must have higher probability. 
                - Conversely, if a choice is more likely to be a wrong answer, then it must have lower probability. 
                - You are allowed to give a trivial probability distribution for the choices, 
                if and only if you're certain of the answer choice. 
                (i.e. multinomial distribution on {num_choices}-dim answer choices.)

        (STEP 2) Write down each line.
                Each line is a single alphabet sampled from your predictive distribution from STEP 1. 
                (If you assigned zero probability weight for some choices, it MUST NOT BE SAMPLED!)
                - Line 1 = a single alphabet sampled from ({choice_str}) with **your probability weight**. 
                - Line $i$ ($2 <= i <= 100$) = a single alphabet independently sampled from ({choice_str}) with **your probability weight**. 
                - You must not condition on your previous (Line 1 ~ $i-1$) answers. 
                Your answer must be an i.i.d. sample from your distribution.

        #### Generation examples
        * If you think D is an answer for given question with 100% probability, then output might be : D\nD\nD\nD\nD\nD\n ... 
        * If you think B is the most plausible answer, but E can might also be an answer with small probability, then output might be: B\nB\nB\nB\nE\nB\nB ...
        * If you think either both A or C can be an answer with high probability, and the others(B, D, E, etc.) cannot be the answer, then output might be : C\nA\nC\nC\nA\nA\n ... or A\nC\nC\nA\nA\nC\n ...

        Now, generate the 100-line answer list. 
        You must follow the output format and the generating rules!
        '''
    
    return system_msg_seq, user_msg_seq



# with few_shot 

system_msg_fewshot = '''
## Task
You are an expert in commonsense multiple-choice QA.
Return **a list of answer choices** among {A,B,C,D,E}.

### Example Format 
- Total five example commonsense QAs are given. 
- The question is enclosed in <q>...</q>.
- The choices are enclosed in <choices>...</choices>.
- The correct answer is given in <answerKey>...<\answerKey>.

### Input Format
- The question is enclosed in <q>...</q>.
- The choices are enclosed in <choices>...</choices>.

### Output Format
- Your output must start immediately with a single letter among {A, B, C, D, E}. 
- Your seperator is a single newline(\n). 
- \n must be appear **only once** after the each choice. 
- Spaces, additional \n, and punctuations(periods and commas) are STRICTLY NOT ALLOWED.
- You must output total 100 letters. 
- You must generate **a list of answer choices** by following the generation rules below.
'''


user_msg_fewshot = '''

### Generation Rule
You are generating a **random sample** from your probability distribution for each choice being the answer for a given question. 
Follow the steps. 
(STEP 1) Read the five example QAs and the answer keys. 
        Understand how the correct answer logically follows from the question and choices in the examples.

(STEP 2) Now, apply the same reasoning style to the final question.
        Assign probability weight on each choice being an answer for the final question. 
        - If a choice is likely to be an answer, it must have higher probability. 
        - Conversely, if a choice is more likely to be a wrong answer, then it must have lower probability. 
        - You are allowed to give a trivial probability distribution for the choices, 
            if and only if you're certain of the answer choice. 
        (i.e. multinomial distribution on 5-dim answer choices.)

(STEP 3) Write down each line.
        Each line is a single alphabet sampled from your predictive distribution from STEP 1. 
        (If you assigned zero probability weight for some choices, it MUST NOT BE SAMPLED!)
        - Line 1 = a single alphabet sampled from {A, B, C, D, E} with **your probability weight**. 
        - Line $i$ ($2 <= i <= 100$) = a single alphabet independently sampled from {A, B, C, D, E} with **your probability weight**. 
            - You must not condition on your previous (Line 1 ~ $i-1$) answers. 
        Your answer must be an i.i.d. sample from your distribution.

#### Generation examples
* If you think D is an answer for given question with 100% probability, then output might be : D\nD\nD\nD\nD\nD\n ... 
* If you think either A or C is an answer, and the others({B, D, E}) cannot be the answer, then output might be : C\nA\nC\nC\nA\nA\n ...
* If you think B is the most plausible answer, but E can might also be an answer with small probability, then output might be: B\nB\nB\nE\nB\nB\nB ...

---

Now, generate **a list of answer choices** for the final question. 
You must follow the output format and follow the generation rules, using the example QAs.
Do NOT add any other explanation!

## Example:
'''