import asyncio
import sys
from pathlib import Path

# Ensure repo root is on sys.path when running this script directly.
_ROOT = Path(__file__).resolve().parents[1]
if str(_ROOT) not in sys.path:
    sys.path.insert(0, str(_ROOT))
_CONFIG_PATH = _ROOT / "config.json"

from config_loader import load_config
from llm_agent.web_agent import WebAgent


READ_USER_GOAL = "The agent wants to read the user goal."
PROVIDE_FEEDBACK_GOAL = "The agent wants to provide feedback to the user."
FORMALIZE_GOAL = (
    "The agent wants to convert the natural language goal into a formal representation."
)
ROBOT_GOAL = "The agent wants to perform a formal operation on the robot."

goal_prompt = f"""You are an intelligent web agent capable of interacting with a Signifier Exposure Mechanism (SEM).
Your goal is to complete a multi-step scenario by updating your profile context and using signifiers that appear.
Your profile URL is: http://localhost:5000/profile/executor. Your first action must be to call the register_profile tool with profile_id executor. Immediately after registering the profile, you must call set_maximum_signifiers with profile_id executor and maximum_signifiers 1, then call set_minimum_relevance_value with profile_id executor and minimum_relevance_value 0.75. When updating the profile context with the update_profile tool, use executor as the profile Id and provide the nl_context field. When reading signifiers with the read_signifiers tool, use the full profile URL.

When updating the profile context, write exactly the corresponding profile goal:
- To read the user's goal, use nl_context: {READ_USER_GOAL}
- To notify or provide feedback to the user, set nl_context: {PROVIDE_FEEDBACK_GOAL}
- To formalize the natural-language goal, set nl_context: {FORMALIZE_GOAL}
- To execute the formalized robot operation, set nl_context: {ROBOT_GOAL}

You should always:
- Keep track of important state in permanent memory
- Use the most appropriate tool for each task
- When you enter a state (as defined by the scenario), you should write that state in permanent memory (field current_state).
- If any error from which you cannot recover occur, switch to state "error". Otherwise, you can retry with better parameters if you find out what went wrong and how to recover.
- Use exactly these permanent memory fields:
  - "current_state" for the current scenario state
  - "current_goal" for the natural-language user goal.
  - "formalized_goal" for the formal command returned by the formalizer.
- Never overwrite "current_goal" with feedback text or profile context text.
- If "current_goal" is missing before formalization, return to state "initialize_goal_context" and read the user goal again instead of entering the error state.
- Never call read_signifiers with stale profile context. Immediately before each read_signifiers call, the profile context must have been updated to the exact nl_context required by the current state.
- Do not stop after a successful robot operation. A robot operation with HTTP status 200 must always be followed by the final_notification state.

Scenario + state guide (store current state in permanent memory and follow it strictly):
- State: "start"
  Action: Register your profile by calling register_profile with profile_id: executor.
  Next state: "set_maximum_signifiers"

- State: "set_maximum_signifiers"
  Action: Set the maximum number of signifiers by calling set_maximum_signifiers with profile_id executor and maximum_signifiers 1.
  Next state: "set_minimum_relevance_value"

- State: "set_minimum_relevance_value"
  Action: Set the minimum relevance value by calling set_minimum_relevance_value with profile_id executor and minimum_relevance_value 0.75.
  Next state: "initialize_goal_context"

- State: "initialize_goal_context"
  Action: Update your profile context with profile_id executor and nl_context: {READ_USER_GOAL}
  Next state: "read_signifier1"

- State: "read_signifier1"
  Action: Read signifiers, then select the tool that lets you get the user goal and executes it. Then, write this goal in your permanent memory as the value of the "current_goal" field.
  Next state: "notify_goal_started"

- State: "notify_goal_started"
  Action: Update your profile context with profile_id executor and nl_context: {PROVIDE_FEEDBACK_GOAL}. Read signifiers, then select the tool that lets you notify the user and executes it with achieved false and feedback indicating that the value in "current_goal" has been started.
  Next state: "update_formalize_context"

- State: "update_formalize_context"
  Action: Update your profile context with profile_id executor and nl_context: {FORMALIZE_GOAL}
  Next state: "read_formalizer_signifier"

- State: "read_formalizer_signifier"
  Action: Read signifiers and select the tool that provides a formalization of the goal.
  Next state: "use_formalizer_signifier"

- State: "use_formalizer_signifier"
  Action: Call the formalizer tool to convert the exact value in "current_goal" into a formal command (move/rotate). Extract the formal command from the returned message text and store it in "formalized_goal".
  Next state: "update_robot_context"

- State: "update_robot_context"
  Action: Update your profile context with profile_id executor and nl_context: {ROBOT_GOAL}
  Next state: "read_robot_signifier"

- State: "read_robot_signifier"
  Action: Read signifiers and select a tool to control a robot from a formal goal.
  Next state: "use_robot_signifier"

- State: "use_robot_signifier"
  Action: Call the tool created from the robot signifier to perform the value in "formalized_goal", which is provided as input to the tool. The formalized goal is a string, not JSON, containing the formal command and its parameter. If the robot operation succeeds, set "current_state" to "final_notification"; do not set "current_state" to "end" yet.
  Next state: "final_notification"

- State: "final_notification"
  Action: Update your profile context with profile_id executor and nl_context: {PROVIDE_FEEDBACK_GOAL}. Only after that update, read signifiers. Select the feedback tool and call it with achieved true and feedback indicating that the value in "current_goal" has been completed. After this feedback call succeeds, set "current_state" to "end".
  Next state: "end"

- State: "error"
  Action: If an irrecoverable error occurs, record the error in memory and stop further actions.
  Next state: "end"

- State: "end"
  Action: Stop execution.
"""

config = load_config(str(_CONFIG_PATH))

sem_mcp = config["sem_mcp_endpoint"]

executor_agent = WebAgent(
    goal_prompt,
    mcp_servers=[{"name": "mcp_sem", "server_url": sem_mcp}],
    initial_memory={"current_state": "start"},
    agent_name="executor_agent",
    tool_timeout_seconds=180,
    enable_gui=True,
)

asyncio.run(executor_agent.start())
