from typing import Optional

from ..prompts.general_prompt import FUNCTION_ACTIONS_FIND_ELEMENT_PROMPT
from ..utils import JSON_models

OUTPUT_FORMAT_BASE = ("""
[Output Format]
First, analyze trajectories in your thought following the workflow:
- Review the execution history for beneficial vs. irrelevant steps, even if the task succeeded, to improve efficiency.
- Perform a Root Cause Analysis (RCA) on failed trajectories to identify the exact reasons for failure.
- Compare successful and failed trajectories, highlighting the differences or weaknesses that need improvement.
- Propose improvements to the code logic and error-handling, providing a detailed optimization strategy in the High-Level Plan.

Then, output `RPAInfo` object.

[Guidelines for RPA Code Generation]
- Please respond using English typography conventions only. 
- Structure code based on your High-Level Plan: organize steps logically, group related operations, and follow the optimization strategy you proposed.
- Do not import `env_op` (it's provided in runtime).
- **All action calls must be prefixed with `env_op.`** (e.g., `env_op.click()`, `env_op.stop()`)
- IMPORTANT: Tailor the code to the task template and cover all encountered task goals.
- Do not hardcode task-specific content; extract as parameters with sensible defaults (e.g., = None, = ""). Only add parameters necessary for the task. Make sure the code won't fail if the parameter is not provided in the function call.
- Only modify `kwargs` for `target_description`, which must always be included.
- Implement appropriate retries and fallbacks when locating UI elements.
- In loops, verify state changes after each iteration to prevent infinite loops.
- Use external error handling-identify issues with assertions, avoid internal try/except.
- Keep prompt constraints strict for stable MLLM results when using `ask_mllm`.
- Remember to use `env_op.stop()` to stop the RPA function when the task is completed.

[Self-Check Before Output - IMPORTANT]
Before finalizing your RPA code, mentally verify:
- Syntax is correct: use single quotes for strings containing double quotes (e.g., r'pattern="value"'), properly escape special characters, verify all parentheses/brackets match, and ensure valid indentation
- NO hardcoded task-specific values
- `target_description` included in all kwargs
- Proper error handling (assertions, no internal try/except)
- Loops have termination conditions and state change checks
- Code is complete and efficient (no missing or redundant steps)
- Example usage demonstrates realistic parameter values

[Example]
{
  "thought": "...",
  "output": {
    "output_type": "rpa_func",
    "task_type": "Tell me the count of comments that have received more downvotes than upvotes for the user who made the latest post on the {{forum}} forum.",
    "parameters": "- forum (str): The name of the target forum.",
    "rpa_description": "Browse the target forum and count the number of comments that have received more downvotes than upvotes for the user who made the latest post on the target forum.",
    "rpa_code": "def function_name(forum: str):\n  # Step 1: Find all forums\n  kwargs = {{\"type\": \"link\", \"text\": \"Forums\", \"target_description\": \"the 'Forums' link\"}}\n  index = env_op.find_element(**kwargs)\n  assert index != -1, \"Element not found!\" # Check whether the corresponding element exists.\n  env_op.click(index)\n  # Step 2: Click the alphabetical link to see all forums.\n  kwargs = {{\"type\": \"link\", \"text\": \"Alphabetical\", \"target_description\": \"link 'Alphabetical' that show all forums alphabetically\"}}\n  index = env_op.find_element(**kwargs)\n  env_op.click(index)\n  ...\n  # Step 4: Sort the posts by the time.\n  kwargs = {{\"type\": \"button\", \"text\": \"Sort by: Hot\", \"target_description\": \"the sort button to change post order\"}}\n  index = env_op.find_element(**kwargs)\n  env_op.click(index)\n  ...\n  # Step 7: Count the number of comments\n  count = env_op.ask_mllm(\"How many comments have received more downvotes than upvotes at this page?\")\n  # Step 8: Provide the answer\n  env_op.stop(count)",
    "example_usage": "function_name(forum='Showerthoughts')",
    "conclusion": "..."
  }
}
""")


OUTPUT_FORMAT_TOOL_USAGE = ("""
[Available Tools]
You may call:
- fetch_info: Fetch detailed information of a specific step in the trajectory, such as screenshots and a list of UI elements.

[Output Format - Conditional]
First, analyze trajectories in your thought following the workflow:
- Review the execution history for beneficial vs. irrelevant steps, even if the task succeeded, to improve efficiency.
- Perform a Root Cause Analysis (RCA) on failed trajectories to identify the exact reasons for failure.
- Compare successful and failed trajectories, highlighting the differences or weaknesses that need improvement.
- Propose improvements to the code logic and error-handling, providing a detailed optimization strategy in the High-Level Plan.

Then, you **must decide** one of the following output formats based on your reasoning:
🔍 Case - Fetch Needed:
Use this if:
  - You think more ui information is needed to improve RPA Function quality.
💡 Action: Use `fetch_info` to fetch step detail. DO NOT repeatedly to fetch the same step.

⚙️ Case - RPA Function Generation:
Use this if:
  - The input provides all necessary information (screenshot, UI list, user action).
  - You are confident you can generate a complete `RPAInfo` object without ambiguity.
💡 Action: Output the `RPAInfo` object directly.

!important: Always use `fetch_info` if any uncertainty exists.

[Example - Fetch Needed]
{
  "thought": "...",
  "info_to_clarify": "Need to examine ...",
  "output": {
    "output_type": "fetch_info",
    "traj_id": "successful_react_traj",
    "step_n": "...",
  }
}

[Guidelines for RPA Code Generation]
- Please respond using English typography conventions only. 
- Structure code based on your High-Level Plan: organize steps logically, group related operations, and follow the optimization strategy you proposed.
- Do not import `env_op` (it's provided in runtime).
- **All action calls must be prefixed with `env_op.`** (e.g., `env_op.click()`, `env_op.stop()`)
- IMPORTANT: Tailor the code to the task template and cover all encountered task goals.
- Do not hardcode task-specific content; extract as parameters with sensible defaults (e.g., = None, = ""). Only add parameters necessary for the task. Make sure the code won't fail if the parameter is not provided in the function call.
- Only modify `kwargs` for `target_description`, which must always be included.
- Implement appropriate retries and fallbacks when locating UI elements.
- In loops, verify state changes after each iteration to prevent infinite loops.
- Use external error handling-identify issues with assertions, avoid internal try/except.
- Keep prompt constraints strict for stable MLLM results when using `ask_mllm`.
- Remember to use `env_op.stop()` to stop the RPA function when the task is completed.

[Self-Check Before Output - IMPORTANT]
Before finalizing your RPA code, mentally verify:
- Syntax is correct: use single quotes for strings containing double quotes (e.g., r'pattern="value"'), properly escape special characters, verify all parentheses/brackets match, and ensure valid indentation
- NO hardcoded task-specific values
- `target_description` included in all kwargs
- Proper error handling (assertions, no internal try/except)
- Loops have termination conditions and state change checks
- Code is complete and efficient (no missing or redundant steps)
- Example usage demonstrates realistic parameter values

[Example - RPA Function Generation]
{
  "thought": "...",
  "info_to_clarify": "No more steps are needed to inspect for the current rpa function.",
  "output": {
    "output_type": "rpa_func",
    "task_type": "Tell me the count of comments that have received more downvotes than upvotes for the user who made the latest post on the {{forum}} forum.",
    "parameters": "- forum (str): The name of the target forum.",
    "rpa_description": "Browse the target forum and count the number of comments that have received more downvotes than upvotes for the user who made the latest post on the target forum.",
    "rpa_code": "def function_name(forum: str):\n  # Step 1: Find all forums\n  kwargs = {{\"type\": \"link\", \"text\": \"Forums\", \"target_description\": \"the 'Forums' link\"}}\n  index = env_op.find_element(**kwargs)\n  assert index != -1, \"Element not found!\" # Check whether the corresponding element exists.\n  env_op.click(index)\n  # Step 2: Click the alphabetical link to see all forums.\n  kwargs = {{\"type\": \"link\", \"text\": \"Alphabetical\", \"target_description\": \"link 'Alphabetical' that show all forums alphabetically\"}}\n  index = env_op.find_element(**kwargs)\n  env_op.click(index)\n  ...\n # Step 4: Sort the posts by the time.\n  kwargs = {{\"type\": \"button\", \"text\": \"Sort by: Hot\", \"target_description\": \"the sort button to change post order\"}}\n  index = env_op.find_element(**kwargs)\n  env_op.click(index)\n  ...\n  # Step 7: Count the number of comments\n  count = env_op.ask_mllm(\"How many comments have received more downvotes than upvotes at this page?\")\n  # Step 8: Provide the answer\n  env_op.stop(count)",
    "example_usage": "function_name(forum='Showerthoughts')",
    "conclusion": "..."
  }
}
""")

ELEMENT_FINDING_STRATEGY = """
### Element Finding Priority
When locating UI elements, follow this priority order:
1. **env_op.find_element(**kwargs)** - Preferred method for direct element matching by attributes.
2. **env_op.get_ui_content()** - Use as fallback to manually parse the UI list when the above methods are insufficient.
3. **env_op.ask_mllm(question)** - Use when you need visual/language understanding to identify elements (e.g., complex layouts, contextual reasoning).

### Handling Parameter Mismatch
When searching for elements using task template parameters (e.g., forum name, subreddit name) on pages showing complete lists (e.g., alphabetical listing, all forums page):
- Include the parameter value in `text` attribute for initial exact match attempt
- Use `target_description` to describe the element semantically (e.g., "the link that matches or contains the parameter value")
- This allows `find_element` to automatically fall back to semantic matching if exact match fails

### Element Finding in Loops
When searching for elements in a loop:
- Before each iteration, perform scroll actions to change the page state and reveal new content.
- After each action (click, scroll, etc.), call `env_op.get_ui_content()` again to get updated UI information.
- Use scroll('up') or scroll('down') to explore content not visible in the current view before re-attempting find_element.
- Allow consecutive scroll('down') to search, counting only effective scroll-downs that cause a visible UI change; after reaching the bottom, scroll('up') step by step and keep searching during rollback.
- To detect bottom: check for pagination elements (e.g., "More" link) or compare `env_op.get_ui_content()` before and after scrolling (e.g., `str(ui_before) == str(ui_after)`); if unchanged, likely reached bottom. Always set max scroll count to prevent infinite loops.
"""

SKILL_BUILDING_PROMPT = ("""
[system]
You are an expert RPA code builder.
Your goal is to create robust, reusable, and generalized RPA functions that abstract away UI noise and adapt to variable layouts.

{output_format}
"""
+ FUNCTION_ACTIONS_FIND_ELEMENT_PROMPT
+ ELEMENT_FINDING_STRATEGY
+ """
[RPA Code Interaction Rules]
- Ensure the index for each UI action is visible in OBSERVATION.
- Avoid unnecessary or irrelevant actions; focus only on steps critical to the task.

[Task Info]
- Task Type: {task_type}
- Task Template: {task_template}
- Encountered Task Goals (exploration & verification):
{encountered_tasks}

[Available History]
- Failed trajectory (`failed_react_traj`):
{failed_react_traj}

- Successful trajectory (`successful_react_traj`):
{successful_react_traj}

{pre_exec_str}

{fetched_info}
""")

def get_rpa_builder_prompt(
  task_type: Optional[str] = '',
  task_template: Optional[str] = '',
  react_trajs: Optional[list[JSON_models.ReActTraj]] = None,
  pre_rpa_exec_traj: Optional[JSON_models.RPAExecTraj] = None,
  fetched_info: dict = None,
  encountered_task_goals: Optional[list[str]] = None,
  use_tool: bool = False,
  rpa_builder_conslusion: str = ''
) -> str:
  fetched_info_str = ''
  if fetched_info:
    fetched_info_str = f"# Here is the ui info of step {fetched_info['step_n']} in {fetched_info['traj_id']} you fetched:\n" + \
                       fetched_info['ui_content']
  
  output_format = OUTPUT_FORMAT_BASE
  # addition= ''
  if use_tool:
    output_format = OUTPUT_FORMAT_TOOL_USAGE
    # addition = "# If you determine that no tool is necessary, proceed using the following guidelines and workflow:"
  
  failed_react_traj = []
  successful_react_traj = []
  for k, react_traj in enumerate(react_trajs or []):
    traj_str = f"## React_Traj_{k+1}: Try with goal: {react_traj.task}\n"
    
    if react_traj.success:
      traj_str += "## Here is the execution trajectory:\n"
      step = 0
      for m, step_info in enumerate(react_traj.traj or []):
        # Provide UI info for the first step if no UI info is provided.
        if m == 0 and not fetched_info and not pre_rpa_exec_traj:
          traj_str += '### Initial UI Content:\n' + step_info.exec_step_info.before_ui_content
        if step_info.exec_step_info.is_screen_changed or \
          'stop(' in (step_info.soft_coded_action or step_info.action) or \
          'answer(' in (step_info.soft_coded_action or step_info.action) or m == len(react_traj.traj) - 1:
          step += 1
          # Build additional info section: related elements or original action
          related_elements = getattr(step_info.exec_step_info, "related_elements", "") or ""
          
          if related_elements:
            additional_info = f'\n### Related element: {related_elements}'
          else:
            # If no related elements, show original action only if it differs from soft_coded_action
            soft_action = (step_info.action or "").strip()
            hard_action = (step_info.action or "").strip()
            if soft_action != hard_action and hard_action:
              additional_info = f'\n### Original action: {step_info.action}'
            else:
              additional_info = ''
          traj_str += (f'Step {step}:'
            f'\n### Action Justification: {step_info.action_reason}'
            f'\n### Action Code:\n{step_info.soft_coded_action or step_info.action}'
            f'{additional_info}'
            f'\n### Execution Summary: {step_info.execution_summary}\n\n')
      
      traj_str += f"\n\n### Episode Conclusion: {react_traj.conclusion}\n"
      successful_react_traj.append(traj_str)
    else:
      traj_str += f"\n\n### Episode Conclusion: {react_traj.conclusion}\n"
      if react_traj.pre_reflection:
        traj_str += f"### Reflection: {react_traj.pre_reflection}\n"
      failed_react_traj.append(traj_str)
  
  # considering the context length, only keep the last traj
  failed_react_traj = failed_react_traj[-1] if len(failed_react_traj) else None # failed_react_traj default does not contain detailed info, detailed info can be retrieved by builder
  successful_react_traj = successful_react_traj[-1] if len(successful_react_traj) else None
  
  pre_exec_str = ''
  if pre_rpa_exec_traj:
    exec_step = len(pre_rpa_exec_traj.env_op_traj)
    previous_action_history = '\n\n'.join(pre_rpa_exec_traj.action_history[:exec_step]) if exec_step > 0 else None
    pre_exec_str = f"# Execution Trajectory of Previous RPA:\n" \
                   f"## RPA_Exec_Traj:\nTry with goal: {pre_rpa_exec_traj.task}\n" \
                   f"## Conclusion of the RPA Code Generation Process: {rpa_builder_conslusion}\n" \
                   f"## Executed Function: {pre_rpa_exec_traj.function_call}\n" \
                   f"## Previous RPA Code:\n" \
                   f"{pre_rpa_exec_traj.rpa_code}\n" \
                   f"## Execution Trajectory (`pre_rpa_exec_traj`):\n{previous_action_history}" \
                   f"\n\n## Execution Result: {pre_rpa_exec_traj.exec_result.exec_feedback}"
    if pre_rpa_exec_traj.env_op_traj:
      pre_exec_str += f"\n\n## UI elements list at the point of code execution failure:\n{pre_rpa_exec_traj.env_op_traj[-1].after_ui_content}"
    if pre_rpa_exec_traj.fix_evaluator_analysis is not None:
      pre_exec_str += f"\n### Analysis at break point:\n {pre_rpa_exec_traj.fix_evaluator_analysis}\n"
    if pre_rpa_exec_traj.fix_react_traj:
      pre_exec_str += f"\n### Fix_Traj (`fix_react_traj`):\n" \
                      f"\nTake attention to Fix_Traj!!"
      for m, step_info in enumerate(pre_rpa_exec_traj.fix_react_traj[0].traj):
        # Build additional info section: related elements or original action
        related_elements = getattr(step_info.exec_step_info, "related_elements", "") or ""
        
        if related_elements:
          additional_info = f'\n### Related element: {related_elements}'
        else:
          # If no related elements, show original action only if it differs from soft_coded_action
          soft_action = (step_info.soft_coded_action or "").strip()
          hard_action = (step_info.action or "").strip()
          if soft_action != hard_action and hard_action:
            additional_info = f'\n### Original action: {step_info.action}'
          else:
            additional_info = ''

        pre_exec_str += (f'\nStep {m + exec_step + 1}:'
                      f'\n### Action Justification: {step_info.action_reason}'
                      f'\n### Action Code:\n{step_info.soft_coded_action or step_info.action}'
                      f'{additional_info}'
                      f'\n### Execution Summary: {step_info.execution_summary}\n\n')
  
  encountered_goals_list = encountered_task_goals or []
  encountered_tasks_str = '\n'.join(f" - Task Goal: {goal}" for goal in encountered_goals_list) if encountered_goals_list else " - (none yet)"
  
  prompt_template = SKILL_BUILDING_PROMPT
  
  return prompt_template.format(
    task_type=task_type,
    task_template=task_template,
    encountered_tasks=encountered_tasks_str,
    failed_react_traj=failed_react_traj,
    successful_react_traj=successful_react_traj,
    pre_exec_str=pre_exec_str,
    output_format=output_format,
    fetched_info=fetched_info_str,
    # addition=addition,
  )