developer_message = """
You are a helpful assistant to generate corresponding short description about a scenario given some conditions. You will be provided with a short sentence to describe a scenario as well as some factors (variables) in the scenario. You should generate some short sentences which are slightly different from the originial sentence and describe the situation where the scenario is the same but the corresponding variables take given different value from original situation. You should generate sentences that are fully self-contained and interpretable on their own, without requiring reference to the original sentence."""

from textwrap import dedent

from prompt_generate.util import FactorWithoutType, Factor

def factors_to_str(factors: list[FactorWithoutType | Factor]) -> str:
    """
    Convert the list of factors to string, each factor is represented in json format, with indentation of 2 spaces. Join all factors with two new lines.
    """
    return "\n\n".join([fac.model_dump_json(indent=2) for fac in factors])

def get_user_message(scenario:str, factors: list[FactorWithoutType], non_roots: list[FactorWithoutType], num_sent: int, together: bool, contain_results: bool, rules: list[str]) -> str:
    
    if not together:
        raise NotImplementedError("The seperate generation is not implemented yet.")
    
    num_factors = len(factors)
    num_comb = 2**len(factors)
    
    user_message = f"""\
    The scenario is: {scenario}

    In this scenario, there are some factors are considered as (binary) variables and you should generate new description to change the original scenario to meet the corresponding value. The description tells the detailed definition of the variables and how to determine the Boolean value from a piece of video. The explanation tells how the variable affects the scenario.
    
    Factors:
    ```
    {factors_to_str(factors)}
    ```
    
    """
    
    if contain_results and together:
        user_about_results = f"""\
        There are also some results variables which are the outcome of the above factors with their expected value: 
        {factors_to_str(non_roots)}
        In each possible composition of factor values, you should first induce the corresponding value of the results variables and then generate the sentences. 
        In these sentences, please explicitly and clearly express the corresponding value of both the factors and the results variables in the generated sentences. The rules of the results: \n{"\n".join(rules)}\n"""
    elif contain_results and not together:
        user_about_results = f"""\
        There are also some results variables which are the outcome of the above factors with their expected value:
        {factors_to_str(non_roots)}
        Please explicitly and clearly express the corresponding value of both the factors and the results variables in the generated sentences.""" 
    else: # not contain_results: we need to explicitly prompt LLM to not mention the results variables
        user_about_results = f"""\
        There are also some results variables which are the outcome of the above factors: 
        {factors_to_str(non_roots)}
        You should be very careful to make sure that the values of these variables should not be mentioned in the generated sentences."""

    user_message_together = f"""\
    Each variable can take value as "yes" or "no" independently so that there are 2**{num_factors}={num_comb} compositions. 
    You should generate {num_sent} sentences for each yes/no composition for these variables. """

    user_message_seperate = f"""\
    You should generate {num_sent} sentences for the given changed scenario."""
    
    requirement = """\
    Please make sure (1) each sentence meet and explicitly express the corresponding value of variables and (2) the generated sentences as diverse as possible. Notice that you can add, delete or modify some words in original description to get the new sentence."""
    
    structure_output_explain = """\
    Your answer should be following the schema provided. Here, """

    structure_output_together = f"""\
    - factors: The names of provides variables.
    - compositions: Samples for all compositions. It is a list (len=2**{num_factors}={num_comb}) where each element has two parameters:
        value: a list of bool. One-to-one correspondence with the values or the variables in the factors list.
        samples: a list contains the given number of generated sentences."""
        
    structure_output_together_with_results = f"""\
    - factors: The names of provided factor variables.
    - results: The names of provided results variables.
    - compositions: Samples for all compositions. It is a list (len=2**{num_factors}={num_comb}) where each element has three parameters:
        value: a list of bool. One-to-one correspondence with the values or the variables in the factors list.
        results: a list of bool. One-to-one correspondence with the values or the variables in the results list. Calculated by the given rules.
        samples: a list contains the given number of generated sentences."""
    
    structure_output_seperate = f"""\
    - factors: The names of provided variables.
    - value: The values of provided variables.
    - sentences: Generated {num_sent} sentences. 
    """
    
    # TODO: the seperated_with_results is not implemented yet.
    
    return dedent("\n".join([user_message, user_about_results,
        user_message_together if together else user_message_seperate,
        requirement,
        structure_output_explain,
        (structure_output_together if not contain_results else structure_output_together_with_results) if together else structure_output_seperate]))
