from textwrap import dedent
from typing import List, Dict, Any, Optional

initial_prompt_dict = {
    "CODE_AGENT": {
        "panda": dedent("""\
            # MANDATORY: Use these EXACT objects and skills. NO placeholders!
            # MANDATORY OBJECTS (use EXACT names): {obj_list_str}
            # MANDATORY PRIMITIVE SKILLS ONLY (no other calls allowed): move, pick, place, push, open_gripper, close_gripper, align_to_quaternion, align_two_axes
            # OUTPUT: Only write valid Python code for run_skill body. No markdown. No comments.
            
            # RULES (IMPORTANT)
            # - pick/place internally move to target_pos. pick() moves then grips; place() moves then releases.
            # - push includes close_gripper() followed by move() to the target position internally.
            # - approach_axis must be a STRING in {{'x','y','z','-x','-y','-z'}} (do NOT use AXIS.* or enums).
            # - Use reasonable values: approach_distance ~ 0.05-0.15, timeout ~ 3.0-10.0.
            # - align_to_quaternion is NOT a default step. Use it ONLY when top-down approach is not possible.
            # - offset is applied when computing target_pos: target_pos = object_pos + [dx, dy, dz]
            #   Example: target_pos = pos + [0.0, 0.0, 0.2]  # 20cm above
            # - yaw_align MUST be either 'parallel' or 'perpendicular' (no other values allowed)

            # Primitive call shapes (reference)
            # pick:  obs, reward, done = pick(env, task, target_pos=[x,y,z], approach_axis='z', approach_distance=FLOAT, timeout=FLOAT)
            # place: obs, reward, done = place(env, task, target_pos=[x,y,z], approach_axis='z', approach_distance=FLOAT, timeout=FLOAT)
            # move:  obs, reward, done = move(env, task, target_pos=[x,y,z], timeout=FLOAT)  # rare; pick/place usually enough
            # push:  obs, reward, done = push(env, task, target_pos=[x,y,z], approach_axis='z', approach_distance=FLOAT, timeout=FLOAT)
            # grip:  obs, reward, done = open_gripper(env, task) / close_gripper(env, task)
            # align:
            #   q = obj.get_quaternion()
            #   obs, reward, done = align_to_quaternion(env, task, quaternion=q, yaw_align='parallel', approach_dir=APPROACH_DIR, timeout=FLOAT)
            # align_two_axes:
            #   obs, reward, done = align_two_axes(env, task, local_axes=('x','y'), world_axes=('z','y'), axis_dirs=(1,1), timeout=FLOAT)

            # Utility functions available (reference):
            # - get_bbox_sizes(obj): returns (sizes, bbox) where sizes=[dx,dy,dz]
            # - quat_mul(q1, q2): quaternion multiplication
            # - normalize_vector(v): normalize vector with error handling

            # Composition patterns (reference)
            # Example 1. pick then place
            # obj = Shape('object_name'); pos = obj.get_position()
            # target_pos = pos + [0.0, 0.0, 0.2]
            # obs, reward, done = pick(env, task, target_pos=pos, approach_axis='z', approach_distance=0.1, timeout=5.0)
            # obs, reward, done = place(env, task, target_pos=target_pos, approach_axis='z', approach_distance=0.1, timeout=5.0)

            # Example 2. sawyer_align_gripper then sawyer_pick (only when alignment needed)
            q = obj.get_quaternion()
            obs, reward, done = sawyer_align_gripper(env, task, approach_direction='down', reference_quat=q, yaw_mode='parallel')
            obs, reward, done = sawyer_pick(env, task, target_object=obj)
            
            # CRITICAL: Output ONLY executable Python code. NO placeholders like ____, FLOAT, etc.
            # Fill in ALL values with actual numbers and strings.

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

                # First, get positions of the available objects
        """),
        "ur5": dedent("""
            # SYSTEM:
            # The REFERENCE CODE defines the required task logic. You MUST follow its behavior and sequence.
            # High-level action functions such as pick() and place() are FORBIDDEN. They MUST be replaced with equivalent primitive skill calls.
            # Object access, sensors, and geometry queries used in the reference (e.g., Shape, ProximitySensor, get_position) are ALLOWED.

            # It is provided ONLY to describe task semantics.
            # You must NOT copy, adapt, or call any function appearing in it.

            # MANDATORY: Use these EXACT objects and skills. NO placeholders!
            # MANDATORY OBJECTS (use EXACT names): {obj_list_str}
            # MANDATORY PRIMITIVE SKILLS ONLY (no other calls allowed): ur5_move_to, ur5_grasp_at, ur5_release_at, ur5_align_gripper, close_ur5_ee, open_ur5_ee
            # OUTPUT: Only write valid Python code for run_skill. No markdown.
            # *** ABSOLUTELY NO COMMENTS IN OUTPUT CODE. DO NOT WRITE ANY # COMMENTS. ***

            # FORBIDDEN (must NEVER appear anywhere in output):
            # pick, place, grasp, release, execute, step, reset, plan, policy, controller
            # Any function call not in the allowed primitive list is INVALID.

            # RULES (IMPORTANT)
            # - ur5_grasp_at/ur5_release_at already include moving/approach internally. Do NOT add extra ur5_move_to() unless truly needed.
            # - Use reasonable values: approach distance ~ 0.05-0.15, timeout_s ~ 3.0-10.0.
            # - ur5_align_gripper is NOT a default step. Use it ONLY when top-down approach is not possible.
            # - BEFORE calling ur5_align_gripper, you MUST first call ur5_move_to() to move above the object (e.g., pos + [0,0,0.15]).
            # - approach_direction MUST be 'down' (not 'downward', 'up', etc. - ONLY 'down' is valid)
            # - offset is applied when computing target_pos: target_pos = object_pos + [dx, dy, dz]
            #   Example: target_pos = pos + [0.0, 0.0, 0.2]
            # - yaw_mode MUST be either 'parallel' or 'perpendicular' (no other values allowed)
            # - After ur5_grasp_at, use ur5_move_to() to move to a prepose before ur5_release_at to avoid collisions.
            # - For push-related tasks, call close_ur5_ee() before any push/press motion.
            # - When pick-and-place is required, prioritize using success-related objects (e.g., goal, target, success_zone)
            #   as the final placement location if available in the object list.
            # - Write all function calls on a SINGLE LINE (no line breaks within function arguments)
            # - Do NOT add success checks (e.g., is_detected(), is_triggered(), print statements). Just return obs, reward, done.
            # - Do NOT use if/else statements. Execute all skill calls sequentially without conditional branching.

            # Primitive call shapes (reference)
            # ur5_grasp_at:  obs, reward, done = ur5_grasp_at(env, task, grasp_pos=[x,y,z], timeout_s=5.0)
            # ur5_release_at: obs, reward, done = ur5_release_at(env, task, place_pos=[x,y,z], timeout_s=5.0)
            # ur5_move_to:  obs, reward, done = ur5_move_to(env, task, target_pos=[x,y,z], timeout_s=3.0)
            # ur5_align_gripper: obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            # close_ur5_ee: obs, reward, done = close_ur5_ee(env, task, gripper_close=0.0, velocity=0.2)
            # open_ur5_ee:  obs, reward, done = open_ur5_ee(env, task, gripper_open=1.0, velocity=0.2)

            # Composition patterns (reference)
            # Example 1. ur5_grasp_at then ur5_release_at (simple pick-and-place)
            obj = Shape('object_name')
            success = ProximitySensor('success')
            pos = obj.get_position()
            target_pos = success.get_position()
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
            prepose = target_pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
            obs, reward, done = ur5_release_at(env, task, place_pos=target_pos, timeout_s=5.0)
            
            # Example 2. ur5_align_gripper then ur5_grasp_at (only when alignment needed)
            # IMPORTANT: MUST move to above the object FIRST before calling ur5_align_gripper!
            q = obj.get_quaternion()
            above_obj = pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=above_obj, timeout_s=3.0)
            obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
                      
            # CRITICAL: Output ONLY executable Python code. NO placeholders like ____, FLOAT, etc.
            # Fill in ALL values with actual numbers and strings.
            # *** REMINDER: DO NOT INCLUDE ANY COMMENTS (lines starting with #) IN YOUR OUTPUT CODE ***

        def run_skill(env, task, descriptions=None, obs=None, variation_index: int = 1):
            \"\"\"
            Main function to run the {task_name} task with the given environment and task.
            Task Description: {descriptions}
            Available Objects: {obj_list_str}
            Available Primitive Skills: ur5_move_to, ur5_grasp_at, ur5_release_at, ur5_align_gripper, close_ur5_ee, open_ur5_ee
            \"\"\"

            # First, get positions of the available objects
        """),
        "sawyer": dedent("""
            # SYSTEM:
            # The REFERENCE CODE defines the required task logic. You MUST follow its behavior and sequence.
            # High-level action functions such as pick() and place() from other robots are FORBIDDEN. Use sawyer_pick/sawyer_place instead.
            # Object access, sensors, and geometry queries used in the reference (e.g., Shape, ProximitySensor, get_position) are ALLOWED.

            # It is provided ONLY to describe task semantics.
            # You must NOT copy, adapt, or call any function appearing in it.

            # MANDATORY: Use these EXACT objects and skills. NO placeholders!
            # MANDATORY OBJECTS (use EXACT names): {obj_list_str}
            # MANDATORY PRIMITIVE SKILLS ONLY (no other calls allowed): sawyer_move_to, sawyer_align_gripper, sawyer_open_gripper, sawyer_close_gripper, sawyer_pick, sawyer_place
            # OUTPUT: Only write valid Python code for run_skill. No markdown.
            # *** ABSOLUTELY NO COMMENTS IN OUTPUT CODE. DO NOT WRITE ANY # COMMENTS. ***

            # FORBIDDEN (must NEVER appear anywhere in output):
            # pick, place, grasp, release, execute, step, reset, plan, policy, controller (except sawyer_pick, sawyer_place)
            # Any function call not in the allowed primitive list is INVALID.

            # RULES (IMPORTANT)
            # - sawyer_pick/sawyer_place are composite skills that handle move + grip/release internally.
            # - sawyer_pick REQUIRES target_object parameter (the Shape object to grasp). target_pos is optional.
            # - sawyer_move_to moves to target position while maintaining current orientation.
            # - sawyer_align_gripper aligns gripper orientation while maintaining current position.
            # - approach_direction MUST be 'down' (not 'downward', 'up', etc. - ONLY 'down' is valid)
            # - sawyer_align_gripper is NOT a default step. Use it ONLY when top-down approach is not possible.
            # - offset is applied when computing target_pos: target_pos = object_pos + [dx, dy, dz]
            #   Example: target_pos = pos + [0.0, 0.0, 0.2]
            # - yaw_mode MUST be either 'parallel' or 'perpendicular' (no other values allowed)
            # - After sawyer_pick, use sawyer_move_to() to move to a prepose before sawyer_place to avoid collisions.
            # - For push-related tasks, call sawyer_close_gripper() before any push/press motion.
            # - When pick-and-place is required, prioritize using success-related objects (e.g., goal, target, success_zone)
            #   as the final placement location if available in the object list.
            # - Write all function calls on a SINGLE LINE (no line breaks within function arguments)
            # - Do NOT add success checks (e.g., is_detected(), is_triggered(), print statements). Just return obs, reward, done.
            # - Do NOT use if/else statements. Execute all skill calls sequentially without conditional branching.

            # Primitive call shapes (reference)
            # sawyer_move_to:  obs, reward, done = sawyer_move_to(env, task, target_pos=[x,y,z])
            # sawyer_align_gripper: obs, reward, done = sawyer_align_gripper(env, task, approach_direction='down', reference_quat=q, yaw_mode='parallel')
            # sawyer_open_gripper:  obs, reward, done = sawyer_open_gripper(env, task, amount=1.0, velocity=0.2)
            # sawyer_close_gripper: obs, reward, done = sawyer_close_gripper(env, task, amount=0.0, velocity=0.2)
            # sawyer_pick:  obs, reward, done = sawyer_pick(env, task, target_object=obj, target_pos=[x,y,z])
            # sawyer_place: obs, reward, done = sawyer_place(env, task, place_pos=[x,y,z])

            # Composition patterns (reference)
            # Example 1. pick then place using composite skills
            # obj = Shape('object_name')
            # pos = obj.get_position()
            # target_pos = pos + [0.0, 0.0, 0.2]
            # obs, reward, done = sawyer_pick(env, task, target_object=obj, target_pos=pos.tolist())
            # obs, reward, done = sawyer_place(env, task, place_pos=target_pos.tolist())
            #
            # Example 2. manual move + grip sequence (for more control)
            # obs, reward, done = sawyer_open_gripper(env, task)
            # obs, reward, done = sawyer_move_to(env, task, target_pos=approach_pos.tolist())
            # obs, reward, done = sawyer_move_to(env, task, target_pos=grasp_pos.tolist())
            # obs, reward, done = sawyer_close_gripper(env, task)

            # CRITICAL: Output ONLY executable Python code. NO placeholders like ____, FLOAT, etc.
            # Fill in ALL values with actual numbers and strings.

        def run_skill(env, task, descriptions=None, obs=None, variation_index: int = 1):
            \"\"\"
            Main function to run the {task_name} task with the given environment and task.
            Task Description: {descriptions}
            Available Objects: {obj_list_str}
            Available Primitive Skills: sawyer_move_to, sawyer_align_gripper, sawyer_open_gripper, sawyer_close_gripper, sawyer_pick, sawyer_place
            \"\"\"

            # First, get positions of the available objects. DO NOT WRITE ANY # COMMENTS.
        """),
    },
    "CODEX": {
        "ur5": dedent("""
            # MANDATORY: Use these EXACT objects and skills. NO placeholders!
            # Available objects: {obj_list_str}
            # Available primitive skills: ur5_move_to, ur5_grasp_at, ur5_release_at, ur5_align_gripper, close_ur5_ee, open_ur5_ee
            # OUTPUT: Only write valid Python code for run_skill. No markdown. No comments.

            # RULES (IMPORTANT)
            # - ur5_grasp_at/ur5_release_at already include moving/approach internally. Do NOT add extra ur5_move_to() unless truly needed.
            # - Use reasonable values: approach distance ~ 0.05-0.15, timeout_s ~ 3.0-10.0.
            # - ur5_align_gripper is NOT a default step. Use it ONLY when top-down approach is not possible.
            # - BEFORE calling ur5_align_gripper, you MUST first call ur5_move_to() to move above the object (e.g., pos + [0,0,0.15]).
            # - approach_direction MUST be 'down' (not 'downward', 'up', etc. - ONLY 'down' is valid)
            # - offset is applied when computing target_pos: target_pos = object_pos + [dx, dy, dz]
            #   Example: target_pos = pos + [0.0, 0.0, 0.2]  # 20cm above
            # - yaw_mode MUST be either 'parallel' or 'perpendicular' (no other values allowed)
            # - After ur5_grasp_at, use ur5_move_to() to move to a prepose before ur5_release_at to avoid collisions.
            # - For push-related tasks, call close_ur5_ee() before any push/press motion.
            # - When pick-and-place is required, prioritize using success-related objects (e.g., goal, target, success_zone)
            #   as the final placement location if available in the object list.
            # - Write all function calls on a SINGLE LINE (no line breaks within function arguments)
            # - Do NOT add success checks (e.g., is_detected(), is_triggered(), print statements). Just return obs, reward, done.
            # - Do NOT use if/else statements. Execute all skill calls sequentially without conditional branching.

            # Primitive call shapes (reference)
            # ur5_grasp_at:  obs, reward, done = ur5_grasp_at(env, task, grasp_pos=[x,y,z], timeout_s=5.0)
            # ur5_release_at: obs, reward, done = ur5_release_at(env, task, place_pos=[x,y,z], timeout_s=5.0)
            # ur5_move_to:  obs, reward, done = ur5_move_to(env, task, target_pos=[x,y,z], timeout_s=3.0)
            # ur5_align_gripper: obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            # close_ur5_ee: obs, reward, done = close_ur5_ee(env, task, gripper_close=0.0, velocity=0.2)
            # open_ur5_ee:  obs, reward, done = open_ur5_ee(env, task, gripper_open=1.0, velocity=0.2)

            # Composition patterns (reference)
            # Example 1. ur5_grasp_at then ur5_release_at (simple pick-and-place)
            obj = Shape('object_name')
            success = ProximitySensor('success')
            pos = obj.get_position()
            target_pos = success.get_position()
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
            prepose = target_pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
            obs, reward, done = ur5_release_at(env, task, place_pos=target_pos, timeout_s=5.0)

            # Example 2. ur5_align_gripper then ur5_grasp_at (only when alignment needed)
            # IMPORTANT: MUST move to above the object FIRST before calling ur5_align_gripper!
            q = obj.get_quaternion()
            above_obj = pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=above_obj, timeout_s=3.0)
            obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
                      
            # [CURRENT SCENE]
            # gripper: is_open={gripper_open}, pos={gripper_pos}
            # objects:
            {obj_list}
            # graspable_objects:
            {graspable_obj_list}

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

            # First, get positions of the available objects
        """),
        "sawyer": dedent("""
            # MANDATORY: Use these EXACT objects and skills. NO placeholders!
            # Available objects: {obj_list_str}
            # Available primitive skills: sawyer_move_to, sawyer_align_gripper, sawyer_open_gripper, sawyer_close_gripper, sawyer_pick, sawyer_place
            # OUTPUT: Only write valid Python code for run_skill. No markdown. No comments.

            # RULES (IMPORTANT)
            # - sawyer_pick/sawyer_place are composite skills that handle move + grip/release internally.
            # - sawyer_pick REQUIRES target_object parameter (the Shape object to grasp). target_pos is optional.
            # - sawyer_move_to moves to target position while maintaining current orientation.
            # - sawyer_align_gripper aligns gripper orientation while maintaining current position.
            # - If approach_direction is not explicitly specified, it MUST default to 'down' (do NOT use AXIS.* or enums).
            # - sawyer_align_gripper is NOT a default step. Use it ONLY when top-down approach is not possible.
            # - offset is applied when computing target_pos: target_pos = object_pos + [dx, dy, dz]
            #   Example: target_pos = pos + [0.0, 0.0, 0.2]  # 20cm above
            # - yaw_mode MUST be either 'parallel' or 'perpendicular' (no other values allowed)
            # - After sawyer_pick, use sawyer_move_to() to move to a prepose before sawyer_place to avoid collisions.
            # - For push-related tasks, call sawyer_close_gripper() before any push/press motion.
            # - When pick-and-place is required, prioritize using success-related objects (e.g., goal, target, success_zone)
            #   as the final placement location if available in the object list.

            # Primitive call shapes (reference)
            # sawyer_move_to:  obs, reward, done = sawyer_move_to(env, task, target_pos=[x,y,z])
            # sawyer_align_gripper:
            #   q = obj.get_quaternion()
            #   obs, reward, done = sawyer_align_gripper(env, task, approach_direction='down', reference_quat=q, yaw_mode='parallel')
            # sawyer_open_gripper:  obs, reward, done = sawyer_open_gripper(env, task, amount=1.0, velocity=0.2)
            # sawyer_close_gripper: obs, reward, done = sawyer_close_gripper(env, task, amount=0.0, velocity=0.2)
            # sawyer_pick:  obs, reward, done = sawyer_pick(env, task, target_object=obj, target_pos=[x,y,z], grasp_offset=[0,0,0], velocity=0.2)
            # sawyer_place: obs, reward, done = sawyer_place(env, task, place_pos=[x,y,z], place_offset=[0,0,0])

            # Composition patterns (reference)
            # Example 1. pick then place using composite skills
            # obj = Shape('object_name')
            # success = ProximitySensor('success')
            # pos = obj.get_position()
            # target_pos = success.get_position()
            # obs, reward, done = sawyer_pick(env, task, target_object=obj, target_pos=pos)
            # obs, reward, done = sawyer_place(env, task, place_pos=target_pos)
            #
            # Example 2. sawyer_pick, sawyer_move_to prepose and pre-place pos, then sawyer_place
            # obs, reward, done = sawyer_pick(env, task, target_object=obj)
            # prepose = pos + [0.0, 0.0, 0.2]
            # obs, reward, done = sawyer_move_to(env, task, target_pos=prepose)
            # pre_place = target_pos + [0.0, 0.0, 0.2]
            # obs, reward, done = sawyer_move_to(env, task, target_pos=pre_place)
            # obs, reward, done = sawyer_place(env, task, place_pos=target_pos)

            # [CURRENT SCENE]
            # gripper: is_open={gripper_open}, pos={gripper_pos}
            # objects:
            {obj_list}
            # graspable_objects:
            {graspable_obj_list}

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

            # First, get positions of the available objects
        """)
    }
}


# =========================
# Repair Prompts (comment-style, concise)
# =========================

repair_prompt_dict = {
    "CODE_AGENT": {
        "panda": dedent("""\
            # [REPAIR TASK] Fix the failed statement.
            # Available objects: {obj_list_str}
            # Available skills: move, pick, place, push, open_gripper, close_gripper, align_to_quaternion
            #
            # [ERROR] {error_type}: {error_msg}
            # [FAILED] {failed_stmt}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Fix failed statement, keep executed unchanged, output complete run_skill body
        """),
        "ur5": dedent("""\
            # [REPAIR TASK] Fix the failed statement.
            # Available objects: {obj_list_str}
            # Available skills: ur5_move_to, ur5_grasp_at, ur5_release_at, ur5_align_gripper, close_ur5_ee, open_ur5_ee
            #
            # [ERROR] {error_type}: {error_msg}
            # [FAILED] {failed_stmt}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Fix failed statement, keep executed unchanged, output complete run_skill body
        """),
        "sawyer": dedent("""\
            # [REPAIR TASK] Fix the failed statement.
            # Available objects: {obj_list_str}
            # Available skills: sawyer_move_to, sawyer_align_gripper, sawyer_open_gripper, sawyer_close_gripper, sawyer_pick, sawyer_place
            #
            # [ERROR] {error_type}: {error_msg}
            # [FAILED] {failed_stmt}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Fix failed statement, keep executed unchanged, output complete run_skill body
        """),
    }
}


# =========================
# Revise Prompts (comment-style, concise)
# =========================

revise_prompt_dict = {
    "CODE_AGENT": {
        "panda": dedent("""\
            # [REVISE TASK] Review remaining code, add steps if needed.
            # Available objects: {obj_list_str}
            # Available skills: move, pick, place, push, open_gripper, close_gripper, align_to_quaternion
            #
            # [LAST STEP] index={step_idx}, skill={skill_name}, task_success={success}
            {status_line}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Keep executed unchanged, revise remaining, output complete run_skill body
        """),
        "ur5": dedent("""\
            # [REVISE TASK] Review remaining code, add steps if needed.
            # Available objects: {obj_list_str}
            # Available skills: ur5_move_to, ur5_grasp_at, ur5_release_at, ur5_align_gripper, close_ur5_ee, open_ur5_ee
            #
            # [LAST STEP] index={step_idx}, skill={skill_name}, task_success={success}
            {status_line}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Keep executed unchanged, revise remaining, output complete run_skill body
        """),
        "sawyer": dedent("""\
            # [REVISE TASK] Review remaining code, add steps if needed.
            # Available objects: {obj_list_str}
            # Available skills: sawyer_move_to, sawyer_align_gripper, sawyer_open_gripper, sawyer_close_gripper, sawyer_pick, sawyer_place
            #
            # [LAST STEP] index={step_idx}, skill={skill_name}, task_success={success}
            {status_line}
            #
            # [EXECUTED - DO NOT MODIFY]
            {executed_stmts_str}
            #
            # [SCENE] gripper: pos={gripper_pos}, open={gripper_open}
            {scene_objects_str}
            #
            # [CURRENT CODE]
            {current_code}

            # [OUTPUT] Keep executed unchanged, revise remaining, output complete run_skill body
        """),
    }
}


# =========================
# Helper Functions
# =========================

def _quote_obj_list(object_names):
    return ", ".join(f"'{name}'" for name in object_names)


def _format_executed_stmts(executed_stmts: List[str]) -> str:
    if not executed_stmts:
        return "
    lines = []
    for stmt in executed_stmts:
        for line in stmt.split('\n'):
            lines.append(f"
    return "\n".join(lines)


def _format_scene_objects(scene: Dict[str, Any], limit: int = 8) -> str:
    objects = scene.get("objects", [])
    if not objects:
        return "
    lines = []
    for obj in objects[:limit]:
        name = obj.get("name", "")
        pos = obj.get("position", [])
        if name and pos:
            pos_str = [round(p, 3) for p in pos]
            lines.append(f"
    return "\n".join(lines)

def _indent_except_first(text: str, indent: str = "    ") -> str:
    lines = text.splitlines()
    if len(lines) <= 1:
        return text
    return "\n".join([lines[0]] + [indent + line for line in lines[1:]])

def _format_scene_objects_with_type(scene: Dict[str, Any], limit: int = 8, is_revise: bool = False) -> str:
    objects = scene.get("objects", [])
    if not objects:
        return "
    lines = []
    for obj in objects[:limit]:
        name = obj.get("name", "")
        if not is_revise and "Cuboid" in name:
            continue
        obj_type = obj.get("type", "")
        pos = obj.get("position", [])
        if name and pos:
            pos_str = [round(p, 3) for p in pos]
            # Build optional attributes
            extras = []
            quat = obj.get("quaternion")
            if quat:
                quat_str = [round(q, 3) for q in quat]
                extras.append(f"quat={quat_str}")
            bbox = obj.get("bounding_box")
            if bbox:
                size = bbox.get("size", [])
                size_str = [round(s, 3) for s in size]
                extras.append(f"bbox_size={size_str}")
            extras_str = ", ".join(extras)
            if extras_str:
                lines.append(f"
            else:
                lines.append(f"#   {name}: pos={pos_str}, type={obj_type}")
    return _indent_except_first("\n".join(lines))


def _format_gripper_pos(scene: Dict[str, Any]) -> str:
    gripper = scene.get("gripper", {})
    pos = gripper.get("position", [])
    if pos:
        return str([round(p, 3) for p in pos])
    return "N/A"


# =========================
# Main Prompt Functions
# =========================

def get_prompt(task, task_name, object_names, target_robot, descriptions, model, grasp_guidance: str = ""):
    model_key = model.upper()

    if model_key == 'OURS':
        return "Temp"

    if model_key not in initial_prompt_dict:
        raise ValueError(f"Unknown model type: {model}")

    if target_robot not in initial_prompt_dict[model_key]:
        raise ValueError(f"Unknown target_robot: {target_robot}")

    base_template = initial_prompt_dict[model_key][target_robot]

    obj_list_str = _quote_obj_list(object_names)
    descriptions = descriptions or ""

    # Task-specific guidance
    if task_name == "PickAndLift":
        descriptions = "Move above pick_and_lift_target, pick it, and then place it pick_and_lift_success."
    if task_name == "PhoneOnBase" and target_robot == "ur5":
        descriptions = "Move above the phone, align gripper to the phone, pick the phone, and release it at 5cm above the success position"
    if task_name == "PhoneOnBase" and target_robot == "sawyer":
        descriptions = "“Pick the phone at the phone’s current position. Then move to a point 5 cm above the success position and place (release) the phone there. Use only pick, move, and place. Do NOT align the gripper to the phone."
    if task_name == "UnplugCharger":
        descriptions = "Move above the charger, align gripper perpendicular to the charger, grasp the charger, move back by [-0.1, 0.0, 0.0], then move to the success position and release it"
    if task_name == "PushButton":
        descriptions = "Close gripper, move above the button, and push it down. DO NOT align the gripper. DO NOT use release_at or place. Just close the gripper and move to the button."
    if task_name == "MeatOffGrill":
        descriptions = "Grasp the chicken, move above the success position, and release it slightly above the success position."

#     if task_name == "PickAndLift":
#         descriptions = """Pick up the block and lift it to the target sensor.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# target_block = Shape('pick_and_lift_target')
# success_sensor = ProximitySensor('pick_and_lift_success')
# grasp_pos = target_block.get_position()
# target_pos = success_sensor.get_position()
# obs, reward, done = ur5_grasp_at(env, task, grasp_pos=grasp_pos, timeout_s=5.0)
# prepose = [target_pos[0], target_pos[1], target_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
# obs, reward, done = ur5_move_to(env, task, target_pos=target_pos, timeout_s=3.0)
# return obs, reward, done"""

#     if task_name == "PhoneOnBase":
#         descriptions = """put the phone on the base
# COPY THIS CODE EXACTLY:
# phone = Shape('phone')
# success = ProximitySensor('success')
# phone_pos = phone.get_position()
# success_pos = success.get_position()
# q = phone.get_quaternion()
# above_phone = [phone_pos[0], phone_pos[1], phone_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=above_phone, timeout_s=3.0)
# obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='perpendicular', approach_direction='down', timeout_s=5.0)
# obs, reward, done = ur5_grasp_at(env, task, grasp_pos=phone_pos, timeout_s=5.0)
# prepose = [success_pos[0], success_pos[1], success_pos[2] + 0.30]
# obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
# obs, reward, done = ur5_release_at(env, task, place_pos=success_pos, timeout_s=5.0)"""

#     if task_name == "UnplugCharger":
#         descriptions = """Unplug the charger from the wall socket.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# charger = Shape('charger')
# charger_success = ProximitySensor('charger_success')
# charger_pos = charger.get_position()
# success_pos = charger_success.get_position()
# obs, reward, done = ur5_grasp_at(env, task, grasp_pos=charger_pos, timeout_s=5.0)
# pull_pos = [charger_pos[0], charger_pos[1] - 0.15, charger_pos[2]]
# obs, reward, done = ur5_move_to(env, task, target_pos=pull_pos, timeout_s=3.0)
# prepose = [success_pos[0], success_pos[1], success_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
# obs, reward, done = ur5_release_at(env, task, place_pos=success_pos, timeout_s=5.0)
# return obs, reward, done"""

#     if task_name == "PushButton":
#         descriptions = """Push the button down.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# button = Shape('target_button_topPlate')
# button_pos = button.get_position()
# obs, reward, done = close_ur5_ee(env, task, gripper_close=0.0, velocity=0.2)
# above_button = [button_pos[0], button_pos[1], button_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=above_button, timeout_s=3.0)
# push_pos = [button_pos[0], button_pos[1], button_pos[2] - 0.02]
# obs, reward, done = ur5_move_to(env, task, target_pos=push_pos, timeout_s=3.0)
# return obs, reward, done"""

#     if task_name == "LampOff" or task_name == "LampOn":
#         descriptions = """Press the button to turn off/on the lamp.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# button = Shape('push_button_target')
# button_pos = button.get_position()
# obs, reward, done = close_ur5_ee(env, task, gripper_close=0.0, velocity=0.2)
# above_button = [button_pos[0], button_pos[1], button_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=above_button, timeout_s=3.0)
# push_pos = [button_pos[0], button_pos[1], button_pos[2] - 0.02]
# obs, reward, done = ur5_move_to(env, task, target_pos=push_pos, timeout_s=3.0)
# return obs, reward, done"""

#     if task_name == "MeatOffGrill":
#         descriptions = """Take the chicken off the grill and place it next to the grill.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# chicken = Shape('chicken')
# success = ProximitySensor('success')
# chicken_pos = chicken.get_position()
# success_pos = success.get_position()
# obs, reward, done = ur5_grasp_at(env, task, grasp_pos=chicken_pos, timeout_s=5.0)
# prepose = [success_pos[0], success_pos[1], success_pos[2] + 0.15]
# obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
# obs, reward, done = ur5_release_at(env, task, place_pos=success_pos, timeout_s=5.0)
# return obs, reward, done"""

#     if task_name == "MeatOnGrill":
#         descriptions = """Put the chicken on the grill.
# DO NOT use if/else statements. Execute all steps sequentially.
# EXACT CODE (copy exactly):
# chicken = Shape('chicken')
# success = ProximitySensor('success')
# chicken_pos = chicken.get_position()
# success_pos = success.get_position()
# above_chicken = [chicken_pos[0], chicken_pos[1], chicken_pos[2] + 0.20]
# obs, reward, done = ur5_move_to(env, task, target_pos=above_chicken, timeout_s=3.0)
# obs, reward, done = ur5_grasp_at(env, task, grasp_pos=chicken_pos, timeout_s=5.0)
# prepose = [success_pos[0], success_pos[1], success_pos[2] + 0.20]
# obs, reward, done = ur5_move_to(env, task, target_pos=prepose, timeout_s=3.0)
# release_pos = [success_pos[0], success_pos[1], success_pos[2] + 0.05]
# obs, reward, done = ur5_release_at(env, task, place_pos=release_pos, timeout_s=5.0)
# return obs, reward, done"""

    # Conditionally include alignment example only when grasp_guidance is provided
    alignment_example = ""
    if grasp_guidance and target_robot == "ur5":
        alignment_example = """
            #
            # Example 2. ur5_align_gripper then ur5_grasp_at (only when alignment needed)
            # IMPORTANT: MUST move to above the object FIRST before calling ur5_align_gripper!
            q = obj.get_quaternion()
            above_obj = pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=above_obj, timeout_s=3.0)
            obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
"""
    elif grasp_guidance and target_robot == "panda":
        alignment_example = """
            #
            # Example 2. align then pick/place
            # q = obj.get_quaternion()
            # obs, reward, done = align_to_quaternion(env, task, quaternion=q, yaw_align='parallel', approach_dir='z', timeout=5.0)
"""
    elif grasp_guidance and target_robot == "sawyer":
        alignment_example = """
            #
            # Example 2. sawyer_align_gripper then sawyer_pick (only when alignment needed)
            q = obj.get_quaternion()
            obs, reward, done = sawyer_align_gripper(env, task, approach_direction='down', reference_quat=q, yaw_mode='parallel')
            obs, reward, done = sawyer_pick(env, task, target_object=obj)
"""

    prompt = base_template.format(
        obj_list_str=obj_list_str,
        task_name=task_name,
        descriptions=descriptions,
        alignment_example=alignment_example,
    )

    return prompt

def get_prompt_for_codex(task, task_name, object_names, target_robot, descriptions, model, scene, grasp_guidance: str = "", graspable_obj_list: List[str] = []):
    model_key = model.upper()

    if model_key == 'OURS':
        return "Temp"

    if model_key not in initial_prompt_dict:
        raise ValueError(f"Unknown model type: {model}")

    if target_robot not in initial_prompt_dict[model_key]:
        raise ValueError(f"Unknown target_robot: {target_robot}")

    base_template = initial_prompt_dict[model_key][target_robot]

    obj_list_str = _quote_obj_list(object_names)
    descriptions = descriptions or ""
    
    if task_name == "WipeDesk":
        descriptions = """Grip the sponge and wipe it back and forth over any dirt you see. \n
    Here are the guidelines for wiping down the desk.
    - Dirt positions: Use task._task.dirt_spots to get actual dirt coordinates and compute min/max, not just dirt_boundary center.
    - Wipe height: Use task._task.z_boundary + 0.02 for wiping, +0.05 for safe travel between rows.
    - Coverage: Zigzag pattern across entire area row by row (row step ~0.04), not single-line back-and-forth.
    - Margins: Add ±0.07 in x-direction, ±0.04 in y-direction to ensure full coverage
"""
    if task_name == "UnplugCharger":
        descriptions = """Grip the black charger and pull it out of the socket. Do NOT use any API that aligns the gripper."""

    if task_name == "PickAndLift":
        descriptions = "Move above pick_and_lift_target, pick it, and then place it pick_and_lift_success."
    if task_name == "PhoneOnBase" and target_robot == "ur5":
        descriptions = "Move above the phone, align gripper to the phone, pick the phone, and release it at 5cm above the success position"
    if task_name == "PhoneOnBase" and target_robot == "sawyer":
        descriptions = "“Pick the phone at the phone’s current position. Then move to a point 5 cm above the success position and place (release) the phone there. Use only pick, move, and place. Do NOT align the gripper to the phone."
    if task_name == "PushButton":
        descriptions = "Close gripper, move above the button, and push it down. DO NOT align the gripper. DO NOT use release_at or place. Just close the gripper and move to the button."
    if task_name == "MeatOffGrill":
        descriptions = "Grasp the chicken, move above the success position, and release it slightly above the success position."


    def _format_obj_list(obj_list: List[str]) -> str:
        if not obj_list:
            return "# (no graspable objects)"
        lines = []
        for obj in obj_list:
            lines.append(f"#   {obj}")
        return _indent_except_first("\n".join(lines))

    # Conditionally include alignment example only when grasp_guidance is provided
    alignment_example = ""
    if grasp_guidance and target_robot == "ur5":
        alignment_example = """
            #
            # Example 2. ur5_align_gripper then ur5_grasp_at (only when alignment needed)
            # IMPORTANT: MUST move to above the object FIRST before calling ur5_align_gripper!
            q = obj.get_quaternion()
            above_obj = pos + [0.0, 0.0, 0.15]
            obs, reward, done = ur5_move_to(env, task, target_pos=above_obj, timeout_s=3.0)
            obs, reward, done = ur5_align_gripper(env, task, reference_quat=q, yaw_mode='parallel', approach_direction='down', timeout_s=5.0)
            obs, reward, done = ur5_grasp_at(env, task, grasp_pos=pos, timeout_s=5.0)
"""
    elif grasp_guidance and target_robot == "sawyer":
        alignment_example = """
            #
            # Example 2. sawyer_align_gripper then sawyer_pick (only when alignment needed)
            q = obj.get_quaternion()
            obs, reward, done = sawyer_align_gripper(env, task, approach_direction='down', reference_quat=q, yaw_mode='parallel')
            obs, reward, done = sawyer_pick(env, task, target_object=obj)
"""

    prompt = base_template.format(
        obj_list_str=obj_list_str,
        task_name=task_name,
        descriptions=descriptions,
        gripper_open=scene.get("gripper", {}).get("is_open", True),
        gripper_pos=_format_gripper_pos(scene),
        obj_list=_format_scene_objects_with_type(scene),
        graspable_obj_list=_format_obj_list(graspable_obj_list),
        grasp_guidance=f"\n\n# [GRASP] {grasp_guidance}" if grasp_guidance else "",
        alignment_example=alignment_example,
    )

    return prompt

def get_repair_prompt(
    task_name: str,
    target_robot: str,
    model: str,
    object_names: List[str],
    current_code: str,
    executed_stmts: List[str],
    failed_stmt: str,
    error_type: str,
    error_msg: str,
    scene: Dict[str, Any],
    grasp_guidance: str = "",
) -> str:
    """Get repair prompt for failed execution."""
    model_key = model.upper()

    if model_key == 'OURS':
        model_key = 'CODE_AGENT'

    if model_key not in repair_prompt_dict:
        model_key = 'CODE_AGENT'

    if target_robot not in repair_prompt_dict[model_key]:
        target_robot = 'panda'

    template = repair_prompt_dict[model_key][target_robot]

    prompt = template.format(
        obj_list_str=_quote_obj_list(object_names),
        error_type=error_type,
        error_msg=error_msg,
        failed_stmt=failed_stmt,
        executed_stmts_str=_format_executed_stmts(executed_stmts),
        gripper_pos=_format_gripper_pos(scene),
        gripper_open=scene.get("gripper", {}).get("is_open", True),
        scene_objects_str=_format_scene_objects_with_type(scene),
        current_code=current_code,
    )

    if grasp_guidance:
        prompt = prompt + f"\n# [GRASP] {grasp_guidance}"

    return prompt


def get_revise_prompt(
    task_name: str,
    target_robot: str,
    model: str,
    object_names: List[str],
    current_code: str,
    executed_stmts: List[str],
    step_idx: int,
    skill_name: str,
    success: bool,
    need_more_steps: bool,
    scene: Dict[str, Any],
    grasp_guidance: str = "",
) -> str:
    """Get revise prompt for step-by-step refinement."""
    model_key = model.upper()

    if model_key == 'OURS':
        model_key = 'CODE_AGENT'

    if model_key not in revise_prompt_dict:
        model_key = 'CODE_AGENT'

    if target_robot not in revise_prompt_dict[model_key]:
        target_robot = 'panda'

    template = revise_prompt_dict[model_key][target_robot]

    status_line = ""
    if need_more_steps:
        status_line = "# [STATUS] Task incomplete - need more steps!"

    prompt = template.format(
        obj_list_str=_quote_obj_list(object_names),
        step_idx=step_idx,
        skill_name=skill_name or "N/A",
        success=success,
        status_line=status_line,
        executed_stmts_str=_format_executed_stmts(executed_stmts),
        gripper_pos=_format_gripper_pos(scene),
        gripper_open=scene.get("gripper", {}).get("is_open", True),
        scene_objects_str=_format_scene_objects_with_type(scene),
        current_code=current_code,
    )

    if grasp_guidance:
        prompt = prompt + f"\n# [GRASP] {grasp_guidance}"

    return prompt


# Legacy stub functions (for backward compatibility)
def get_regenerate_prompt_on_failure():
    pass

def get_regenerate_prompt_on_not_finished():
    pass
