from typing import Optional

from .general_prompt import get_builder_action_prompt
from ..utils import models

SHELL_ACTION_GUIDANCE = """
[Optional: Shell Action for System-Level Operations]
When shell action is enabled, you MAY use `env_op.shell(command)` for system-level operations that cannot be easily achieved with standard UI actions.

**Usage Rules (Use as LAST RESORT):**
1. Prefer standard actions: open_app, click, input_text, swipe, etc.
2. Only use `env_op.shell` when standard actions are insufficient or overly complex
3. Common valid use cases:
   - System settings: `env_op.shell("settings put system screen_brightness {value}")`
   - System queries: `env_op.shell("dumpsys battery")`, `env_op.shell("wm size")`
   - App info: `env_op.shell("pm list packages | grep {keyword}")`
4. `env_op.shell` returns a string, which is extracted from the shell response by env_op, like: "255"

**When NOT to use shell:**
- ❌ Opening apps → Use `env_op.open_app(app_name)`
- ❌ Clicking UI → Use `env_op.click()` or `find_element + click`
- ❌ Input text → Use `env_op.input_text()`

**Example - Appropriate shell usage:**
```python
# Setting brightness (simpler than multiple UI clicks)
brightness_value = brightness if brightness else 100
env_op.shell(f"settings put system screen_brightness {brightness_value}")
```
"""

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: 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 assertions for error detection; avoid internal try/except blocks.
- Do not add unnecessary validations; only verify essential conditions that are critical for the task.
- Use `env_op.wait()` sparingly; frequent waiting is unnecessary.
- Remember to use `env_op.stop()` or `env_op.answer()` to stop the RPA function when the task is completed.
- Do not check permissions in RPA code; permission handling is managed outside the RPA execution phase.

[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": "...",
    "parameters": "- file_name (str): The name of the file to be created in the Markor app.\n- file_suffix (str): The type of the file (such as md, txt).\n- content (str): The content to be written into the file.",
    "rpa_description": "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.",
    "rpa_code": "def function_name(file_name: str, file_suffix: str, content: str):\n  # Step 1: Open the Markor app\n  env_op.open_app('Markor')\n  ...\n  # Step 3: Find the input field for the file name in the note creation dialog\n  kwargs = {{\"text\": \"my_note\", \"actions\": [\"input_text\"], \"target_description\": \"Input field for the file name in the note creation dialog\"}}\n  index = env_op.find_element(**kwargs)\n  assert index != -1, \"Element not found!\" # Check whether the corresponding element exists.\n  env_op.input_text('ordered_items', 1, True)\n  ...\n",
    "example_usage": "function_name(file_name=ordered_items, file_suffix=md, content=\"Apple\\nDog\\nPotato\")",
    "conclusion": "..."
  }
}
""")


OUTPUT_FORMAT_TOOL_USAGE = ("""
[Available Tools]
You may call:
- fetch_info: Fetch screenshot and xml/html of a specific step in the trajectory.

[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: 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 assertions for error detection; avoid internal try/except blocks.
- Do not add unnecessary validations; only verify essential conditions that are critical for the task.
- Use `env_op.wait()` sparingly; frequent waiting is unnecessary.
- Remember to use `env_op.stop()` or `env_op.answer()` to stop the RPA function when the task is completed.
- Do not check permissions in RPA code; permission handling is managed outside the RPA execution phase.

[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": "...",
    "parameters": "- file_name (str): The name of the file to be created in the Markor app.\n- file_suffix (str): The type of the file (such as md, txt).\n- content (str): The content to be written into the file.",
    "rpa_description": "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.",
    "rpa_code": "def function_name(file_name: str, file_suffix: str, content: str):\n  # Step 1: Open the Markor app\n  env_op.open_app('Markor')\n  ...\n  # Step 3: Find the input field for the file name in the note creation dialog\n  kwargs = {{\"text\": \"my_note\", \"actions\": [\"input_text\"], \"target_description\": \"Input field for the file name in the note creation dialog\"}}\n  index = env_op.find_element(**kwargs)\n  assert index != -1, \"Element not found!\" # Check whether the corresponding element exists.\n  env_op.input_text('ordered_items', 1, True)\n  ...\n",
    "example_usage": "function_name(file_name=ordered_items, file_suffix=md, content=\"Apple\\nDog\\nPotato\")",
    "conclusion": "..."
  }
}
""")

SKILL_BUILDING_PROMPT_TEMPLATE = """
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}

{action_space}

{shell_guidance}

[RPA Code Interaction Rules]
- Always use `env_op.open_app()` to launch apps.
- Ensure the index for each UI action is visible in both the UI list and screenshot.
- Respect supported actions in `actions`.
- Operate only on the exact target file or UI element—never on visually similar alternatives.
- Use long-press and selection bar for text selection.
- Avoid unnecessary or irrelevant actions; focus only on steps critical to the task.
"""

USER_PROMPT_TEMPLATE = """
[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[models.ReActTraj]] = None,
  pre_rpa_exec_traj: Optional[models.RPAExecTraj] = None,
  fetched_info: dict = None,
  use_tool: bool = False,
  rpa_builder_conclusion: str = '',
  action_space_prompt: Optional[str] = None,
  agent_name: Optional[str] = None,
  encountered_task_goals: Optional[list[str]] = None,
  enable_shell_action: bool = False,
) -> dict[str, str]:
  fetched_info_str = ''
  if fetched_info:
    step_n = fetched_info.get('step_n', 'N/A')
    traj_id = fetched_info.get('traj_id', 'N/A')
    ui_content_full_dict = fetched_info.get('ui_content_full_dict', '')
    screenshot_error = fetched_info.get('screenshot_error', '')
    from autorpa.utils import agent_utils
    fetched_info_str = (
      f"# Here is the ui info of step {step_n} in {traj_id} you fetched:\n"
      + agent_utils._generate_ui_elements_description_str(ui_content_full_dict or [])
    )
    if screenshot_error:
      fetched_info_str += f"\n\n# Screenshot fetch failed (non-fatal):\n{screenshot_error}\n"
  
  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.final_success_bool:
      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:
          from autorpa.utils import agent_utils
          traj_str += '### Initial UI Content:\n' + agent_utils._generate_ui_elements_description_str(getattr(step_info.exec_step_info, "before_ui_content_full_dict", None) or []) + '\n'
        if step_info.exec_step_info.is_screen_changed or \
          'stop(' in (step_info.soft_coded_action or step_info.hard_coded_action) or \
          'answer(' in (step_info.soft_coded_action or step_info.hard_coded_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.soft_coded_action or "").strip()
            hard_action = (step_info.hard_coded_action or "").strip()
            if soft_action != hard_action and hard_action:
              additional_info = f'\n### Original action: {step_info.hard_coded_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.hard_coded_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.reflection:
        traj_str += f"### Reflection: {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  # failed_react_traj has no detail by default; detail can be fetched 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.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_conclusion}\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.traj:
      from autorpa.utils import agent_utils
      pre_exec_str += f"\n\n## UI elements list at the point of code execution failure:\n{agent_utils._generate_ui_elements_description_str(pre_rpa_exec_traj.traj[-1].after_ui_content_full_dict or [])}"
    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.hard_coded_action or "").strip()
          if soft_action != hard_action and hard_action:
            additional_info = f'\n### Original action: {step_info.hard_coded_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.hard_coded_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)"
  
  # Get action space prompt (uses agent_name or default if not provided)
  action_space = get_builder_action_prompt(
    action_space_prompt=action_space_prompt,
    agent_name=agent_name
  )
  
  # Check if trajectories contain shell actions
  has_shell_in_trajs = False
  # Check react_trajs
  for react_traj in (react_trajs or []):
    for step_info in (react_traj.traj or []):
      action_code = step_info.soft_coded_action or step_info.hard_coded_action or ""
      if 'shell(' in action_code or 'env_op.shell(' in action_code:
        has_shell_in_trajs = True
        break
    if has_shell_in_trajs:
      break
  
  # Check pre_rpa_exec_traj
  if not has_shell_in_trajs and pre_rpa_exec_traj:
    # Check action_history
    for action_str in (pre_rpa_exec_traj.action_history or []):
      if 'shell(' in action_str or 'env_op.shell(' in action_str:
        has_shell_in_trajs = True
        break
    # Check fix_react_traj
    if not has_shell_in_trajs and pre_rpa_exec_traj.fix_react_traj:
      for fix_traj in pre_rpa_exec_traj.fix_react_traj:
        for step_info in (fix_traj.traj or []):
          action_code = step_info.soft_coded_action or step_info.hard_coded_action or ""
          if 'shell(' in action_code or 'env_op.shell(' in action_code:
            has_shell_in_trajs = True
            break
        if has_shell_in_trajs:
          break
  
  # Add shell guidance if:
  # 1. enable_shell_action is True (autorpa supports shell), OR
  # 2. trajectories contain shell actions (even if enable_shell_action=False)
  shell_guidance = SHELL_ACTION_GUIDANCE if (enable_shell_action or has_shell_in_trajs) else ""
  
  system_prompt = SKILL_BUILDING_PROMPT_TEMPLATE.format(
    output_format=output_format,
    action_space=action_space,
    shell_guidance=shell_guidance,
  )

  user_prompt = USER_PROMPT_TEMPLATE.format(
    encountered_tasks=encountered_tasks_str,
    failed_react_traj=failed_react_traj,
    successful_react_traj=successful_react_traj,
    pre_exec_str=pre_exec_str,
    fetched_info=fetched_info_str,
    task_type=task_type,
    task_template=task_template,
  )
  
  return {
    'system': system_prompt,
    'user': user_prompt
  }