from typing import List, Optional
import omnigibson as og
from aenum import IntEnum, auto
from omnigibson import object_states
from omnigibson.action_primitives.action_primitive_set_base import (
    ActionPrimitiveError, 
    ActionPrimitiveErrorGroup
)
from omnigibson.action_primitives.starter_semantic_action_primitives import (
    StarterSemanticActionPrimitives,
)
from omnigibson.envs import Environment
from omnigibson.objects import StatefulObject
from omnigibson.object_states.object_state_base import RelativeObjectState 
from omnigibson.robots.robot_base import BaseRobot
from omnigibson.systems import BaseSystem
from omnigibson.transition_rules import SlicingRule
from omnigibson.utils.constants import PrimType
import torch
import sys
import math
import json
import os
from PIL import Image
from og_ego_prim.utils.constants import CAMERAS

from .object_states_utils import (
    get_covered_systems, 
    get_contained_systems,
    get_produced_systems,
    get_supported_systems,
    get_modified_systems,
    get_container,
    capture_attachments,
    recover_attachments,
    check_open_before_grasp,
    check_open_before_placement,
    check_close_before_toggle_on,
    check_toggle_off_before_open,
    check_heat_source_before_cook,
    get_cooked_system,
    is_cloth_place_on_other,
    get_placement_objects,
    is_visual_or_physical_particle_system,
    find_task_related_object,
    get_obj_with_state,
    is_target_object_predicate_with_obj
)


class EgoSemanticActionPrimitiveSet(IntEnum):
    _init_ = "value __doc__"
    PLACE_ON_TOP = auto(), "Place the target_obj on top of placement_obj"
    PLACE_INSIDE = auto(), "Place the target_obj inside placement_obj"
    OPEN = auto(), "Open an target_obj"
    CLOSE = auto(), "Close an target_obj"
    TOGGLE_ON = auto(), "Toggle an target_obj on"
    TOGGLE_OFF = auto(), "Toggle an target_obj off"
    WIPE = auto(), "Wipe the target_obj with the cleaning_tool"
    CUT = auto(), "Cut (slice or dice) the target_obj with the cutting_tool"
    SOAK_UNDER = auto(), "Soak the target_obj with particles produced by the fluid_source"
    SOAK_INSIDE = auto(), "Soak the target_obj with particles in the fluid_container"
    FILL_WITH = auto(), "Fill the target_obj with particles produced by the fluid source"
    POUR_INTO = auto(), "Pour the particle in the fluid_container into the target_obj (usually a container)"
    WAIT_FOR_COOKED = auto(), "Wait for the cook process of the object to final"
    WAIT_FOR_WASHED = auto(), "Wait for the wash process fo the wash machine to final"
    WAIT = auto(), "Wait for the object to change, such as waiting for the object to rise to room temperature."
    SPREAD = auto(), "Spread some particles onto some object, make object covered with these particles"
    WAIT_FOR_FROZEN = auto(), "Wait something in the refridge to frozen"


VALID_PRIMITIVES = {
    "PLACE_ON_TOP": 2,
    "PLACE_INSIDE": 2,
    "OPEN": 1,
    "CLOSE": 1,
    "TOGGLE_ON": 1,
    "TOGGLE_OFF": 1,
    "WIPE": 2,
    "CUT": 2,
    "SOAK_INSIDE": 2,
    "SOAK_UNDER": 2,
    "FILL_WITH": 2,
    "POUR_INTO": 2,
    "SPREAD": 2,
    "WAIT": 1,
    "WAIT_FOR_COOKED": 1,
    "WAIT_FOR_WASHED": 1,
    "WAIT_FOR_FROZEN": 2,
}


class EgoSemanticActionPrimitives(StarterSemanticActionPrimitives):

    def __init__(self, env: Environment, guardrail_model=None):
        super().__init__(env)
        self.controller_functions = {
            EgoSemanticActionPrimitiveSet.PLACE_ON_TOP: self._place_on_top,
            EgoSemanticActionPrimitiveSet.PLACE_INSIDE: self._place_inside,
            EgoSemanticActionPrimitiveSet.OPEN: self._open,  # done
            EgoSemanticActionPrimitiveSet.CLOSE: self._close,  # done
            EgoSemanticActionPrimitiveSet.TOGGLE_ON: self._toggle_on,  # done
            EgoSemanticActionPrimitiveSet.TOGGLE_OFF: self._toggle_off,  # done
            EgoSemanticActionPrimitiveSet.WIPE: self._wipe,
            EgoSemanticActionPrimitiveSet.CUT: self._cut,
            EgoSemanticActionPrimitiveSet.SOAK_INSIDE: self._soak_inside,
            EgoSemanticActionPrimitiveSet.SOAK_UNDER: self._soak_under,
            EgoSemanticActionPrimitiveSet.FILL_WITH: self._fill_with,
            EgoSemanticActionPrimitiveSet.POUR_INTO: self._pour_into,
            EgoSemanticActionPrimitiveSet.WAIT_FOR_COOKED: self._wait_for_cooked,
            EgoSemanticActionPrimitiveSet.WAIT_FOR_WASHED: self._wait_for_washed,
            EgoSemanticActionPrimitiveSet.WAIT: self._wait,
            EgoSemanticActionPrimitiveSet.SPREAD: self._spread,
            EgoSemanticActionPrimitiveSet.WAIT_FOR_FROZEN: self._wait_for_frozen,

        }
        self.env = env
        self.attachments = []
        self.grasped_objects = {}
        self.guardrail_model = guardrail_model

    def apply_ref(self, primitive, *args, save_img_dir=None, plan_str=None, caution=None):
        robot = self.env.robots[0]
        self._fold_robot_arm_to_home(robot)
    
        if any(isinstance(arg, BaseRobot) for arg in args):
            raise ActionPrimitiveErrorGroup(
                [
                    ActionPrimitiveError(
                        ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                        "Cannot call a symbolic semantic action primitive with a robot as an argument.",
                    )
                ]
            )
        
        # For all actions, the first argument is target_obj, so move camera in front of target_obj
        if args and len(args) > 0:
            target_obj = None
            # Exception cases: PLACE_INSIDE navigates to second argument (placement_obj)
            # _pour_into, _spread have second argument as target_obj
            if (primitive in [EgoSemanticActionPrimitiveSet.POUR_INTO, 
                             EgoSemanticActionPrimitiveSet.SPREAD, 
                             EgoSemanticActionPrimitiveSet.PLACE_INSIDE,
                             EgoSemanticActionPrimitiveSet.CUT,
                             EgoSemanticActionPrimitiveSet.PLACE_ON_TOP] 
                and len(args) > 1 
                and isinstance(args[1], StatefulObject)):
                target_obj = args[1]
            # For most primitives, the first argument is target_obj
            elif isinstance(args[0], StatefulObject):
                target_obj = args[0]
            
        # Set camera pose and save images (obs_a.png, obs_b.png, obs_c.png, obs_d.png)
        # Save images from all directions before calling _navigate_to_obj
        guardrail_img_path = None
        if save_img_dir is not None and target_obj is not None:
            # Make robot invisible
            if len(self.env.robots) > 0:
                self.env.robots[0].visible = False
            
            os.makedirs(save_img_dir, exist_ok=True)
            
            # Get object position
            obj_pos, _ = target_obj.get_position_orientation()
            obj_x, obj_y, obj_z = obj_pos[0].item(), obj_pos[1].item(), obj_pos[2].item()
            
            # Get scene name
            scene_name = None
            try:
                if hasattr(self.env, 'scene'):
                    if hasattr(self.env.scene, 'scene_id'):
                        scene_name = self.env.scene.scene_id
                    elif hasattr(self.env.scene, 'scene_model'):
                        scene_name = self.env.scene.scene_model
            except Exception:
                pass
            
            # Define camera direction offsets
            offset_distance = 1.1 if scene_name and ('Wainscott_1_int' in scene_name or 'Beechwood_1_int' in scene_name or 'Benevolence_2_int' in scene_name or 'Beechwood_0_int' in scene_name or 'Beechwood_0_garden' in scene_name or 'Wainscott_0_int' in scene_name) else 1.5
            
            camera_offsets = {
                'a': (+offset_distance,  0.0),  # +X direction
                'b': (-offset_distance,  0.0),  # -X direction
                'c': ( 0.0, +offset_distance),  # +Y direction
                'd': ( 0.0, -offset_distance),  # -Y direction
            }
            
            camera_height = 1.5
            obj_pos_tensor = torch.tensor([obj_x, obj_y, obj_z], dtype=torch.float32)
            
            # Save images from each direction (a, b, c, d)
            for direction in ['a', 'b', 'c', 'd']:
                dx, dy = camera_offsets[direction]
                camera_pos = torch.tensor([
                    obj_x + dx,
                    obj_y + dy,
                    camera_height
                ], dtype=torch.float32)
                
                # Calculate look-at quaternion
                camera_quat = self._calculate_look_at_quaternion(camera_pos, obj_pos_tensor)
                
                # Move camera
                og.sim.viewer_camera.set_position_orientation(
                    position=camera_pos,
                    orientation=camera_quat
                )
                
                # Run simulator loop (stabilize camera movement)
                if hasattr(self, 'env') and hasattr(self.env, 'sim'):
                    for _ in range(20):  # Increased from 10 to 20
                        og.sim.step()
                
                # Force rendering (ensure camera position is reflected)
                try:
                    # Render multiple times for stabilization
                    for _ in range(3):
                        og.sim.render()
                        og.sim.step()
                except Exception:
                    pass
                
                # Check actual camera position (debugging)
                actual_camera_pos, actual_camera_quat = og.sim.viewer_camera.get_position_orientation()
                print(f"[NAVIGATE_TO] Direction '{direction}' - Target pos: ({camera_pos[0]:.3f}, {camera_pos[1]:.3f}, {camera_pos[2]:.3f}), Actual pos: ({actual_camera_pos[0]:.3f}, {actual_camera_pos[1]:.3f}, {actual_camera_pos[2]:.3f})")
                print(f"[NAVIGATE_TO] Direction '{direction}' - Target quat: ({camera_quat[0]:.3f}, {camera_quat[1]:.3f}, {camera_quat[2]:.3f}, {camera_quat[3]:.3f}), Actual quat: ({actual_camera_quat[0]:.3f}, {actual_camera_quat[1]:.3f}, {actual_camera_quat[2]:.3f}, {actual_camera_quat[3]:.3f})")
                
                # Capture and save image
                obs, info = og.sim.viewer_camera.get_obs()
                rgb = obs['rgb'].cpu().numpy()
                save_path = os.path.join(save_img_dir, f'obs_{direction}.png')
                img = Image.fromarray(rgb)
                img.save(save_path)
                print(f"[NAVIGATE_TO] Saved image for direction '{direction}': {save_path}")
        
        # Call _navigate_to_obj (after saving images)
            if target_obj is not None:
                # For WAIT_FOR_COOKED, if target_obj is inside or on top of a container, look at the container
                if primitive == EgoSemanticActionPrimitiveSet.WAIT_FOR_COOKED:
                    # Directly check if target_obj is inside or on top of any container
                    container_obj = None
                    if isinstance(target_obj, StatefulObject):
                        # Check Inside relationship first
                        if object_states.Inside in target_obj.states:
                            for obj_name in self.env.task.object_scope.keys():
                                if 'agent' in obj_name:
                                    continue
                                candidate_obj = find_task_related_object(self.env, obj_name.strip())
                                # Exclude cloth objects (to prevent ValueError during Inside check)
                                if candidate_obj is None or (hasattr(candidate_obj, 'prim_type') and candidate_obj.prim_type == PrimType.CLOTH):
                                    continue
                                if target_obj.states[object_states.Inside].get_value(candidate_obj):
                                    container_obj = candidate_obj
                                    break
                        
                        # If Inside is not found, check OnTop relationship
                        if container_obj is None and object_states.OnTop in target_obj.states:
                            for obj_name in self.env.task.object_scope.keys():
                                if 'agent' in obj_name:
                                    continue
                                candidate_obj = find_task_related_object(self.env, obj_name.strip())
                                # Exclude cloth objects
                                if candidate_obj is None or (hasattr(candidate_obj, 'prim_type') and candidate_obj.prim_type == PrimType.CLOTH):
                                    continue
                                if target_obj.states[object_states.OnTop].get_value(candidate_obj):
                                    container_obj = candidate_obj
                                    break
                    
                    if container_obj is not None:
                        # If container exists, look at container (e.g., microwave, oven, saucepot, etc.)
                        yield from self._navigate_to_obj(container_obj)
                    else:
                        yield from self._navigate_to_obj(target_obj)
                else:
                    yield from self._navigate_to_obj(target_obj)
        
        # Guardrail check: after capturing obs_before.png, before executing action
        if self.guardrail_model is not None and plan_str is not None:
            # 1. Convert Action to Natural Language
            action_nl = self._convert_action_to_nl(plan_str)
            
            # 2. Use EMBGuard's inference method
            guardrail_result = self.guardrail_model.inference(
                action=action_nl,
                image=guardrail_img_path,
                caution=caution,
                use_few_shot=False,
                use_thinking=False
            )
            
            # 3. If potential_risk = risky -> raise RiskyActionError
            # Also log safe cases
            potential_risk = guardrail_result.get('potential_risk', 'benign').lower()
            risk_type = guardrail_result.get('risk_type', 'none')
            hazard = guardrail_result.get('hazard', 'none')

            print(f'[Guardrail] Action: {plan_str}, Risk: {potential_risk}, Risk Type: {risk_type}, Hazard: {hazard}')
            import sys
            sys.stdout.flush()

            if potential_risk == 'risky':
                from og_ego_prim.primitives.executor import RiskyActionError
                raise RiskyActionError(
                    f'Risky action detected: {plan_str}',
                    risk_type=risk_type,
                    hazard=hazard
                )
            # 4. potential_risk = benign -> continue to action execution
        
        try:
            yield from self.controller_functions[primitive](*args)
        except ActionPrimitiveError as e:
            raise ActionPrimitiveErrorGroup([e])

        # Settle before returning.
        try:
            yield from self._settle_robot()
        except ActionPrimitiveError:
            pass

    def _fold_robot_arm_to_home(self, robot: BaseRobot):
        """
        Fold robot arm to home position.
        """
        try:
            # Use robot_specific_reset() (full reset)
            if hasattr(robot, 'robot_specific_reset'):
                robot.robot_specific_reset()
                print(f"[NAVIGATE_TO] Robot arm reset to home position")
        except Exception as e:
            print(f"[NAVIGATE_TO] Warning: Could not reset robot arm: {e}")
    
    def _get_camera_direction_for_object(
        self, 
        target_obj: StatefulObject
    ) -> str:
        """
        Get camera direction for object from mapping file
        Returns default 'a' if not found
        
        Returns:
            str: One of 'a', 'b', 'c', 'd'
        """
        try:
            mapping_file = os.path.join(CAMERAS, 'object_camera_mappings.json')
            if not os.path.exists(mapping_file):
                print(f"[NAVIGATE_TO] Mapping file not found at {mapping_file}, using default direction 'a'")
                return 'a'
            
            with open(mapping_file, 'r') as f:
                mappings = json.load(f)
            
            # Try to get scene name
            scene_name = None
            try:
                # Try env.scene.scene_id or env.scene.scene_model
                if hasattr(self.env, 'scene'):
                    if hasattr(self.env.scene, 'scene_id'):
                        scene_name = self.env.scene.scene_id
                    elif hasattr(self.env.scene, 'scene_model'):
                        scene_name = self.env.scene.scene_model
            except Exception as e:
                print(f"[NAVIGATE_TO] Warning: Could not get scene name: {e}")
            
            object_name = target_obj.name
            
            # Check scene-specific mapping
            if scene_name and scene_name in mappings:
                scene_mapping = mappings[scene_name]
                if object_name in scene_mapping:
                    direction = scene_mapping[object_name]
                    if direction in ['a', 'b', 'c', 'd']:
                        print(f"[NAVIGATE_TO] Found mapping for {object_name} in {scene_name}: direction '{direction}'")
                        return direction
                    else:
                        print(f"[NAVIGATE_TO] Warning: Invalid direction '{direction}' in mapping, using default 'a'")
            
            # Use default if no mapping found
            print(f"[NAVIGATE_TO] No mapping found for {object_name} (scene: {scene_name}), using default direction 'a'")
            return 'a'
            
        except Exception as e:
            print(f"[NAVIGATE_TO] Error reading mapping file: {e}, using default direction 'a'")
            return 'a'
    
    def _calculate_look_at_quaternion(
        self, 
        camera_pos: torch.Tensor, 
        target_pos: torch.Tensor
    ) -> torch.Tensor:
        """
        Calculate quaternion for camera to look at target (look-at method)
        
        Args:
            camera_pos: Camera position (x, y, z)
            target_pos: Target position (x, y, z)
            
        Returns:
            torch.Tensor: Quaternion (x, y, z, w)
        """
        try:
            # Calculate direction to look at object (look-at)
            # Direction vector from camera to object
            direction = target_pos - camera_pos
            direction_norm = torch.norm(direction)
            if direction_norm > 0:
                direction = direction / direction_norm
            else:
                # Use default direction if direction is zero
                direction = torch.tensor([1.0, 0.0, 0.0], dtype=torch.float32)
            
            # Calculate camera orientation (look-at quaternion)
            # Up vector is Z-axis (vertical)
            up = torch.tensor([0.0, 0.0, 1.0], dtype=torch.float32)
            
            # right = direction × up (right-hand rule)
            right = torch.linalg.cross(direction, up)
            right_norm = torch.norm(right)
            if right_norm > 1e-6:
                right = right / right_norm
            else:
                # When direction and up are almost parallel (vertical direction)
                right = torch.tensor([1.0, 0.0, 0.0], dtype=torch.float32)
            
            # up = right × direction (recalculate to ensure orthogonality)
            up_corrected = torch.linalg.cross(right, direction)
            up_corrected = up_corrected / torch.norm(up_corrected)
            
            # Construct rotation matrix: [right, up_corrected, -direction]
            # Camera's forward is -direction (looking at object)
            R = torch.stack([right, up_corrected, -direction], dim=1)
            
            # Convert rotation matrix to quaternion
            trace = R[0, 0] + R[1, 1] + R[2, 2]
            
            if trace > 0:
                s = math.sqrt(trace + 1.0) * 2
                w = 0.25 * s
                x = (R[2, 1] - R[1, 2]) / s
                y = (R[0, 2] - R[2, 0]) / s
                z = (R[1, 0] - R[0, 1]) / s
            elif (R[0, 0] > R[1, 1]) and (R[0, 0] > R[2, 2]):
                s = math.sqrt(1.0 + R[0, 0] - R[1, 1] - R[2, 2]) * 2
                w = (R[2, 1] - R[1, 2]) / s
                x = 0.25 * s
                y = (R[0, 1] + R[1, 0]) / s
                z = (R[0, 2] + R[2, 0]) / s
            elif R[1, 1] > R[2, 2]:
                s = math.sqrt(1.0 + R[1, 1] - R[0, 0] - R[2, 2]) * 2
                w = (R[0, 2] - R[2, 0]) / s
                x = (R[0, 1] + R[1, 0]) / s
                y = 0.25 * s
                z = (R[1, 2] + R[2, 1]) / s
            else:
                s = math.sqrt(1.0 + R[2, 2] - R[0, 0] - R[1, 1]) * 2
                w = (R[1, 0] - R[0, 1]) / s
                x = (R[0, 2] + R[2, 0]) / s
                y = (R[1, 2] + R[2, 1]) / s
                z = 0.25 * s
            
            return torch.tensor([x, y, z, w], dtype=torch.float32)
        except Exception as e:
            print(f"[NAVIGATE_TO] Warning: Could not calculate camera quaternion: {e}, using default")
            return torch.tensor([0.0, 0.0, 0.0, 1.0], dtype=torch.float32)
    
    def _navigate_to_obj(self, target_obj: StatefulObject):
        """
        Move camera to object location, capturing only the required direction based on mapping
        - 4 directions: a (+X), b (-X), c (+Y), d (-Y)
        - Check object-specific camera direction from mapping file
        - Use default 'a' if no mapping found
        """
        print(f"[NAVIGATE_TO] Moving viewer camera to observe {target_obj.name}...")
        
        # 1. Get camera direction from mapping
        camera_direction = self._get_camera_direction_for_object(target_obj)
        
        # 2. Get object position
        obj_pos, _ = target_obj.get_position_orientation()
        obj_x, obj_y, obj_z = obj_pos[0].item(), obj_pos[1].item(), obj_pos[2].item()
        print(f"[NAVIGATE_TO] Target object position: ({obj_x:.3f}, {obj_y:.3f}, {obj_z:.3f})")
        
        # 3. Get scene name (for checking Wainscott_1_int scene)
        scene_name = None
        try:
            if hasattr(self.env, 'scene'):
                if hasattr(self.env.scene, 'scene_id'):
                    scene_name = self.env.scene.scene_id
                elif hasattr(self.env.scene, 'scene_model'):
                    scene_name = self.env.scene.scene_model
        except Exception as e:
            print(f"[NAVIGATE_TO] Warning: Could not get scene name: {e}")
        
        # 4. Define camera direction offsets (relative to object center)
        # 1.1m for Wainscott_1_int or Beechwood_1_int scenes, 2m for others
        offset_distance = 1.1 if scene_name and ('Wainscott_1_int' in scene_name or 'Beechwood_1_int' in scene_name or 'Benevolence_2_int' in scene_name or 'Beechwood_0_int' in scene_name or 'Beechwood_0_garden' in scene_name or 'Wainscott_0_int' in scene_name) else 2.0
        
        camera_offsets = {
            'a': (+offset_distance,  0.0),  # +X direction
            'b': (-offset_distance,  0.0),  # -X direction
            'c': ( 0.0, +offset_distance),  # +Y direction
            'd': ( 0.0, -offset_distance),  # -Y direction
        }
        
        dx, dy = camera_offsets[camera_direction]
        camera_height = 1.5  # Fixed absolute height
        
        # 5. Calculate camera position
        camera_pos = torch.tensor([
            obj_x + dx,
            obj_y + dy,
            camera_height
        ], dtype=torch.float32)
        
        # 6. Calculate look-at quaternion (to look at object)
        camera_quat = self._calculate_look_at_quaternion(camera_pos, obj_pos)
        
        # 7. Move Viewer Camera
        try:
            og.sim.viewer_camera.set_position_orientation(
                position=camera_pos,
                orientation=camera_quat
            )
            print(f"[NAVIGATE_TO] Viewer camera moved to direction '{camera_direction}' at position: ({camera_pos[0]:.3f}, {camera_pos[1]:.3f}, {camera_pos[2]:.3f})")
            
            # Run simulator loop (stabilize camera movement)
            if hasattr(self, 'env') and hasattr(self.env, 'sim'):
                for _ in range(5):
                    og.sim.step()
            
            print(f"[NAVIGATE_TO] Viewer camera navigation completed")
        except Exception as e:
            print(f"[NAVIGATE_TO] Error moving viewer camera: {e}")
            # Continue even if error occurs (action can still be performed even if camera movement fails)
        
        # 8. Stabilize
        yield from self._settle_robot()
        
        print(f"[NAVIGATE_TO] Navigation completed (direction: '{camera_direction}')")
        sys.stdout.flush()
        
    def _pick_up(self, target_obj: StatefulObject):
        """
        Robot picks up object.
        Makes object invisible so it cannot be grasped by hand.
        When picking up a plate, also picks up objects on top of it (e.g., tofu).
        """
        # Check preconditions
        check_open_before_grasp(target_obj, self.env)
        
        # Find objects on top of plate
        objects_on_top = []
        placements = get_placement_objects(target_obj, self.env, [object_states.OnTop])
        if placements:
            for placement in placements:
                obj_on_top = placement.object
                if obj_on_top.name != target_obj.name:
                    objects_on_top.append(obj_on_top)
                    print(f"[PICK_UP] Found {obj_on_top.name} on top of {target_obj.name}")
        
        # Make object invisible
        print(f"[PICK_UP] Making {target_obj.name} invisible")
        target_obj.visible = False
        
        # Make objects on top of plate also invisible
        for obj_on_top in objects_on_top:
            print(f"[PICK_UP] Making {obj_on_top.name} invisible (on top of {target_obj.name})")
            obj_on_top.visible = False
        
        # Stabilize
        yield from self._settle_robot()
        
        # Record in grasped_objects (plate and objects on top)
        self.grasped_objects[target_obj.name] = {
            'object': target_obj,
            'method': 'invisible',
            'objects_on_top': objects_on_top  # Also record objects on top of plate
        }
        
        # Also record objects on top of plate in grasped_objects
        for obj_on_top in objects_on_top:
            self.grasped_objects[obj_on_top.name] = {
                'object': obj_on_top,
                'method': 'invisible',
                'parent': target_obj.name  # Record parent object
            }
        
        print(f"[PICK_UP] Successfully picked up {target_obj.name} (made invisible)")
        if objects_on_top:
            print(f"[PICK_UP] Also picked up objects on top: {[obj.name for obj in objects_on_top]}")
        
    def _open(self, target_obj: StatefulObject):
        """Open an object, navigating to it first."""
        yield from self._open_or_close(target_obj, should_open=True)
    
    def _close(self, target_obj: StatefulObject):
        """Close an object, navigating to it first."""
        yield from self._open_or_close(target_obj, should_open=False)

    def _open_or_close(self, target_obj: StatefulObject, should_open: bool):
        if object_states.Open not in target_obj.states:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not openable.",
                {"target object": target_obj.name},
            )
        
        # Don't do anything if the object is already closed and we're trying to close.
        if should_open == target_obj.states[object_states.Open].get_value():
            return
        
        if should_open is True:
            check_toggle_off_before_open(target_obj)

        inside_placements = get_placement_objects(target_obj, self.env, object_states.Inside)
        
        # Set the value
        target_obj.states[object_states.Open].set_value(should_open, fully=True)
        yield from self._settle_robot()

        if inside_placements:
            for placement in inside_placements:
                obj = placement.object
                if not is_target_object_predicate_with_obj(obj, target_obj, object_states.Inside):
                    yield from self._place_inside(obj, target_obj, skip_check=True)

        if target_obj.states[object_states.Open].get_value() != should_open:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.POST_CONDITION_ERROR,
                "The object did not open or close as expected. Maybe try again",
                {"target object": target_obj.name, "is it currently open": target_obj.states[object_states.Open].get_value()},
            )

    def _place_on_top(self, target_obj: StatefulObject, placement_obj: StatefulObject, **kwargs):
        yield from self._place_with_predicate(target_obj, placement_obj, object_states.OnTop, **kwargs)

    def _place_inside(self, target_obj: StatefulObject, placement_obj: StatefulObject, **kwargs):
        yield from self._place_with_predicate(target_obj, placement_obj, object_states.Inside, **kwargs)

    def _sample_on_top_heat_source(self, target_obj: StatefulObject, heat_source: StatefulObject):
        heating_element = heat_source.states[object_states.HeatSourceOrSink].link
        heating_element_positions = heating_element.get_position_orientation()[0] + torch.tensor([0, 0, 0.1])
        target_obj.set_position_orientation(position=heating_element_positions)
        yield from self._settle_robot()

    def _sample_placement_with_predicate(
        self, 
        target_obj: StatefulObject, 
        placement_obj: StatefulObject, 
        predicate: RelativeObjectState,
    ):
        placement_obj_pose = placement_obj.get_position_orientation()

        attempts = 0
        while attempts < 5:
            attempts += 1
            try:
                # Find a spot to put it
                predicated_pose = self._sample_pose_with_object_and_predicate(
                    predicate, target_obj, placement_obj
                )
            except Exception as e:
                print(f'Attempt {attempts}: {e}')
                continue

            # Actually move the target object to the spot and step a bit to settle it.
            target_obj.set_position_orientation(*predicated_pose)
            yield from self._settle_robot()

            if target_obj.states[predicate].get_value(placement_obj):
                break
            else:
                # recover if failed
                placement_obj.set_position_orientation(*placement_obj_pose)
                placement_obj.keep_still()
                yield from self._settle_robot()

    def _place_with_predicate(
        self, 
        target_obj: StatefulObject | BaseSystem, 
        placement_obj: StatefulObject, 
        predicate: RelativeObjectState,
        skip_check: bool = False,
    ):
        if isinstance(target_obj, BaseSystem) and is_visual_or_physical_particle_system(target_obj.scene, target_obj):
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "Cannot place system to desired position, perhaps place its container like bottle to the position",
                {"system (target object)": target_obj.name},
            )

        if predicate == object_states.OnTop and is_cloth_place_on_other(target_obj, placement_obj):
            predicate = object_states.Overlaid
        if predicate not in [object_states.OnTop, object_states.Inside, object_states.Overlaid]:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "Only support place the target object OnTop, OverLaid or Inside the placement object.",
                {"provided_predicate": predicate.__class__.__name__},
            )

        if not skip_check:
            check_open_before_grasp(target_obj, self.env)
            if predicate == object_states.Inside:
                check_open_before_placement(placement_obj)

        attachments = capture_attachments(target_obj, self.env)
        
        if object_states.HeatSourceOrSink in placement_obj.states and predicate == object_states.OnTop and \
              not placement_obj.states[object_states.HeatSourceOrSink].requires_inside:
            yield from self._sample_on_top_heat_source(target_obj, placement_obj)

        elif predicate == object_states.Overlaid:
            placement_pose = placement_obj.get_position_orientation()[0]
            position = placement_pose + torch.tensor([0, 0, 0.1])
            target_obj.set_position_orientation(position=position)
            yield from self._settle_robot()

        else:
            yield from self._sample_placement_with_predicate(target_obj, placement_obj, predicate)

            # Last attempt to directly place the target object with predicate
            if not target_obj.states[predicate].get_value(placement_obj):
                if predicate == object_states.OnTop:  # ontop 
                    placement_pose = placement_obj.get_position_orientation()[0]
                    position = placement_pose + torch.tensor([0, 0, 0.1])
                else: # others, inside
                    placement_obj_center = placement_obj.get_base_aligned_bbox()[0]
                    position = placement_obj_center

                target_obj.set_position_orientation(position=position)
                yield from self._settle_robot()

        if attachments:
            recover_attachments(attachments)
            yield from self._settle_robot()

        # check
        error = ActionPrimitiveError(
            ActionPrimitiveError.Reason.EXECUTION_ERROR,
            "Failed to place target object at the desired place (probably dropped).",
            {"dropped object": target_obj.name, "placement object": placement_obj.name, "predicate": predicate.__class__.__name__},
        )
        if predicate == object_states.OnTop:
            adjacency = target_obj.states[object_states.VerticalAdjacency].get_value()
            if not placement_obj in adjacency.negative_neighbors or placement_obj in adjacency.positive_neighbors:
                raise error
        elif predicate == object_states.Inside and not target_obj.states[predicate].get_value(placement_obj):
                raise error
        elif predicate == object_states.Overlaid and not target_obj.states[object_states.Touching].get_value(placement_obj):
                raise error

    def _toggle_on(self, target_obj: StatefulObject):
        """Toggle an object on, navigating to it first."""
        yield from self._toggle(target_obj, value=True)
    
    def _toggle_off(self, target_obj: StatefulObject):
        """Toggle an object off, navigating to it first."""
        yield from self._toggle(target_obj, value=False)

    def _toggle(self, target_obj: StatefulObject, value: bool):
        if object_states.ToggledOn not in target_obj.states:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not toggleable.",
                {"target object": target_obj.name},
            )
        
        if target_obj.states[object_states.ToggledOn].get_value() == value:
            return
        
        if value is True:
            check_close_before_toggle_on(target_obj)

        # Call the setter
        target_obj.states[object_states.ToggledOn].set_value(value)
        yield from self._settle_robot()

        # Check that it actually happened
        if target_obj.states[object_states.ToggledOn].get_value() != value:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.POST_CONDITION_ERROR,
                "The object did not toggle as expected - maybe try again",
                {
                    "target object": target_obj.name,
                    "is it currently toggled on": target_obj.states[object_states.ToggledOn].get_value(),
                },
            )
        
    def _wipe(self, target_obj: StatefulObject, cleaning_tool: StatefulObject):
        # Check that the cleaning tool can remove those particles
        if object_states.ParticleRemover not in cleaning_tool.states:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The cleaning tool is not a particle remover.",
                {"cleaning tool": cleaning_tool.name},
            )
        
        covered_systems = get_covered_systems(target_obj)
        # Check that the target object is coverable
        if covered_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not coverable by any particles, so there is no need to wipe it.",
                {"target object": target_obj.name},
            )
        
        check_open_before_grasp(target_obj, self.env)
        check_open_before_grasp(cleaning_tool, self.env)

        # Check if the target object has any particles on it
        if not covered_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not covered by any particles.",
                {"target object": target_obj.name},
            )
        
        supported_systems = get_supported_systems(
            cleaning_tool, covered_systems, object_states.ParticleRemover
        )
        if not supported_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is covered only by particles that this cleaning tool cannot remove.",
                {
                    "target object": target_obj.name,
                    "cleaning tool": cleaning_tool.name,
                    "particles the target object is covered by": sorted(x.name for x in covered_systems),
                    "particles the cleaning tool can remove": sorted(
                        [x for x in cleaning_tool.states[object_states.ParticleRemover].conditions.keys()]
                    ),
                },
            )
        
        removed_systems = get_modified_systems(
            cleaning_tool, supported_systems, object_states.ParticleRemover
        )
        if not removed_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is covered by some particles that this cleaning tool can normally remove, but needs to be in a different state to do so (e.g. toggled on, soaked by another fluid first, etc.).",
                {
                    "target object": target_obj.name,
                    "cleaning tool": cleaning_tool.name,
                    "particles the target object is covered by": sorted(x.name for x in covered_systems),
                },
            )
        
        # Fix object to prevent collision
        was_kinematic = False
        try:
            if hasattr(target_obj, 'kinematic_only'):
                was_kinematic = target_obj.kinematic_only
                target_obj.kinematic_only = True  # Fix object
            elif hasattr(target_obj, 'keep_still'):
                target_obj.keep_still()  # Fix object
        except Exception as e:
            print(f"[WIPE] Warning: Could not fix object: {e}")
        
        try:
            # If so, remove the particles on the target object
            MAX_WIPE_NUMS = 3
            for i in range(MAX_WIPE_NUMS):
                print(f"######[INFO] Try to Wipe at times {i}")
                for system in removed_systems:
                    target_obj.states[object_states.Covered].set_value(system, False)
                    yield from self._settle_robot()
                if not get_covered_systems(target_obj):  # wipe OK, exit early
                    break
        finally:
            # Restore object's kinematic state
            try:
                if hasattr(target_obj, 'kinematic_only'):
                    target_obj.kinematic_only = was_kinematic
            except:
                pass


    def _cut(self, target_obj: StatefulObject, cutting_tool: StatefulObject):
        # Check that cutting tool is a slicer
        if "slicer" not in cutting_tool._abilities:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The cutting tool cannot slice object.",
                {"cutting tool": cutting_tool.name},
            )
        
        # Check that the target object is sliceable
        if "sliceable" not in target_obj._abilities and "diceable" not in target_obj._abilities:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not sliceable or diceable.",
                {"target object": target_obj.name},
            )

        check_open_before_grasp(target_obj, self.env)
        check_open_before_grasp(cutting_tool, self.env)

        added_obj_attrs, removed_objs = [], []
        (slicing_rule,) = [
            rule 
            for rule in target_obj.scene.transition_rule_api.active_rules 
            if isinstance(rule, SlicingRule)
        ]
        output = slicing_rule.transition({"sliceable": [target_obj]})

        added_obj_attrs += output.add
        removed_objs += output.remove
        target_obj.scene.transition_rule_api.execute_transition(
            added_obj_attrs=added_obj_attrs, removed_objs=removed_objs
        )
        yield from self._settle_robot()
    
    def _soak_with_fluid_systems(self, target_obj: StatefulObject, systems: List[BaseSystem]) -> Optional[List[BaseSystem]]:
        # Check that the target object can saturated (remove) with particles in container or producer
        supported_systems = get_supported_systems(
            target_obj, systems, object_states.ParticleRemover
        )
        if not supported_systems:
            return None
        
        removed_systems = get_modified_systems(
            target_obj, supported_systems, object_states.ParticleRemover
        )
        if not removed_systems:
            return None
        
        return removed_systems

    def _soak_under(self, target_obj: StatefulObject, fluid_source: StatefulObject):
        # Check that the target object can saturated (remove) with particles
        if object_states.Saturated not in target_obj.states:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The currently grasped object cannot soak particles.",
                {"object in hand": target_obj.name},
            )

        check_open_before_grasp(target_obj, self.env)

        # Check that the fluid source should either be a particle producer or a particle container
        produced_systems = get_produced_systems(fluid_source)
        if produced_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid source is not a particle producer, so you can not soak target object.",
                {"fluid source": fluid_source.name},
            )
        if not produced_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid source currently does not produce particles, may be some conditions for producing particles not met, e.g., the fluid source should be toggled on.",
                {"fluid source": fluid_source.name}
            )
        
        removed_produced_systems = self._soak_with_fluid_systems(target_obj, produced_systems)
        if removed_produced_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object cannot soak with particles from fluid source, maybe target object cannot support the particles or saturation reaches upper limit",
                {
                    "target object": target_obj.name,
                    "fluid source": fluid_source.name,
                    "particles the target object can soak:": sorted(
                        [x for x in target_obj.states[object_states.ParticleRemover].conditions.keys()]
                    ),
                    "particles the fluid source produces": sorted(x.name for x in produced_systems) if produced_systems else None,
                },
            )
        
        # Remove the particles.
        for system in removed_produced_systems:
            target_obj.states[object_states.Saturated].set_value(system, True)
        
        yield from self._settle_robot()
    
    def _soak_inside(self, target_obj: StatefulObject, fluid_container: StatefulObject):
        # Check that the target object can saturated (remove) with particles
        if object_states.Saturated not in target_obj.states:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The currently grasped object cannot soak particles.",
                {"target object": target_obj.name},
            )

        check_open_before_grasp(target_obj, self.env)
        check_open_before_grasp(fluid_container, self.env)

        contained_systems = get_contained_systems(fluid_container)
        if contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid container is not a particle container, so you can not soak target object.",
                {"fluid container": fluid_container.name},
            )
        
        if not contained_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid source currently does not contain any particles.",
                {"fluid container": fluid_container.name}
            )
        
        removed_contained_systems = self._soak_with_fluid_systems(target_obj, contained_systems)
        if removed_contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object cannot soak with particles from fluid container, maybe target object cannot support the particles or saturation reaches upper limit",
                {
                    "target object": target_obj.name,
                    "fluid container": fluid_container.name,
                    "particles the target object can soak:": sorted(
                        [x for x in target_obj.states[object_states.ParticleRemover].conditions.keys()]
                    ),
                    "particles in the fluid container": sorted(x.name for x in contained_systems) if contained_systems else None,
                },
            )
        
        # Remove the particles.
        for system in removed_contained_systems:
            target_obj.states[object_states.Saturated].set_value(system, True)
        
        yield from self._settle_robot()

    def _fill_with(self, target_obj: StatefulObject, fluid_source: StatefulObject):
        # Check that target object is fillable
        contained_systems = get_contained_systems(target_obj)
        if contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not fillable by particles, so you can not fill anything in it.",
                {"target object": target_obj.name},
            )

        check_open_before_grasp(target_obj, self.env)

        # Check that the fluid source should be a particle producer
        produced_systems = get_produced_systems(fluid_source)
        if produced_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid source is not a particle producer, so you can not fill target object.",
                {"fluid source": fluid_source.name},
            )
        if not produced_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid source currently does not produce any particles, may be some conditions for producing particles not met, e.g., the fluid source should be toggled on.",
                {"fluid source": fluid_source.name}
            )
        
        # If so, fill the target object with all the particles from fluid source
        for system in produced_systems:
            target_obj.states[object_states.Filled].set_value(system, True)
            yield from self._settle_robot()

        # for system in produced_systems:
        #     if not target_obj.states[object_states.Contains].get_value(system):
        #         raise ActionPrimitiveError(
        #             ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
        #             "The container does not contain target particle, maybe the container fall over and the particles are scattered.",
        #             {"container": target_obj.name, "particle": system.name}
        #         )

    def _pour_into(self, fluid_container: StatefulObject, target_obj: StatefulObject):
        # Check that target object is fillable
        contained_systems = get_contained_systems(target_obj)
        if contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object is not fillable by particles, so you can not fill anything in it.",
                {"target object": target_obj.name},
            )

        check_open_before_grasp(target_obj, self.env)
        check_open_before_grasp(fluid_container, self.env)

        # Check that the fluid container contains particles
        contained_systems = get_contained_systems(fluid_container)
        if contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid container is not a particle container, so you can not fill target object.",
                {"fluid container": fluid_container.name},
            )
        if not contained_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The fluid container currently does not contain any particles.",
                {"fluid container": fluid_container.name}
            )

        # If so, fill the target object with all the particles from fluid source
        for system in contained_systems:
            target_obj.states[object_states.Filled].set_value(system, True)
            yield from self._settle_robot()

        # for system in contained_systems:
        #     if not target_obj.states[object_states.Contains].get_value(system):
        #         raise ActionPrimitiveError(
        #             ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
        #             "The container does not contain target particle, maybe the container fall over and the particles are scattered.",
        #             {"container": target_obj.name, "particle": system.name}
        #         )
    
    def _spread(self, fluid_container: StatefulObject, target_obj: StatefulObject):
        contained_systems = get_contained_systems(fluid_container)
        # check current object is a particle container
        if contained_systems is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The current container object is not fillable by particles, so you can not use it to spread",
                {"target object": fluid_container.name},
            )
        # Check if the current object has any particles in it
        if not contained_systems:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The current container object does not contain any particles.",
                {"target object": fluid_container.name},
            )

        check_open_before_grasp(fluid_container, self.env)
        check_open_before_grasp(target_obj, self.env)

        for system in contained_systems:
            if target_obj.prim_type != PrimType.CLOTH:
                target_obj.states[object_states.Covered].set_value(system, True)
            else: 
                target_obj.states[object_states.Saturated].set_value(system, True)
            yield from self._settle_robot()

    def _cook_particle_system(self, target_system: BaseSystem):
        container = get_container(target_system, self.env)
        if container is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                f"The target particle waiting for cooked should be contained in a container",
                {"target particle": target_system.name},
            )

        check_heat_source_before_cook(container, self.env)

        system_name = target_system.name
        cooked_system_name = f'cooked__{system_name}'
        cooked_system = get_cooked_system(cooked_system_name, self.env)
        if cooked_system is None:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                f"Cannot cook target particle.",
                {"target particle": target_system.name},
            )

        contained_particles = container.states[object_states.ContainedParticles].get_value(target_system)
        in_volume_idx = torch.where(contained_particles.in_volume)[0]
        if len(in_volume_idx) < 1:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                f"Container does not contain any target particles.",
                {"target particle": target_system.name, "container": container.name},
            )
        
        # Remove uncooked particles
        target_system.remove_particles(idxs=in_volume_idx)

        # Generate cooked particles
        particle_positions = contained_particles.positions[in_volume_idx]
        cooked_system.generate_particles(positions=particle_positions)

        yield from self._settle_robot()

        cooked_system = find_task_related_object(
            self.env, cooked_system_name, retain_wrapper=True
        )
        if not cooked_system.exists:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                f"Fail to add cooked particles.",
                {"target system": target_system.name, "cooked system": cooked_system_name},
            )

    def _cook_non_particle_object(self, target_obj: StatefulObject):
        heating_temperature = -1.0
        if object_states.Cooked in target_obj.states:
            heating_temperature = max(
                heating_temperature, target_obj.states[object_states.Cooked].cook_temperature
            )
        if object_states.Heated in target_obj.states:
            heating_temperature = max(
                heating_temperature, target_obj.states[object_states.Heated].heat_temperature
            )

        # Check that the current object is cookable or heatable
        if heating_temperature < 0:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "Target object is not cookable or heatable.",
                {"target object": target_obj.name},
            )
        heating_temperature = heating_temperature + 15

        check_heat_source_before_cook(target_obj, self.env)

        target_obj.states[object_states.Temperature].set_value(heating_temperature)
        target_obj.states[object_states.MaxTemperature].set_value(heating_temperature)
        yield from self._settle_robot()

        if not ((object_states.Cooked in target_obj.states and target_obj.states[object_states.Cooked].get_value()) \
              or (object_states.Heated in target_obj.states and target_obj.states[object_states.Heated].get_value())):
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "Target object is not cooked and heated.",
                {"target object": target_obj.name},
            )

    def _wait_for_cooked(self, target_obj: StatefulObject | BaseSystem):
        if isinstance(target_obj, BaseSystem):
            yield from self._cook_particle_system(target_obj)
        else:
            # check if cooked proberty
            if object_states.Cooked in target_obj.states:
                yield from self._cook_non_particle_object(target_obj)
            else: 
                # target_obj is a cook_tool
                contained_systems = get_contained_systems(target_obj)
                if contained_systems:
                    # cook the fluid system inside the container
                    for particle_system in contained_systems:
                        if hasattr(particle_system, 'is_fluid') and particle_system.is_fluid:
                            yield from self._cook_particle_system(particle_system)

                placement_objects = get_placement_objects(target_obj, self.env)
                if placement_objects:
                    for placement_object in placement_objects:
                        if object_states.Cooked in placement_object.states:
                            yield from self._cook_non_particle_object(placement_object)

    def _wait_for_washed(self, wash_machine: StatefulObject):
        if not (wash_machine.name.split("_")[0] == 'washer' or wash_machine.name.split("_")[0] == 'dishwasher'):
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "Wait_for_washed api must input with an wach_machine named 'washer'.",
                {"target object": wash_machine.name},
            )
        
        if wash_machine.states[object_states.Open].get_value():
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The washer is still open on so you are not able to toggled on to wash",
                {"target object": wash_machine.name},
            )

        if not wash_machine.states[object_states.ToggledOn].get_value():
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The washer is not toggled on so you are not able to wash",
                {"target object": wash_machine.name},
            )
        
        inside_placements = get_placement_objects(wash_machine, self.env, object_states.Inside)
        if not inside_placements:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "There are no objects inside the wash machine, so there is no need to wash",
                {"target object": wash_machine.name},
            )
     
        for inside_placement in inside_placements:
            inside_obj = inside_placement.object
            covered_systems = get_covered_systems(inside_obj)
            if covered_systems is None:
                raise ActionPrimitiveError(
                    ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                    "The target object inside the washer is not coverable by any particles, so there is no need to wipe it.",
                    {"target object": inside_obj.name},
                )
            if not covered_systems:
                raise ActionPrimitiveError(
                    ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                    "The target object inside the washer is not covered by any particles, so there is no need to wipe it.",
                    {"target object": inside_obj.name},
                )
            for system in covered_systems:
                inside_obj.states[object_states.Covered].set_value(system, False)
                yield from self._settle_robot()

    def _wait(self, target_obj: StatefulObject):
        if hasattr(target_obj, 'states'):
            if object_states.Frozen in target_obj.states:
                target_obj.states[object_states.Frozen].set_value(False)
            if object_states.Heated in target_obj.states:
                target_obj.states[object_states.Heated].set_value(False)

        yield from self._settle_robot()
        
    def _wait_for_frozen(self, target_obj: StatefulObject, refrigerator_obj: StatefulObject):
        if not 'fridge' in refrigerator_obj.name:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The second arguments should be refriderator",
                {"refrigerator_obj": refrigerator_obj.name},
            )
            
        if hasattr(target_obj, 'states') and object_states.Frozen in target_obj.states:
            if target_obj.states[object_states.Inside].get_value(refrigerator_obj):
                target_obj.states[object_states.Frozen].set_value(True)
                yield from self._settle_robot()
            else: 
                raise ActionPrimitiveError(
                    ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                    "The target object is not inside the refrigerator",
                    {"target_obj": target_obj.name},
                )
        else:
            raise ActionPrimitiveError(
                ActionPrimitiveError.Reason.PRE_CONDITION_ERROR,
                "The target object have no frozen states to make if frozen",
                {"target_obj": target_obj.name},
            )
    
    def _convert_action_to_nl(self, plan: str) -> str:
        """Convert Action to Natural Language"""
        if not self.guardrail_model or not hasattr(self.guardrail_model, 'model'):
            return f"Execute action: {plan}"
        
        try:
            from og_ego_prim.utils.prompts import ActionToNLPrompt
            prompt = ActionToNLPrompt + f'\n- action: "{plan}"\n\nOutput:'
            # Use EMBGuard's model to convert to NL
            messages = [{"role": "user", "content": prompt}]
            response = self.guardrail_model.model.generate_with_retry(messages)
            action_nl = response.get("content", "").strip()
            if not action_nl:
                action_nl = f"Execute action: {plan}"
            return action_nl
        except Exception as e:
            return f"Execute action: {plan}"
                        