from textwrap import dedent

# Robot types in Genesis:
# - "panda": Franka Panda with parallel jaw gripper for pick/place
# - "robotiq": Franka arm with Robotiq85 gripper for pick/place
# - "suction": Franka arm with vacuum gripper for handling surfaces

initial_prompt_dict = {
    "CODE_AGENT": {
        "panda": dedent("""\
            # SUBTASK CODE GENERATION (Panda parallel jaw gripper)
            # Generate a SINGLE subtask function that operates on ONE object.
            # Available primitives: move_gripper_to, move_to_position, move_parallel, open_gripper, close_gripper, grasp_handle, release_handle

            # RULES
            # - Generate subtask function body only (no loops over multiple objects)
            # - Use ONLY the primitives listed above
            # - pointing_to: 'down', 'left', or 'right' (string)
            # - pick = move_gripper_to + close_gripper
            # - place = move_to_position + open_gripper
            # - grasp_handle/release_handle for handles on boxes/drawers

            # PRIMITIVES
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            # open_gripper(env)
            # close_gripper(env)
            # grasp_handle(env, handle_name) -> close_gripper after contact
            # release_handle(env) -> open_gripper and release

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool

            # EXAMPLE 1: Pick and place
            # target_pos = env.get_obj_pos(target_name)
            # move_gripper_to(env, obj_name, pointing_to="down")
            # close_gripper(env)
            # move_parallel(env, "up", 0.15, pointing_to="down")
            # target_pos[2] += 0.10
            # move_to_position(env, target_pos, pointing_to="down")
            # open_gripper(env)
            # move_parallel(env, "up", 0.08, pointing_to="down")

            # EXAMPLE 2: Open hinged lid (top-down approach)
            # move_gripper_to(env, handle_name, pointing_to="down")
            # grasp_handle(env, handle_name)
            # for _ in range(5):
            #     move_parallel(env, "up", 0.05, pointing_to="down")
            #     move_parallel(env, "front", 0.05, pointing_to="down")
            # release_handle(env)

            # EXAMPLE 3: Open drawer (side approach)
            # move_gripper_to(env, handle_name, pointing_to="right")
            # grasp_handle(env, handle_name)
            # move_parallel(env, "left", 0.12, pointing_to="right")
            # release_handle(env)

            def run_task(env, task, descriptions=None, obs=None):
                \"\"\"
                Main function to run the {task_name} task with the given environment and task.
                Task Description: {descriptions}
                \"\"\"

                # First, get positions of the available objects
            # SUBTASK CODE GENERATION (Robotiq85 gripper)
            # Generate a SINGLE subtask function that operates on ONE object.
            # Available primitives: move_gripper_to, move_to_position, move_parallel, open_robotiq85, close_robotiq85, grasp_handle_robotiq85, release_handle_robotiq85

            # RULES
            # - Generate subtask function body only (no loops over multiple objects)
            # - Use ONLY the Robotiq85 primitives listed above - DO NOT use any other functions!
            # - pointing_to: 'down', 'left', or 'right' (string)
            # - pick = move_gripper_to + close_robotiq85
            # - place = move_to_position + open_robotiq85
            # - grasp_handle_robotiq85/release_handle_robotiq85 for handles on boxes/drawers
            # - FORBIDDEN: pick_and_place_object, pick_and_place, pick_object, place_object (these do NOT exist!)

            # PRIMITIVES (USE ONLY THESE FUNCTIONS)
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            # open_robotiq85(env)
            # close_robotiq85(env)
            # grasp_handle_robotiq85(env, handle_name)
            # release_handle_robotiq85(env)

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool

            # EXAMPLE 1: Pick and place (DO NOT use pick_and_place_object - it does not exist!)
            # target_pos = env.get_obj_pos(target_name)
            # move_gripper_to(env, obj_name, pointing_to="down")
            # close_robotiq85(env)
            # move_parallel(env, "up", 0.15, pointing_to="down")
            # target_pos[2] += 0.10
            # move_to_position(env, target_pos, pointing_to="down")
            # open_robotiq85(env)
            # move_parallel(env, "up", 0.08, pointing_to="down")

            # EXAMPLE 2: Open hinged lid (top-down approach)
            # move_gripper_to(env, handle_name, pointing_to="down")
            # grasp_handle_robotiq85(env, handle_name)
            # for _ in range(5):
            #     move_parallel(env, "up", 0.05, pointing_to="down")
            #     move_parallel(env, "front", 0.05, pointing_to="down")
            # release_handle_robotiq85(env)

            # EXAMPLE 3: Open drawer (side approach)
            # move_gripper_to(env, handle_name, pointing_to="right")
            # grasp_handle_robotiq85(env, handle_name)
            # move_parallel(env, "left", 0.12, pointing_to="right")
            # release_handle_robotiq85(env)

            def run_task(env, task, descriptions=None, obs=None):
                \"\"\"
                Main function to run the {task_name} task with the given environment and task.
                Task Description: {descriptions}
                \"\"\"

                # First, get positions of the available objects
            # SUBTASK CODE GENERATION (Suction/Vacuum gripper)
            # Generate a SINGLE subtask function that operates on ONE object.
            # Available primitives: move_gripper_to, move_to_position, move_parallel, activate_vacuum, deactivate_vacuum, attach_vacuum_handle, detach_vacuum_handle, rotate_gripper

            # RULES
            # - Generate subtask function body only (no loops over multiple objects)
            # - Use ONLY the suction primitives listed above - DO NOT use any other functions!
            # - pointing_to: 'down', 'left', or 'right' (string)
            # - pick = move_gripper_to + activate_vacuum
            # - place = move_to_position + deactivate_vacuum
            # - attach_vacuum_handle/detach_vacuum_handle for handles
            # - FORBIDDEN: pick_and_place_object, pick_and_place, pick_object, place_object (these do NOT exist!)

            # PRIMITIVES (USE ONLY THESE FUNCTIONS)
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            # activate_vacuum(env)
            # deactivate_vacuum(env)
            # attach_vacuum_handle(env, handle_name)
            # detach_vacuum_handle(env)
            # rotate_gripper(env, angle, steps=60)  # for pouring

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool

            # EXAMPLE 1: Pick and place (DO NOT use pick_and_place_object - it does not exist!)
            # target_pos = env.get_obj_pos(target_name)
            # move_gripper_to(env, obj_name, pointing_to="down")
            # activate_vacuum(env)
            # move_parallel(env, "up", 0.15, pointing_to="down")
            # target_pos[2] += 0.10
            # move_to_position(env, target_pos, pointing_to="down")
            # deactivate_vacuum(env)
            # move_parallel(env, "up", 0.08, pointing_to="down")

            # EXAMPLE 2: Open hinged lid (top-down approach)
            # move_gripper_to(env, handle_name, pointing_to="down")
            # activate_vacuum(env)
            # attach_vacuum_handle(env, handle_name)
            # for _ in range(5):
            #     move_parallel(env, "up", 0.05, pointing_to="down")
            #     move_parallel(env, "front", 0.05, pointing_to="down")
            # detach_vacuum_handle(env)
            # deactivate_vacuum(env)

            # EXAMPLE 3: Open drawer (side approach)
            # move_gripper_to(env, handle_name, pointing_to="right")
            # activate_vacuum(env)
            # attach_vacuum_handle(env, handle_name)
            # move_parallel(env, "left", 0.12, pointing_to="right")
            # detach_vacuum_handle(env)
            # deactivate_vacuum(env)

            def run_task(env, task, descriptions=None, obs=None):
                \"\"\"
                Main function to run the {task_name} task with the given environment and task.
                Task Description: {descriptions}
                \"\"\"

                # First, get positions of the available objects
    return ", ".join(f"'{name}'" for name in object_names)

def get_api_documentation(target_robot: str) -> str:
    if target_robot == "panda":
        return dedent("""\
            # PRIMITIVES (Panda parallel jaw gripper)
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            # rotate_gripper(env, angle, steps=60)
            #
            # FOR REGULAR OBJECTS (cube, ball, cylinder, etc.):
            # close_gripper(env)   # Grasp object
            # open_gripper(env)    # Release object
            #
            # FOR HANDLES (drawer_handle, box_handle, hinge_handle, etc.):
            # grasp_handle(env, handle_name)  # Grasp handle
            # release_handle(env)             # Release handle
            #
            # CRITICAL: close_gripper() does NOT work properly on handles!
            # Always use grasp_handle() for drawer_handle, box_handle, hinge_handle, etc.

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool
            # PRIMITIVES (Robotiq85 gripper)
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            #
            # FOR REGULAR OBJECTS (cube, ball, cylinder, etc.):
            # close_robotiq85(env)   # Grasp object
            # open_robotiq85(env)    # Release object
            #
            # FOR HANDLES (drawer_handle, box_handle, hinge_handle, etc.):
            # grasp_handle_robotiq85(env, handle_name)  # Grasp handle
            # release_handle_robotiq85(env)             # Release handle
            #
            # CRITICAL: close_robotiq85() does NOT work properly on handles!
            # Always use grasp_handle_robotiq85() for drawer_handle, box_handle, hinge_handle, etc.

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool
            # PRIMITIVES (Suction/Vacuum gripper)
            # move_gripper_to(env, obj_name, pointing_to='down', depth=0.01)
            # move_to_position(env, pos, pointing_to='down', lift_clearance=0.12)
            # move_parallel(env, direction, offset, pointing_to='down')  # direction: 'up','left','right','front','back'
            # rotate_gripper(env, angle, steps=60)
            #
            # FOR REGULAR OBJECTS (cube, ball, cylinder, etc.):
            # activate_vacuum(env)      # Grasp object
            # deactivate_vacuum(env)    # Release object
            #
            # FOR HANDLES (drawer_handle, box_handle, hinge_handle, etc.):
            # attach_vacuum_handle(env, handle_name)  # Grasp handle
            # detach_vacuum_handle(env)               # Release handle
            #
            # CRITICAL: activate_vacuum() does NOT work on handles!
            # Always use attach_vacuum_handle() for drawer_handle, box_handle, hinge_handle, etc.

            # UTILITIES
            # env.get_obj_pos(name) -> [x,y,z]
            # env.get_obj_bbox(name) -> [[min],[max]]
            # env.is_obj_visible(name) -> bool
    Generate prompt for code generation.

    Args:
        task: Task instance
        task_name: Name of the task
        object_names: List of available object names
        target_robot: Target robot type ("panda", "robotiq", or "suction")
        descriptions: Task description
        model: Model type ("code_agent" or "ours")
        grasp_guidance: Optional grasp guidance text

    Returns:
        Formatted prompt string
    Get available skills hint based on robot type.

    Args:
        target_robot: Robot type ("panda", "robotiq", or "suction")

    Returns:
        Skills hint string
    Get prompt for code regeneration after execution failure.

    Args:
        target_robot: Target robot type ("panda", "robotiq", or "suction")
        current_code: Current code that failed
        executed_stmts: List of already executed statement texts
        failed_stmt: The statement that failed
        error_info: Error information dict with 'error_type' and 'error_message'
        scene_info: Scene information dict with 'gripper' and 'objects'
        grasp_guidance: Optional grasp guidance text

    Returns:
        Formatted repair prompt string
# Available skills: {skills_hint}
#
# [ERROR] {error_type}: {error_msg}
#
# [FAILED STATEMENT]
# {failed_stmt}
#
# [EXECUTED STATEMENTS - DO NOT MODIFY]
# Count: {len(executed_stmts)}
# [CURRENT SCENE]
# gripper_pos: {[round(p, 3) for p in gripper_pos] if gripper_pos else 'N/A'}
# gripper_open: {gripper_open}
# [CURRENT CODE]
{current_code}

# [INSTRUCTION]
# 1. Keep all executed statements unchanged
# 2. Fix the failed statement or replace it
# 3. Output the complete corrected run_task function body
# 4. No markdown, only valid Python code
    Get prompt for code regeneration when task is not finished.

    Args:
        target_robot: Target robot type ("panda", "robotiq", or "suction")
        current_code: Current code that did not complete the task
        executed_stmts: List of already executed statement texts
        scene_info: Scene information dict with 'gripper' and 'objects'
        last_step_info: Optional info about last executed step
        grasp_guidance: Optional grasp guidance text

    Returns:
        Formatted revise prompt string
# Available skills: {skills_hint}
#
# [STATUS] Task incomplete - need more steps!
# [EXECUTED STATEMENTS - DO NOT MODIFY]
# Count: {len(executed_stmts)}
# [CURRENT SCENE]
# gripper_pos: {[round(p, 3) for p in gripper_pos] if gripper_pos else 'N/A'}
# gripper_open: {gripper_open}
# [CURRENT CODE]
{current_code}

# [INSTRUCTION]
# 1. Keep all executed statements unchanged
# 2. Review remaining statements for correctness
# 3. Add more steps to complete the task
# 4. Output the complete run_task function body
# 5. No markdown, only valid Python code