import json
from typing import Dict, List, Any, Optional

class LLMPlanner:
    """
    Handles all interactions with the LLM for planning, time estimation, and replanning.
    """

    def __init__(self, api_key: str, model_name: str = "DUMMY_KEY_FOR_MANUAL_INPUT"):
        self.api_key = api_key
        self.model_name = model_name
        self.response_file_counter = 1
        print(f"[LLMPlanner] Initialized with model: {self.model_name}")

    def _get_llm_response_from_file(self, prompt: str) -> Dict[str, Any]:
        """
        A manual wrapper to get LLM responses for offline testing.
        It prints a prompt to the console and waits for the user to manually save
        the corresponding LLM output into a sequentially numbered JSON file.
        """


        response_filename = f"llm_response_{self.response_file_counter}.json"
        
        print("\n" + "-"*50)
        print(f"ACTION REQUIRED: Please save the LLM's response to '{response_filename}'")
        print("PROMPT:")
        print(prompt)
        print("-" * 50)
        
        input(f"After saving the file, press Enter to continue...")
        
        try:
            with open(response_filename, 'r', encoding='utf-8') as f:
                data = json.load(f)
            self.response_file_counter += 1
            return data
        except FileNotFoundError:
            print(f"ERROR: File '{response_filename}' not found. Please try again.")
            return self._get_llm_response_from_file(prompt)
        except json.JSONDecodeError:
            print(f"ERROR: Invalid JSON in '{response_filename}'. Please check the file and try again.")
            return self._get_llm_response_from_file(prompt)

    
    
    def decompose_task(self, goal: str, scene_info: Dict[str, Any]) -> Dict[str, Any]:
        """[LLM 1] Decomposes a high-level goal into a list of sub-tasks and their dependencies."""
        print("\n[LLMPlanner-Step1] Task Decomposition LLM")
        prompt_lines = []
        prompt_lines.append("# ROLE & GOAL:")
        prompt_lines.append("You are an expert AI planner specializing in human-robot collaboration within a simulated kitchen environment.")
        prompt_lines.append("Your task is to decompose a high-level goal into a sequence of sub-tasks and define each dependency.")
        
        prompt_lines.append("\n# CONTEXT:")
        prompt_lines.append("The JSON object defines the entire kitchen scene, including all available objects, their properties, and their initial locations. You must use objects and receptacles found in this JSON.")
        prompt_lines.append("You must perform the following actions:")
        prompt_lines.append('[“pickup_<object_type>",“put_<object_type>_on_<receptacle_type>",“put_<object_type>_in_<receptacle_type>", “open_<object_type>", “close_<object_type>", “slice_<object_type>", "Heat_<object_type>_by_<receptacle_type>", "Boil_<object_type>_by_<receptacle_type>"]')
        prompt_lines.append("Note that, the placeholders `<object_type>` and `<receptacle_type>` refer to a single, specific object or receptacle type.")
        prompt_lines.append("If a similar action is needed again (e.g., opening a microwave twice), append a number to distinguish it (e.g., `open_Microwave_1`, `open_Microwave_2`).")
        
        prompt_lines.append("\n# INSTRUCTION:")
        prompt_lines.append(f'The high-level goal is "{goal}".')
        prompt_lines.append('\nWhen you define the dependencies, follow the guidelines:')
        prompt_lines.append('\n1. Identify Independent Branches: Analyze the goal to find sub-processes that can be completed independently. Structure these as separate dependency branches that can be executed simultaneously by different agents.')
        prompt_lines.append("2. Minimize the Critical Path: Do NOT create a long, single chain of dependencies. It is important to make this sequence as short as possible by branching out all non-dependent tasks.")
        prompt_lines.append("3. Delay Dependencies: Only create a dependency between two tasks if one is strictly required for the other to begin. Avoid creating dependencies between the parallel branches until the final combination step.")

        prompt_lines.append("\n# OUTPUT FORMAT:")
        prompt_lines.append("The output must be a single, valid JSON object. Do not add any explanations or text outside of the JSON object. You MUST use the following keys and structure precisely:")
        prompt_lines.append("""
{
  "tasks": [
    "sub_task_1",
    "sub_task_2",
    ...
  ],
  "dependencies": [
    ["sub_task_1", "sub_task_2"],
    ...
  ]
}""")
        
        prompt_lines.append("\n# EXAMPLE:")
        prompt_lines.append('Goal: "Make a cup of coffee"')
        prompt_lines.append("""
{
  "tasks": [
    "pickup_Mug",
    "put_Mug_in_CoffeeMachine",
    "toggle_on_CoffeeMachine",
    "toggle_off_CoffeeMachine"
  ],
  "dependencies": [
    ["pickup_Mug", "put_Mug_in_CoffeeMachine"],
    ["put_Mug_in_CoffeeMachine", "toggle_on_CoffeeMachine"],
    ["toggle_on_CoffeeMachine", "toggle_off_CoffeeMachine"]
  ]
}""")
        prompt_lines.append("\n* NOTE: DO NOT OUTPUT ANYTHING EXTRA OTHER THAN WHAT HAS BEEN SPECIFIED")

        prompt = "\n".join(prompt_lines)
        decomposed_plan = self._get_llm_response_from_file(prompt)
        print("[LLMPlanner-Step1] Decomposed plan loaded successfully.")
        return decomposed_plan

    def estimate_eet(self, tasks: List[str]) -> Dict[str, Any]:
        """
        [LLM 2] Takes a list of sub-tasks and generates physical data for the optimizer,
        such as estimated execution times (EET), agent capabilities, and navigation times.
        """

        print(f"\n[LLMPlanner-Step2] Estimating EET for {len(tasks)} tasks...")

        ### Physics-guided LLM ###
        prompt_lines = []
        
        prompt_lines.append("# ROLE & GOAL:")
        prompt_lines.append("You are a robotics assistant specializing in motion planning and task time estimation for a mobile manipulator.")
        prompt_lines.append("Your goal is to analyze a list of sub-tasks and a scene definition JSON to produce all realistically and conservatively estimated execution time and capability data for an optimizer.")

        prompt_lines.append("\n# CONTEXT:")
        prompt_lines.append("The JSON object defines the entire kitchen scene, including all available objects, their properties, and their initial locations.")
        prompt_lines.append("Given sub-tasks:")
        prompt_lines.append(json.dumps(tasks, indent=4))

        prompt_lines.append("\n# INSTRUCTION:")
        prompt_lines.append("Follow these steps precisely to gather all the necessary data.")
        prompt_lines.append("\n## 1. Identify All Necessary Locations")
        prompt_lines.append("- For each sub-task, determine its primary location from the environmental information.")
        prompt_lines.append("- Compile a list of all unique locations involved in the plan.")
        prompt_lines.append("\n## 2. Estimate Time Components (Navigation and Manipulation)")
        prompt_lines.append("- Based on the plan and locations, you will now estimate two distinct types of execution time:")
        prompt_lines.append("\n### 2-A. Navigation Time:")
        prompt_lines.append("- This is the time it takes for the robot's mobile base to travel between the locations")
        prompt_lines.append('- Use the coordinates from "navigation_targets", the robot\'s specifications and physical considerations.')
        prompt_lines.append("- This data will populate the `nav_time` key in the JSON output.")
        prompt_lines.append("\n### 2-B. Manipulation Time:")
        prompt_lines.append("- This is the time it takes to perform the action at the location (e.g., arm movement, grasping, slicing, toggling).")
        prompt_lines.append("- Estimate this for each sub-task in the plan using the robot's specifications and physical considerations.")
        prompt_lines.append("- This data will populate the `mani_time` key in the JSON output.")
        prompt_lines.append("\n### 2-C. Cooking Time:")
        prompt_lines.append("- For cooking-related actions like Heat or Boil, the time estimate is not only based on robot movement but also on the physical process of cooking.")
        prompt_lines.append("- Use your commonsense knowledge to provide a realistic time estimate for the food to be cooked.")
        prompt_lines.append("\n## 3: Determine Capability (`capability`)")
        prompt_lines.append("- For each sub-task, determine if the robot and human can perform it. This will populate the `capability` key.")
        prompt_lines.append("- You must realistically consider the robot's physical limitations such as human-level dexterity and complex judgment related to cooking.")
        
        prompt_lines.append("\n# ROBOT SPECIFICATIONS")
        prompt_lines.append("- Mobile_Base (Clearpath Robotics's Husky):")
        prompt_lines.append("  - max_linear_velocity: 1.0 m/s")
        prompt_lines.append("  - max_angular_velocity: 120 deg/s")
        prompt_lines.append("  - linear_acceleration: 0.5 m/s^2")
        prompt_lines.append("- Manipulator_Arm (Universal Robots's UR3):")
        prompt_lines.append("  - dof: 6-DOF")
        prompt_lines.append("  - max_reach: 0.8 m")
        prompt_lines.append("  - gripper_open_time: 0.7 s")
        prompt_lines.append("  - gripper_close_time: 0.7 s")
        prompt_lines.append("- Overheads:")
        prompt_lines.append("  - navigation_overhead_time: 1.0 s (for perception and planning before moving)")
        prompt_lines.append("  - manipulation_overhead_time: 2.0 s (for perception and fine-tuning before acting)")

        prompt_lines.append("\n# PHYSICAL CONSIDERATIONS")
        prompt_lines.append("\n## 1. Navigation")
        prompt_lines.append("- Consider the phases of motion: acceleration, cruising, and deceleration. The robot may not reach its maximum speed if the travel distance is short.")
        prompt_lines.append("- Account for both rotation and translation time, plus the specified navigation overhead.")
        prompt_lines.append("- Rotation Time: t = angle / angular_velocity (where angle is in degrees, and angular_velocity is in deg/s)")
        prompt_lines.append("- Uniform Acceleration Motion: v = v0 + a * t, s = v0 * t + 0.5 * a * t^2 (where s is distance, v is final velocity, v0 is initial velocity, a is acceleration, and t is time)")
        prompt_lines.append("- Distance to Reach Max Velocity: To determine if the robot will have a cruising phase, you can calculate the minimum distance required to accelerate to max velocity (v) from rest using s = v^2 / (2 * a).")
        prompt_lines.append("\n## 2. Manipulation")
        prompt_lines.append("- Manipulation includes multiple stages: pre-shaping Gripper, arm movement to the target, fine-tuning the pose, gripper actuation (grasping or releasing), Secure Lift and retreating. Time estimate must account for all these phases.")
        prompt_lines.append("- Include the specified manipulation_overhead_time for perception and planning.")
        prompt_lines.append("- Arm Movement Time: Estimate the time for the arm to move by approximating the distance the end-effector travels and dividing by an average speed. t = d / v_avg (where d is the estimated travel distance of the end-effector. Since v_avg is not specified, assume a reasonable, conservative value like 0.25 m/s.)")
        prompt_lines.append("- Fixed Time Actions: Always add the specified time for gripper actuation (gripper_open_time or gripper_close_time).")
        prompt_lines.append("- Fine-Tuning: Add a fixed time for the fine adjustment, such as 1.0 to 1.5 seconds.")
        
        prompt_lines.append("\n# OUTPUT FORMAT")
        prompt_lines.append("The output must be a single, valid JSON object. Do not add any explanations or text outside of the JSON object. You MUST use the following keys and structure precisely:")
        prompt_lines.append("""
{
  "capability": {
    "task_name_1": { "robot": 1(Yes) or 0 (No), "human": 1(Yes) or 0 (No) },
    "task_name_2": { "robot": 1(Yes) or 0 (No), "human": 1(Yes) or 0 (No) },
    "...": "..."
  },
  "mani_time": {
    "task_name_1": time(seconds),
    "task_name_2": time(seconds),
    "...": "..."
  },
  "nav_time": {
    "LocationA -> LocationB": time(seconds),
    "LocationB -> LocationC": time(seconds),
    "...": "..."
  }
}""")

        prompt_lines.append("\n# EXAMPLE")
        prompt_lines.append('This is an example for a goal "Make a salad" with tasks ["Pickup_Lettuce", "Slice_Lettuce"] and locations ["Fridge", "CounterTop"].')
        prompt_lines.append("""
{
  "capability": {
    "Pickup_Lettuce": { "robot": 1, "human": 1 },
    "Slice_Lettuce": { "robot": 0, "human": 1 }
  },
  "mani_time": {
    "Pickup_Lettuce": 6.8,
    "Slice_Lettuce": 25.0
  },
  "nav_time": {
    "Fridge|-02.10|+00.00|+01.09 -> CounterTop|-00.08|+01.15|00.00": 5.1,
    "CounterTop|-00.08|+01.15|00.00 -> Fridge|-02.10|+00.00|+01.09": 5.1
  }
}""")
        prompt_lines.append("\n* NOTE: DO NOT OUTPUT ANYTHING EXTRA OTHER THAN WHAT HAS BEEN SPECIFIED")

        prompt = "\n".join(prompt_lines)

        duration_data = self._get_llm_response_from_file(prompt)
        print("[LLMPlanner-Step2] EETs loaded successfully.")
        return duration_data
    

    def generate_replanning(self, interrupt_context: Dict[str, Any], current_plan: Dict[str, Any]) -> Dict[str, Any]:
        """LLM for SimManager when an interrupt flag (Goal change or PAUSE) occurs during execution."""

        print(f"\n[LLMPlanner-Step1-Replan] Re-planning due to interrupt.")
        

        reason = interrupt_context.get("reason", "Unknown interrupt")
        current_time = interrupt_context.get("current_time", 0.0)
        completed_tasks = interrupt_context.get("completed_tasks", [])
        active_tasks = interrupt_context.get("active_tasks", {})
        multiplier = interrupt_context.get("learned_human_multiplier")


        prompt_lines = []
        if 'goal change' in reason:     ## goal change
          prompt_lines.append("# ROLE & GOAL:")
          prompt_lines.append("You are an expert AI planner specializing in human-robot collaboration within a simulated kitchen environment.")
          prompt_lines.append("Your task is to decompose a high-level goal into a sequence of sub-tasks and define each dependency.")
          prompt_lines.append("\n# CONTEXT:")
          prompt_lines.append("The JSON object defines the entire kitchen scene, including all available objects, their properties, and their initial locations. You must use objects and receptacles found in this JSON.")
          prompt_lines.append("You must perform the following actions:")
          prompt_lines.append('[“pickup_<object_type>",“put_<object_type>_on_<receptacle_type>",“put_<object_type>_in_<receptacle_type>", “open_<object_type>", “close_<object_type>", “slice_<object_type>", “toggle_on_<object_type>”, “toggle_off_<object_type>”]')
          prompt_lines.append("Note that, the placeholders `<object_type>` and `<receptacle_type>` refer to a single, specific object or receptacle type.")
          prompt_lines.append("If a similar action is needed again (e.g., opening a microwave twice), append a number to distinguish it (e.g., `open_Microwave_1`, `open_Microwave_2`).")
          prompt_lines.append("\n# INSTRUCTION:")
          try:

              details = reason.split("'")[1]
              prompt_lines.append(f"The changed high-level goal is '{details}'")
          except IndexError:

              prompt_lines.append(f"The changed high-level goal is '{reason}'")
          prompt_lines.append("\nYour primary function is to decompose this goal into a list of all necessary sub-tasks and define the direct prerequisite dependencies between them. You must only focus on the logical sequence of actions.")
        else:     ## pause
          prompt_lines.append("# ROLE & GOAL:")
          prompt_lines.append("You are an AI planner for human-robot collaboration, and you need to modify an existing plan due to an interruption.")
          prompt_lines.append("Your task is to decompose a high-level goal into a sequence of sub-tasks and define each dependency.")
          prompt_lines.append("\n# CURRENT SITUATION")
          prompt_lines.append(f"- The current simulation time is t={current_time:.2f}s.")
          prompt_lines.append(f"- Reason for Re-planning: {reason}")
          if multiplier:
              prompt_lines.append("- A human performance multiplier between planned and actual duration has been learned and should be considered.")
              prompt_lines.append("- This multiplier is an indicator of the human's expertise. A higher value points to a more cautious or novice-like approach, whereas a lower value points to greater efficiency and skill.")

          prompt_lines.append("\n# TASK STATUS")
          prompt_lines.append(f"- Completed Tasks: {completed_tasks if completed_tasks else 'None'}")
          
          prompt_lines.append("- In-Progress Tasks:")
          if active_tasks:
              for task, info in active_tasks.items():
                  prompt_lines.append(f"  - {task} ({info['agent']}) started at t={info['start_time']:.2f}s.")
          else:
              prompt_lines.append("  - None")
              

          prompt_lines.append("\n# ORIGINAL PLAN SCOPE")
          original_tasks = current_plan.get("tasks", [])
          prompt_lines.append(f"- The full list of tasks in the original plan was: {original_tasks}")

          prompt_lines.append("\n# INSTRUCTION")
          prompt_lines.append("Based on the situation described above, generate a new plan for the remaining tasks.")
          prompt_lines.append('\nWhen you define the dependencies, follow the guidelines:')
          prompt_lines.append('\n1. Identify Independent Branches: Analyze the goal to find sub-processes that can be completed independently. Structure these as separate dependency branches that can be executed simultaneously by different agents.')
          prompt_lines.append("2. Minimize the Critical Path: Do NOT create a long, single chain of dependencies. It is important to make this sequence as short as possible by branching out all non-dependent tasks.")
          prompt_lines.append("3. Delay Dependencies: Only create a dependency between two tasks if one is strictly required for the other to begin. Avoid creating dependencies between the parallel branches until the final combination step.")

        prompt_lines.append("\n# OUTPUT FORMAT:")
        prompt_lines.append("The output must be a single, valid JSON object. Do not add any explanations or text outside of the JSON object. You MUST use the following keys and structure precisely:")
        prompt_lines.append("""
{
  "tasks": [
    "sub_task_1",
    "sub_task_2",
    ...
  ],
  "dependencies": [
    ["sub_task_1", "sub_task_2"],
    ...
  ]
}""")
        
        prompt_lines.append("\n# EXAMPLE:")
        prompt_lines.append('Goal: "Make a cup of coffee"')
        prompt_lines.append("""
{
  "tasks": [
    "pickup_Mug",
    "put_Mug_in_CoffeeMachine",
    "toggle_on_CoffeeMachine",
    "toggle_off_CoffeeMachine"
  ],
  "dependencies": [
    ["pickup_Mug", "put_Mug_in_CoffeeMachine"],
    ["put_Mug_in_CoffeeMachine", "toggle_on_CoffeeMachine"],
    ["toggle_on_CoffeeMachine", "toggle_off_CoffeeMachine"]
  ]
}""")
        prompt_lines.append("\n* NOTE: DO NOT OUTPUT ANYTHING EXTRA OTHER THAN WHAT HAS BEEN SPECIFIED")

        prompt = "\n".join(prompt_lines)
        

        replan_data = self._get_llm_response_from_file(prompt)
        print("[LLMPlanner-Step1-Replan] Re-decomposed plan loaded successfully.")
        return replan_data