from typing import Optional, Literal
import re

PROMPT_HEADER = """
You are an intelligent agent that operates an Android phone for a user.
Your role is to break down complex requests into optimal, step-by-step actions and execute them while adapting to changes.
"""

OPERATION_GUIDANCE_PROMPT_INDEX = """
[OPERATION_GUIDANCE]
Follow these guidelines:
General:
- Choose the simplest method.
- Navigate to gather necessary info (e.g., open the calendar for schedule queries) before answering.
- For file-related operations, make sure to operate only on the exact target file; do not interact with similar files. You must locate and use the precise file required.
- Remember to save the file after making any changes to its contents.
- When naming files, ensure proper file extensions.

Action Related:
- Ensure the index for click, long_press, and input_text is visible in both the screenshot and UI list.
- Use swipe('up') or swipe('down') to reveal elements not currently visible.
- Confirm that a UI element supports the intended action before interacting.

Text Operations:
- Use standard text selection methods (long-press and selection bar) when needed.
"""

OPERATION_GUIDANCE_PROMPT_COORDINATE = """
[OPERATION_GUIDANCE]
Follow these guidelines:
General:
- Choose the simplest method.
- Navigate to gather necessary info (e.g., open the calendar for schedule queries) before answering.
- For file-related operations, make sure to operate only on the exact target file; do not interact with similar files. You must locate and use the precise file required.
- Remember to save the file after making any changes to its contents.
- When naming files, ensure proper file extensions.

Action Related:
- When using coordinates, estimate them from the screenshot or use UI element bounding box information (bbox_pixels) if available.
- The screen resolution is 1080*2400 pixels. For coordinates, (0, 0) is the top-left corner. X increases to the right, Y increases downward.
- Calculate coordinates based on screen dimensions and element positions visible in the screenshot.
- Use swipe('up') or swipe('down') to reveal elements not currently visible.
- Confirm that a UI element supports the intended action before interacting.

Text Operations:
- Use standard text selection methods (long-press and selection bar) when needed.
"""

OPERATION_GUIDANCE_PROMPT_COORDINATE_SOM = """
[OPERATION_GUIDANCE]
Follow these guidelines:
General:
- Choose the simplest method.
- Navigate to gather necessary info (e.g., open the calendar for schedule queries) before answering.
- For file-related operations, make sure to operate only on the exact target file; do not interact with similar files. You must locate and use the precise file required.
- Remember to save the file after making any changes to its contents.
- When naming files, ensure proper file extensions.

Action Related:
- The screenshot contains SOM (Set-of-Mark) bounding boxes that mark UI elements based on the accessibility tree. These boxes help identify interactive elements and their approximate locations.
- When using coordinates, you can estimate them from the screenshot by referencing the SOM bounding boxes. The center of a SOM box is typically a good coordinate for clicking that element.
- The screen resolution is 1080*2400 pixels. For coordinates, (0, 0) is the top-left corner. X increases to the right, Y increases downward.
- Calculate coordinates based on screen dimensions and element positions visible in the screenshot, using the SOM boxes as visual guides.
- Use swipe('up') or swipe('down') to reveal elements not currently visible.
- Confirm that a UI element supports the intended action before interacting.

Text Operations:
- Use standard text selection methods (long-press and selection bar) when needed.
"""

OPERATION_GUIDANCE_PROMPT_INDEX_SOM = """
[OPERATION_GUIDANCE]
Follow these guidelines:
General:
- Choose the simplest method.
- Navigate to gather necessary info (e.g., open the calendar for schedule queries) before answering.
- For file-related operations, make sure to operate only on the exact target file; do not interact with similar files. You must locate and use the precise file required.
- Remember to save the file after making any changes to its contents.
- When naming files, ensure proper file extensions.

Action Related:
- The screenshot contains SOM (Set-of-Mark) bounding boxes with numeric indexes marked at the top-left corner of each box. These indexes are based on the accessibility tree and mark interactive UI elements.
- Read the numeric index from each SOM bounding box in the screenshot. Use these indexes directly in your actions (e.g., click(5) for the element marked with index 5).
- Ensure the index you use is visible in the screenshot's SOM bounding boxes.
- Use swipe('up') or swipe('down') to reveal elements not currently visible.
- Confirm that a UI element supports the intended action before interacting.
- Every element is clickable by default.

Text Operations:
- Use standard text selection methods (long-press and selection bar) when needed.
"""

OPERATION_GUIDANCE_PROMPT = OPERATION_GUIDANCE_PROMPT_INDEX  # Default for backward compatibility

OUTPUT_FORMAT_TEMPLATE_base = """
[Output Format]
Your output must consist of following things:
- Observations: Summarize key input details and immediate screen observations. Include all obvious insights.
{reflection_block}
- Completed Tasks: List completed tasks, each starting with a ✅.
- Plan Justification: Briefly explain the rationale behind your plan.
- Plan List: List tasks to achieve the goal, each starting with a ⬜; if the goal is achieved, output "goal completed."
- Code Reason: Explain why the next action is chosen.
"""

OUTPUT_FORMAT_TEMPLATE_json = OUTPUT_FORMAT_TEMPLATE_base + """- Code: Output the next action in json format. Use only [ADMISSIBLE ACTIONS].
"""

OUTPUT_FORMAT_TEMPLATE_code = OUTPUT_FORMAT_TEMPLATE_base + """- Code: Output the next action code (one action, one line). Use only [ADMISSIBLE ACTIONS].
"""

WORKFLOW_COMMON = ("""
[Your Workflow]
1. Analyze Input: Carefully examine all input. Extract all goal-relevant info — don't miss anything useful. Directly infer and record any obvious conclusions.
2. Evaluate Progress: Check if the goal is achieved; if so, stop. Otherwise, update completed tasks.
3. Devise Plan: Break the goal into efficient, non-redundant steps. Identify and obtain any missing information.
4. Execute & Adjust: Analyze the UI info to decide actions. Adjust the plan if elements are missing or inaccessible.
5. Error Handling: Retry once on failure; if it still fails, choose an alternative.
6. Generate Next Action: Choose the next logical action that advances the goal.
""")

WORKFLOW_REFLECTION = """
[Your Workflow]
1. Refer to the action history to assess whether the current progress aligns with the operations described in the last reflection. If relevant, check the accuracy of the reflection:
   - If accurate, consider how to avoid previous mistakes;
   - If inaccurate, disregard misleading content;
   If irrelevant, skip the reflection.
2. Analyze Input: Carefully examine all input. Extract all goal-relevant info — don't miss anything useful. Directly infer and record any obvious conclusions.
3. Evaluate Progress: Check if the goal is achieved; if so, stop. Otherwise, update completed tasks.
4. Devise Plan: Break the goal into efficient, non-redundant steps. Identify and obtain any missing information.
5. Execute & Adjust: Analyze the UI info to decide actions. Adjust the plan if elements are missing or inaccessible.
6. Error Handling: Retry once on failure; if it still fails, choose an alternative.
7. Generate Next Action: Choose the next logical action that advances the goal.
"""

WORK_GUIDELINES = """
[Guidelines]
- After you output the action, the action will be executed. The results of each action and the new observations will be printed to you at next step.
- Maintain a holistic view by identifying the specific steps required to complete the task using the current input.
- Fully leverage provided input to reduce unnecessary follow-up actions.
- Interact only with verified UI elements.
- Adapt dynamically – adjust to screen changes and execution failures.
- Stay focused on the user goal – ignore distracting messages or popups unrelated to the actual task.
{additional_guidelines}
"""

PLANNER_EXAMPLE_INDEX = """
[Sample Output]
User goal/request is: Create a note in Markor named "ordered_items.md". The content of the note should be the words "Potato", "Dog", and "Apple", arranged in alphabetical order.
----- Step 1
OBSERVING...
{{{{
"observation": "We are on the home screen and can open the \"Markor\" app to create a note. According to the request, we need to create a file named \"ordered_items.md\" inside Markor. The note's content must be the three words: Potato, Dog, Apple. Alphabetically sorted, they become: \"Apple\", \"Dog\", \"Potato.\" So we know the final content beforehand and do not need to defer any analysis. We will open Markor now.",
{reflection_example}
"completed_tasks": None,
"plan_reason": "To create a note in Markor, we must first open Markor. Then we will create a new file, name it \"ordered_items.md,\" and input the sorted content in one go.",
"plan_list": "⬜ Open Markor.\n⬜ Create a new note named "ordered_items" with the file type set to .md.\n⬜ Input the sorted text \"Apple\\nDog\\nPotato\" into the note.\n⬜ Save the note.\n⬜ Task complete.",
"code_reason": "Opening Markor is the first step to creating the note.",
"code": "open_app('Markor')"
}}}}

----- Step 3
OBSERVING...
{{{{
"observation": "The screen displays a dialog for creating a new file or folder in the Markor app. The \"Name\" field (index 1) is editable, and there are options for \"Type\" and \"Template\". On the right side of the Name field shows '.md' and the type is Markdown. The \"OK\" button (index 11) is present to confirm the action.",
{reflection_example}
"completed_tasks": "✅ Opened the Markor app.\n✅ Clicked the \"Create a new file or folder\" button.",
"plan_reason": "To create a new note named \"ordered_items\", the name must be entered in the \"Name\" field. Because the Name field shows '.md' on the right, there is no need to switch the type. And the creation confirmed by clicking \"OK\".",
"plan_list": "⬜ Enter the name \"ordered_items\" for the new note.\n⬜ Confirm the creation of the new note by clicking \"OK\".\n⬜ Input \"Apple\\nDog\\nPotato\" into the note.\n⬜ Save the note.\n⬜ Task complete.",
"code_reason": "Entering the note name is the next logical step in creating the note.",
"code": "input_text('ordered_items', 1)"
}}}}

----- Step 7
OBSERVING...
{{{{
"observation": "The note \"ordered_items.md\" has been created and saved with the correct sorted content.",
{reflection_example}
"completed_tasks": "✅ Opened the Markor app.\n✅ Clicked the \"Create a new file or folder\" button.\n✅ Entered \"ordered_items\" into the \"Name\" field.\n✅ Successfully created the new note.\n✅ Input the sorted content into the note.\n✅ Saved the note.",
"plan_reason": "The user's goal is achieved. No further actions are needed.",
"plan_list": "⬜ No more actions. Complete the task now.",
"code_reason": "The task requirements have been fully met.",
"code": "stop(\"complete\")"
}}}}
"""

PLANNER_EXAMPLE_COORDINATE = """
[Sample Output]
User goal/request is: Create a note in Markor named "ordered_items.md". The content of the note should be the words "Potato", "Dog", and "Apple", arranged in alphabetical order.
----- Step 1
OBSERVING...
{{{{
"observation": "We are on the home screen and can open the \"Markor\" app to create a note. According to the request, we need to create a file named \"ordered_items.md\" inside Markor. The note's content must be the three words: Potato, Dog, Apple. Alphabetically sorted, they become: \"Apple\", \"Dog\", \"Potato.\" So we know the final content beforehand and do not need to defer any analysis. We will open Markor now.",
{reflection_example}
"completed_tasks": None,
"plan_reason": "To create a note in Markor, we must first open Markor. Then we will create a new file, name it \"ordered_items.md,\" and input the sorted content in one go.",
"plan_list": "⬜ Open Markor.\n⬜ Create a new note named "ordered_items" with the file type set to .md.\n⬜ Input the sorted text \"Apple\\nDog\\nPotato\" into the note.\n⬜ Save the note.\n⬜ Task complete.",
"code_reason": "Opening Markor is the first step to creating the note.",
"code": "open_app('Markor')"
}}}}

----- Step 3
OBSERVING...
{{{{
"observation": "The screen displays a dialog for creating a new file or folder in the Markor app. The \"Name\" field is visible at coordinates approximately (200, 400). On the right side of the Name field shows '.md' and the type is Markdown. The \"OK\" button is visible at coordinates approximately (400, 600).",
{reflection_example}
"completed_tasks": "✅ Opened the Markor app.\n✅ Clicked the \"Create a new file or folder\" button.",
"plan_reason": "To create a new note named \"ordered_items\", the name must be entered in the \"Name\" field. Because the Name field shows '.md' on the right, there is no need to switch the type. And the creation confirmed by clicking \"OK\".",
"plan_list": "⬜ Enter the name \"ordered_items\" for the new note.\n⬜ Confirm the creation of the new note by clicking \"OK\".\n⬜ Input \"Apple\\nDog\\nPotato\" into the note.\n⬜ Save the note.\n⬜ Task complete.",
"code_reason": "Entering the note name is the next logical step in creating the note.",
"code": "input_text('ordered_items', 200, 400)"
}}}}

----- Step 7
OBSERVING...
{{{{
"observation": "The note \"ordered_items.md\" has been created and saved with the correct sorted content.",
{reflection_example}
"completed_tasks": "✅ Opened the Markor app.\n✅ Clicked the \"Create a new file or folder\" button.\n✅ Entered \"ordered_items\" into the \"Name\" field.\n✅ Successfully created the new note.\n✅ Input the sorted content into the note.\n✅ Saved the note.",
"plan_reason": "The user's goal is achieved. No further actions are needed.",
"plan_list": "⬜ No more actions. Complete the task now.",
"code_reason": "The task requirements have been fully met.",
"code": "stop(\"complete\")"
}}}}
"""

INPUT_TEMPLATE_WITH_TREE = ("""
# User goal/request: {goal}
{reflection}

[Input Provided]
- Action History & Execution Result:
{action_history}

- Main changes on the page after the previous action is executed:
{screen_changes}

- Completed Tasks:
{completed_plan}

- Previous Plan:
{previous_plan}

- Current screenshot, annotated screenshot, and UI elements list:
{index_range_info}
{ui_content}
""")

INPUT_TEMPLATE_SCREENSHOT_ONLY = ("""
# User goal/request: {goal}
{reflection}

[Input Provided]
- Action History & Execution Result:
{action_history}

- Main changes on the page after the previous action is executed:
{screen_changes}

- Completed Tasks:
{completed_plan}

- Previous Plan:
{previous_plan}

- Current screenshot (original resolution, no annotations):
The screenshot is provided as an image. Analyze it directly to determine coordinates for actions.
{index_range_info}
""")

INPUT_TEMPLATE_SCREENSHOT_ONLY_SOM = ("""
# User goal/request: {goal}
{reflection}

[Input Provided]
- Action History & Execution Result:
{action_history}

- Main changes on the page after the previous action is executed:
{screen_changes}

- Completed Tasks:
{completed_plan}

- Previous Plan:
{previous_plan}

- Current screenshot (original resolution, with SOM bounding boxes):
The screenshot is provided as an image with SOM (Set-of-Mark) bounding boxes overlaid on UI elements. These boxes mark interactive elements based on the accessibility tree. Use the SOM boxes to identify elements and estimate coordinates for actions. The center of a SOM box is typically a good coordinate for clicking that element.
{index_range_info}
""")

INPUT_TEMPLATE_SCREENSHOT_ONLY_SOM_INDEX = ("""
# User goal/request: {goal}
{reflection}

[Input Provided]
- Action History & Execution Result:
{action_history}

- Main changes on the page after the previous action is executed:
{screen_changes}

- Completed Tasks:
{completed_plan}

- Previous Plan:
{previous_plan}

- Current screenshot (original resolution, with SOM bounding boxes and numeric indexes):
The screenshot is provided as an image with SOM (Set-of-Mark) bounding boxes overlaid on UI elements. Each bounding box has a numeric index marked at its top-left corner. These indexes are based on the accessibility tree and mark interactive UI elements. Read the numeric index from each SOM bounding box and use these indexes directly in your actions (e.g., click(5) for the element marked with index 5).
{index_range_info}
""")

INPUT_TEMPLATE = INPUT_TEMPLATE_WITH_TREE  # Default for backward compatibility

def get_prompt_template(
    reflection: Optional[str] | None = None,
    action_space_mode: Literal['index', 'coordinate'] = 'index',
    ui_info_mode: Literal['screenshot_with_tree', 'screenshot_only', 'screenshot_only_som'] = 'screenshot_with_tree',
    img_resize_mode: Literal['original', 'resized'] = 'resized',
    enable_shell_action: bool = False
) -> dict[str, str]:
  """
  Generate prompt template for ReAct* planner.
  
  ReAct* only generates hardcoded actions during exploration.
  Action translation (soft-coding) is handled by AutoRPA in the building phase.
  
  Args:
    reflection: Optional reflection from previous failed attempt
    action_space_mode: 'index' to use element indexes, 'coordinate' to use pixel coordinates
    ui_info_mode: 'screenshot_with_tree' to include UI elements list, 'screenshot_only' to use only screenshot, 'screenshot_only_som' to use screenshot with SOM bounding boxes
    img_resize_mode: 'original' to use original screenshots, 'resized' to use resized screenshots (461x1024)
    enable_shell_action: Whether to enable shell command action (default: False)
  """
  # Import action space prompts
  from .action_space import (
    FUNCTION_ACTIONS_PROMPT_COORDINATE, FUNCTION_ACTIONS_PROMPT,
    FUNCTION_ACTIONS_PROMPT_COORDINATE_WITH_SHELL, FUNCTION_ACTIONS_PROMPT_WITH_SHELL
  )
  
  # Select action space based on mode and shell enablement
  if action_space_mode == 'coordinate':
    action_space = FUNCTION_ACTIONS_PROMPT_COORDINATE_WITH_SHELL if enable_shell_action else FUNCTION_ACTIONS_PROMPT_COORDINATE
    # Use SOM-specific guidance if in screenshot_only_som mode
    if ui_info_mode == 'screenshot_only_som':
      operation_guidance = OPERATION_GUIDANCE_PROMPT_COORDINATE_SOM
    else:
      operation_guidance = OPERATION_GUIDANCE_PROMPT_COORDINATE
    
    # Dynamically modify operation_guidance to reflect image resolution when using resized images
    if img_resize_mode == 'resized':
      # Replace screen resolution info with resized image resolution info
      operation_guidance = operation_guidance.replace(
        "The screen resolution is 1080*2400 pixels",
        "The screenshot image resolution is 461*1024 pixels (resized from original 1080*2400). Output coordinates relative to this resized image."
      )
    
    planner_example = PLANNER_EXAMPLE_COORDINATE
  else:
    # index mode
    action_space = FUNCTION_ACTIONS_PROMPT_WITH_SHELL if enable_shell_action else FUNCTION_ACTIONS_PROMPT
    # Use SOM-specific guidance if in screenshot_only_som mode
    if ui_info_mode == 'screenshot_only_som':
      operation_guidance = OPERATION_GUIDANCE_PROMPT_INDEX_SOM
    else:
      operation_guidance = OPERATION_GUIDANCE_PROMPT_INDEX
    planner_example = PLANNER_EXAMPLE_INDEX
  
  output_template = OUTPUT_FORMAT_TEMPLATE_code
  
  if reflection:
    reflection_block = "- Reflection Consideration: Explain how the previous reflection influences the next action."
    workflow = WORKFLOW_REFLECTION
    reflection_example = '"consider_reflection": "...",'
  else:
    reflection_block = ""
    workflow = WORKFLOW_COMMON
    reflection_example = ""
  
  output_template = output_template.format(reflection_block=reflection_block)
  planner_example = planner_example.format(reflection_example=reflection_example)
  
  # Select input template based on ui_info_mode and action_space_mode
  if ui_info_mode == 'screenshot_only':
    user_prompt_template = INPUT_TEMPLATE_SCREENSHOT_ONLY
  elif ui_info_mode == 'screenshot_only_som':
    # Use different template for index mode vs coordinate mode
    if action_space_mode == 'index':
      user_prompt_template = INPUT_TEMPLATE_SCREENSHOT_ONLY_SOM_INDEX
    else:
      user_prompt_template = INPUT_TEMPLATE_SCREENSHOT_ONLY_SOM
  else:
    user_prompt_template = INPUT_TEMPLATE_WITH_TREE
  
  return PROMPT_HEADER + action_space + operation_guidance + output_template + workflow + WORK_GUIDELINES + planner_example, user_prompt_template

def _extract_index_range(
  ui_content: Optional[str] = None,
  ui_content_full_dict: Optional[list] = None,
  ui_info_mode: str = 'screenshot_with_tree'
) -> Optional[str]:
  """Extract valid index range from UI content (string or dict format).
  
  Args:
    ui_content: UI content string containing JSON-formatted element descriptions
    ui_content_full_dict: UI content as list of dictionaries (from ui_content_full_dict)
    ui_info_mode: UI info mode to determine the appropriate prompt text
    
  Returns:
    String describing the valid index range, or None if no valid indices found
  """
  indices = []
  
  # Try to extract from ui_content_full_dict first (more reliable)
  if ui_content_full_dict and isinstance(ui_content_full_dict, list):
    for elem in ui_content_full_dict:
      if isinstance(elem, dict) and 'index' in elem:
        idx = elem.get('index', -1)
        # Ensure index is an integer and is non-negative
        try:
          idx = int(idx)
          if idx >= 0:  # Only include valid indices
            indices.append(idx)
        except (ValueError, TypeError):
          # Skip invalid index values
          continue
  
  # Fall back to extracting from ui_content string if no dict provided
  # Only extract if ui_content is not "Not available" or similar
  if not indices and ui_content and not ui_content.startswith('Not available'):
    # Extract all index values from JSON objects in ui_content
    # Pattern: "index": <number>
    index_pattern = r'"index":\s*(\d+)'
    index_strs = re.findall(index_pattern, ui_content)
    # Regex already ensures digits only, but add safety check for type conversion
    try:
      indices = [int(idx) for idx in index_strs]
    except (ValueError, TypeError):
      # If conversion fails, indices remains empty list
      indices = []
  
  if not indices:
    return None
  
  # Convert to integers and find min/max
  min_index = min(indices)
  max_index = max(indices)
  
  # Generate appropriate prompt text based on ui_info_mode
  if ui_info_mode == 'screenshot_only_som':
    # For screenshot_only_som mode, reference is to screenshot's SOM bounding boxes
    if min_index == max_index:
      return f"\n[IMPORTANT] Valid index range: Only index {min_index} is available in the screenshot's SOM bounding boxes. Use only this index in your actions."
    else:
      return f"\n[IMPORTANT] Valid index range: {min_index} to {max_index} (inclusive). The index you use in actions (click, long_press, input_text) must be within this range and must be visible in the screenshot's SOM bounding boxes."
  else:
    # For screenshot_with_tree mode, reference is to UI elements list
    if min_index == max_index:
      return f"\n[IMPORTANT] Valid index range: Only index {min_index} is available in the UI elements list. Use only this index in your actions."
    else:
      return f"\n[IMPORTANT] Valid index range: {min_index} to {max_index} (inclusive). The index you use in actions (click, long_press, input_text) must be within this range and must appear in the UI elements list above."


def get_planner_prompt(
  previous_plan: Optional[str],
  completed_plan: Optional[str],
  goal: str,
  action_history: list,
  reflection: Optional[str] | None = None,
  ui_content: str = "",
  screen_changes: str = '',
  additional_guidelines: Optional[str] = '',
  action_space_mode: Literal['index', 'coordinate'] = 'index',
  ui_info_mode: Literal['screenshot_with_tree', 'screenshot_only', 'screenshot_only_som'] = 'screenshot_with_tree',
  img_resize_mode: Literal['original', 'resized'] = 'resized',
  enable_shell_action: bool = False,
  ui_content_full_dict: Optional[list] = None,
) -> dict[str, str]:
  if action_history:
    action_history = '\n'.join(action_history)
  else:
    action_history = 'You just started, no action has been performed yet.'
  
  # Log action space mode for debugging
  import logging
  logger = logging.getLogger('gui_agents.react_star.prompts')
  logger.debug(f"get_planner_prompt: action_space_mode={action_space_mode}, "
               f"ui_info_mode={ui_info_mode}, img_resize_mode={img_resize_mode}, "
               f"enable_shell_action={enable_shell_action}")
  
  system_prompt_template, user_prompt_template = get_prompt_template(reflection, action_space_mode, ui_info_mode, img_resize_mode, enable_shell_action)

  system_prompt = system_prompt_template.format(
    additional_guidelines=additional_guidelines,
  )

  # Extract index range for index mode
  # Unified extraction from either ui_content string or ui_content_full_dict
  if action_space_mode == 'index' and ui_info_mode in ['screenshot_with_tree', 'screenshot_only_som']:
    index_range_info = _extract_index_range(
      ui_content=ui_content,
      ui_content_full_dict=ui_content_full_dict,
      ui_info_mode=ui_info_mode
    ) or ""
  else:
    index_range_info = ""

  user_prompt = user_prompt_template.format(
    goal=goal,
    previous_plan=previous_plan,
    completed_plan=completed_plan,
    action_history=action_history,
    ui_content=ui_content if ui_content else 'Not available',
    screen_changes=screen_changes,
    reflection='\n## The following is the reflection of the previous fail attempt:\n' + reflection if reflection else '',
    index_range_info=index_range_info,
  )
  
  return {
    'system': system_prompt,
    'user': user_prompt
  }