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]
- Structure code based on your High-Level Plan.
- Do not import `env_op` (it's provided in runtime).
- IMPORTANT: Tailor the code to the task template and cover all encountered task goals.
- Ensure the function can handle diverse inputs by giving every parameter a sensible default value (for example, = None, = ""). And the code won't fail if the parameter is not provided.
- Use `env_op.find_element(xpath, target_description)` with robust xpath expressions and clear target descriptions.
- **CRITICAL: For matching multiple element types, use "or" condition, NOT union operator in XPath**:
  * CORRECT: `//*[(name()='span' or name()='a') and text()='OK']`
  * WRONG: `//*[name()='div'](//*[name()='span'] | //*[name()='a'])` ← Invalid XPath syntax!
- Implement appropriate retries and fallbacks when locating UI elements.
- **Add popup dismissal logic ONLY if pop-ups were actually encountered in exploration trajectories. If the trajectories show no pop-ups, do not include dismissal code.**
- In loops, verify state changes after each iteration to prevent infinite loops.
- Use assertions for error detection; avoid internal try/except blocks.
- **Ensure valid Python syntax: use single quotes for strings containing double quotes (e.g., r'pattern="value"'), properly escape special characters, and verify all parentheses/brackets match.**

[Self-Check Before Output - IMPORTANT]
Before finalizing your RPA code, mentally verify:
- Syntax is correct (brackets match, strings properly escaped, valid indentation)
- **XPath syntax is valid** (use "or" for multiple types, NOT union operator with parent path)
- NO hardcoded task-specific values (all dynamic content as parameters with defaults)
- `target_description` is clear and provided for all `find_element` calls
- 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": "...",
    "parameters": "- text (str): The text label of the button to be clicked.",
    "rpa_description": "Click on the specified button on the interface using its label text.",
    "rpa_code": "def click_button(text: str):\n    # Click the button using XPath, identified by its text.\n    xpath = env_op.find_element(f\"//*[name()='button' and text()='{text}']\", f\"Button labeled '{text}' for confirming actions\")\n    assert xpath is not None, f\"Button with text '{text}' not found\"\n    env_op.click_xpath(xpath)",
    "example_usage": "click_button(text=\"Ok\")",
    "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]
- Structure code based on your High-Level Plan.
- Do not import `env_op` (it's provided in runtime).
- IMPORTANT: Tailor the code to the task template and cover all encountered task goals.
- Ensure the function can handle diverse inputs by giving every parameter a sensible default value (for example, = None, = ""). And the code won't fail if the parameter is not provided.
- Use `env_op.find_element(xpath, target_description)` with robust xpath expressions and clear target descriptions.
- **CRITICAL: For matching multiple element types, use "or" condition, NOT union operator in XPath**:
  * CORRECT: `//*[(name()='span' or name()='a') and text()='OK']`
  * WRONG: `//*[name()='div'](//*[name()='span'] | //*[name()='a'])` ← Invalid XPath syntax!
- Implement appropriate retries and fallbacks when locating UI elements.
- **Add popup dismissal logic ONLY if pop-ups were actually encountered in exploration trajectories. If the trajectories show no pop-ups, do not include dismissal code.**
- In loops, verify state changes after each iteration to prevent infinite loops.
- Use assertions for error detection; avoid internal try/except blocks.
- **Ensure valid Python syntax: use single quotes for strings containing double quotes (e.g., r'pattern="value"'), properly escape special characters, and verify all parentheses/brackets match.**

[Self-Check Before Output - IMPORTANT]
Before finalizing your RPA code, mentally verify:
- Syntax is correct (brackets match, strings properly escaped, valid indentation)
- **XPath syntax is valid** (use "or" for multiple types, NOT union operator with parent path)
- NO hardcoded task-specific values (all dynamic content as parameters with defaults)
- `target_description` is clear and provided for all `find_element` calls
- 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": "...",
    "parameters": "- text (str): The text label of the button to be clicked.",
    "rpa_description": "Click on the specified button on the interface using its label text.",
    "rpa_code": "def click_button(text: str):\n    # Click the button using XPath, identified by its text.\n    xpath = env_op.find_element(f\"//*[name()='button' and text()='{text}']\", f\"Button labeled '{text}' for confirming actions\")\n    assert xpath is not None, f\"Button with text '{text}' not found\"\n    env_op.click_xpath(xpath)",
    "example_usage": "click_button(text=\"Ok\")",
    "conclusion": "..."
  }
}
""")

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
+ """
[RPA Code Interaction Rules]
- Ensure the xpath for each UI action is visible in both the UI list and screenshot.
- 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,
  use_tool: bool = False,
  rpa_builder_conslusion: str = '',
  encountered_task_goals: Optional[list[str]] = None
) -> 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 HTML for the first step if no HTML is provided.
        if m == 0 and not fetched_info and not pre_rpa_exec_traj:
          traj_str += '### Initial HTML Content:\n' + step_info.ui_content + '\n'
        if step_info.exec_step_info.is_screen_changed or m == len(react_traj.traj) - 1:
          step += 1
          traj_str += (f'Step {step}:\n'
            f'\n### Action Justification: {step_info.action_reason}'
            f'\n### Action Code:\n{step_info.soft_coded_action}'
            f'\n### Related elements: {step_info.related_elements}'
            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.reflection:
        traj_str += f"### Reflection:\n{react_traj.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
  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### Fix_Traj (`fix_react_traj`): " \
                      f"\nTake attention to Fix_Traj!!" \
                      f"\n### Analysis at break point:\n {pre_rpa_exec_traj.fix_evaluator_analysis}\n"
      for m, step_info in enumerate(pre_rpa_exec_traj.fix_react_traj):
        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}'
                      f'\n### Related elements: {step_info.related_elements}'
                      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,
    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,
    encountered_tasks=encountered_tasks_str,
    # addition=addition,
  )