# simulation_app.close()
from isaacsim import SimulationApp
simulation_app = SimulationApp({"headless":True})

# load external package
import os
import sys
import time
import numpy as np
import open3d as o3d
from termcolor import cprint
import re
import threading
import json
# load isaac-relevant package
import omni.replicator.core as rep
import isaacsim.core.utils.prims as prims_utils
from pxr import UsdGeom,UsdPhysics,PhysxSchema, Gf
from isaacsim.core.api import World
from isaacsim.core.api.objects import DynamicCuboid, FixedCuboid, VisualCuboid
from isaacsim.core.utils.prims import is_prim_path_valid, set_prim_visibility, delete_prim
from isaacsim.core.utils.string import find_unique_string_name
from isaacsim.core.utils.viewports import set_camera_view
from isaacsim.core.utils.stage import add_reference_to_stage, is_stage_loading
from isaacsim.core.prims import SingleXFormPrim, SingleRigidPrim
from isaacsim.core.utils.rotations import euler_angles_to_quat
from omni.physx.scripts import physicsUtils

# load custom package
sys.path.append(os.getcwd())
from Env_StandAlone.BaseEnv import BaseEnv
from Env_Config.Robot.Bimanual_Franka import Bimanual_Franka
from Env_Config.Camera.Recording_Camera import Recording_Camera
from Env_Config.Room.Real_Ground import Real_Ground
from Env_Config.Room.Object_Tools import set_prim_visible_group
from Env_Config.Utils_Project.Code_Tools import get_unique_filename
from Env_Config.Utils_Project.Parse import parse_args_record
from Env_Config.Utils_Project.Point_Cloud_Manip import furthest_point_sampling
from Env_Config.Utils_Project.Object_Contact_Config import get_contact_config
from Env_Config.Table.Table import Table
from Env_Config.Flat_Object.Flat_Object import Rigid_Flat_Object
from Env_Config.Utils_Project.Pos_Ori import calculate_target_from_world_contact, get_xy_move_from_angle

# Convert data to serializable format
def _to_serializable(value):
    if isinstance(value, np.ndarray):
        return value.tolist()
    if isinstance(value, (np.integer, np.floating)):
        return value.item()
    if isinstance(value, (list, tuple)):
        return [_to_serializable(v) for v in value]
    if isinstance(value, dict):
        return {k: _to_serializable(v) for k, v in value.items()}
    return value

def _write_point_cloud_ply3(
    file_path: str,
    points_cloud,
    colors=None,
    save_or_not: bool = True,
    sample_flag: bool = True,
    sampled_point_num: int = 8192,
    real_time_watch: bool = False,
):
    if points_cloud is None:
        return
    pts = np.asarray(points_cloud)
    if sample_flag:
        points_cloud, colors = furthest_point_sampling(points_cloud, colors, sampled_point_num)
    pcd = o3d.geometry.PointCloud()
    pcd.points = o3d.utility.Vector3dVector(points_cloud)
    pcd.colors = o3d.utility.Vector3dVector(colors)
    if real_time_watch:
        o3d.visualization.draw_geometries([pcd])
    if save_or_not:
        o3d.io.write_point_cloud(file_path, pcd)

def _write_pose_txt(file_path: str, sample: dict):
    payload = {
        "joint_pos_L": _to_serializable(sample.get("joint_pos_L")),
        "joint_pos_R": _to_serializable(sample.get("joint_pos_R")),
        "ee_pose_L": _to_serializable(sample.get("ee_pose_L")),
        "ee_pose_R": _to_serializable(sample.get("ee_pose_R")),
    }
    with open(file_path, "w", encoding="utf-8") as handle:
        json.dump(payload, handle, ensure_ascii=True, indent=2)

def save_mid_samples(samples, target_dir: str):
    if not samples:
        cprint("[Save] No callback samples captured; skip mid outputs", "yellow")
        return
    os.makedirs(target_dir, exist_ok=True)
    for name in os.listdir(target_dir):
        if name.startswith("Mid") and (name.endswith("_env.ply") or name.endswith("_pos.txt")):
            try:
                os.remove(os.path.join(target_dir, name))
            except OSError:
                pass
    for idx, sample in enumerate(samples, start=1):
        env_path = os.path.join(target_dir, f"Mid{idx}_env.ply")
        pose_path = os.path.join(target_dir, f"Mid{idx}_pos.txt")
        _write_point_cloud_ply3(
            env_path,
            sample.get("env_point_cloud"),
            sample.get("env_point_cloud_color"),
        )
        _write_pose_txt(pose_path, sample)

class WearScarf_Env(BaseEnv):
    def __init__(
        self,
        pos: np.ndarray = None,
        ori: np.ndarray = None,
        usd_path: str = None,
        env_dx: float = 0.0,
        env_dy: float = 0.0,
        object_name: str = "Cutting_Board3",
        object_random_init: bool = False,
        ground_material_usd: str = None,
        record_video_flag: bool = False,
    ):
        super().__init__()
        self.scene = self.world.get_physics_context()._physics_scene
        self.scene.CreateGravityDirectionAttr().Set(Gf.Vec3f(0, 0.0, -1))
        self.scene.CreateGravityMagnitudeAttr().Set(5.5)
        self.ground = Real_Ground(
            self.scene, 
            visual_material_usd = ground_material_usd,
        )
        
        self.bimanual_franka = Bimanual_Franka(
            self.world,
            left_pos=np.array([0.57325, 0.52302,  0.52954]),
            left_ori=np.array([0.0, 0.0, -135.0]),
            left_franka_scale=np.array([1.0, 1.0, 1.0]),
            right_pos=np.array([-0.57325, 0.52302,  0.52954]),
            right_ori=np.array([0.0, 0.0, -45.0]),
            right_franka_scale=np.array([1.0, 1.0, 1.0]),
            gripper_open_position_nounits=np.array([0.05, 0.05])
        )
        
        self.Table = Table(
            self.world, 
            path=os.getcwd() + "/" +"Assets/Table/Collected_Willow/Wood.usd",
            position=[0.0, 0.02152, 0.0],
            orientation=[0.0, 0.0, 0.0],
            scale = np.array([0.0088, 0.0104, 0.01])
        )

        self.object_name = object_name
        category_name = re.sub(r"\d+$", "", self.object_name)
        self.object = Rigid_Flat_Object(
            self.world,
            name=self.object_name,
            path=os.path.join(
                os.getcwd(), "Assets/Flat_Object", category_name, f"{self.object_name}.usd"
            ),
            random_init=object_random_init,
            type = 2,
            scale=np.array([0.8, 0.8, 1]),
        )

        self.obj_camera = Recording_Camera(
            camera_position=np.array([0.0, -3.1, 3.8]),
            camera_orientation=np.array([0.92388, 0.38268, 0.0, 0.0]),
            resolution=(600, 400),
            prim_path="/World/obj_camera",
        )
        self.env_camera = Recording_Camera(
            camera_position=np.array([0.0, -3.1, 3.8]),
            camera_orientation=np.array([0.92388, 0.38268, 0.0, 0.0]),
            resolution=(600, 400),
            prim_path="/World/env_camera",
        )
        self.recording_camera = Recording_Camera(
            camera_position=np.array([0.0, -5.04154, 3.61856]),
            camera_orientation=np.array([0.85717, 0.51504, 0.0, 0.0]),
            resolution=(2048, 1080),
            prim_path="/World/recording_camera",
        )
        cube = FixedCuboid(
            prim_path="/World/Cube",
            name="my_cube",
            position=np.array([-0.7, 0.0, 0.9]),
            scale=np.array([0.01, 0.01, 0.01]),
            color=np.array([0.0, 0.5, 1.0]),
        )
        self.judge_camera = Recording_Camera(
            camera_position=np.array([2.23, 0, 0.9]),
            camera_orientation=np.array([0.5,0.5, 0.5, 0.5]),
            resolution=(600, 400),
            prim_path="/World/obj_only_camera",
        )

        self.env_pcd = None
        self.flat_object_pcd = None
        self.saved_callback_files = []
        self.name_slug = (self.object_name or "").lower()
        self.pos_index = None
        self.callback_save_count = 0

        self.reset()

        if record_video_flag:
            self.thread_record = threading.Thread(
                target=self.recording_camera.collect_rgb_graph_for_video
            )
            self.thread_record.daemon = True
        
        self.recording_camera.initialize()
        self.obj_camera.initialize(
            segment_pc_enable=True,
            segment_prim_path_list=[f"/World/Rigid_Flat_Object/{self.object_name}"],
        )

        self.judge_camera.initialize(
            segment_pc_enable=True, 
            segment_prim_path_list=[
                f"/World/Rigid_Flat_Object/{self.object_name}",
                "/World/Table",
            ],
        )

        self.env_camera.initialize(
            segment_pc_enable=True,
            segment_prim_path_list=[
                f"/World/Rigid_Flat_Object/{self.object_name}",
                "/World/Table",
            ],
        )
        self.obj_only_camera = Recording_Camera(
            camera_position=np.array([2.23, 0, 0.9]),
            camera_orientation=np.array([0.5,0.5, 0.5, 0.5]),
            resolution=(600, 400),
            prim_path="/World/obj_only_camera",
        )

        self.franka_only_camera = Recording_Camera(
            camera_position=np.array([2.23, 0, 0.9]),
            camera_orientation=np.array([0.5,0.5, 0.5, 0.5]),
            resolution=(600, 400),
            prim_path="/World/franka_only_camera",
        )

        self.obj_only_camera.initialize(segment_pc_enable=True, segment_prim_path_list=[f"/World/Garment"])
        self.franka_only_camera.initialize(segment_pc_enable=True, segment_prim_path_list=[f"/World/Franka_Left"])
        
        for i in range(100):
            self.step()
        
        cprint("World Ready!", "green", "on_green")

    def record_callback(self, step_size):
        if self.step_num % 5 == 0:
            joint_pos_L = self.bimanual_franka.left_franka.get_joint_positions()
            joint_pos_R = self.bimanual_franka.right_franka.get_joint_positions()
            joint_state = np.array([*joint_pos_L, *joint_pos_R])
            self.callback_save_count += 1
            self.env_pcd, env_color = self.env_camera.get_point_cloud_data_from_segment(
                save_or_not=False,
                sample_flag=True,
                sampled_point_num=8192,
                real_time_watch=False,
            )

            pos_L, quat_L = self.bimanual_franka.left_franka.get_cur_ee_pos()
            pos_R, quat_R = self.bimanual_franka.right_franka.get_cur_ee_pos()
            ee_pose_L = np.array([*pos_L, *quat_L])
            ee_pose_R = np.array([*pos_R, *quat_R])
            self.saving_data.append({
                "env_point_cloud": self.env_pcd,
                "env_point_cloud_color": env_color,
                "joint_pos_L": joint_pos_L,
                "joint_pos_R": joint_pos_R,
                "ee_pose_L": ee_pose_L,
                "ee_pose_R": ee_pose_R,
            })
        
        self.step_num += 1

def get_point_cloud(env):
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Franka_Left","/World/Franka_Right"], visible=False)
    for i in range(2):
        env.step()
    judge_flat_object_pcd, _ = env.obj_only_camera.get_point_cloud_data_from_segment(
        save_or_not=False,
        sampled_point_num=8192,
        real_time_watch=False
    )    
    set_prim_visible_group(prim_path_list=["/World/Table","/World/Franka_Left","/World/Franka_Right"], visible=True)

    for i in range(10):
        env.step()

    return judge_flat_object_pcd

def WearScarf(pos, ori, usd_path, env_dx, env_dy, object_name, ground_material_usd, data_collection_flag, record_video_flag, object_random_init=False, pos_index=None):
    env = WearScarf_Env(pos, ori, usd_path, env_dx, env_dy, object_name, object_random_init, ground_material_usd, record_video_flag)
    cprint(f"[Info]: data_collection_flag: {data_collection_flag}", "green", attrs=["bold"])
    cprint(f"[Info]: record_video_flag: {record_video_flag}", "green", attrs=["bold"])
    cprint(f"[Info]: object_name: {object_name}", "green", attrs=["bold"])

    name_slug = (object_name or getattr(env, "object_name", None) or getattr(env.object, "name", None))
    if not name_slug:
        raise ValueError("object_name is not provided and cannot be inferred from the environment; please pass it explicitly via --object_name")
    name_slug = name_slug.lower()

    object_name_initial = getattr(env.object, "name", "None")
    contact_cfg = get_contact_config(object_name_initial)
    env.name_slug = name_slug
    env.pos_index = pos_index if pos_index is not None else 1
    env.callback_save_count = 0

    env.bimanual_franka.Gripper_Both_Open()
    env.object.set_mass(contact_cfg.get("mass", 0.08))
    cprint(f"{env.object.get_mass()}")

    if record_video_flag:
        env.thread_record.start()
        cprint("record video start", "green", attrs=["bold"])

    pcd_files_main = []
    video_path = None
    npz_path = None        

    task_name = "StrategyB_V2"
    if not os.path.exists(f"Data/{task_name}/train_data/"):
                os.makedirs(f"Data/{task_name}/train_data/")
    if not os.path.exists(f"Data/{task_name}/pointcloud/"):
                os.makedirs(f"Data/{task_name}/pointcloud/")

    set_prim_visible_group(prim_path_list=["/World/Franka_Left","/World/Franka_Right"], visible=False)
    for i in range(2):
        env.step()
    whole_point_cloud_pcd, whole_point_cloud_pcd_color = env.env_camera.get_point_cloud_data_from_segment(
        save_or_not=False,
        sampled_point_num=8192,
        real_time_watch=False
    )    
    set_prim_visible_group(prim_path_list=["/World/Franka_Left","/World/Franka_Right"], visible=True)

    c_target_position_left, c_target_orientation_left = calculate_target_from_world_contact(
        contact_world_init=contact_cfg["contact_world_init_left"],
        initial_board_position=contact_cfg["initial_board_position"],
        initial_board_euler=contact_cfg.get("initial_board_euler"),
        initial_gripper_orientation_euler=contact_cfg.get("initial_gripper_orientation_euler_left"),
        current_board_position=env.object.my_position,
        current_board_euler=env.object.my_orientation
    )
    c_target_position_right, c_target_orientation_right = calculate_target_from_world_contact(
        contact_world_init=contact_cfg["contact_world_init_right"],
        initial_board_position=contact_cfg["initial_board_position"],
        initial_board_euler=contact_cfg.get("initial_board_euler"),
        initial_gripper_orientation_euler=contact_cfg.get("initial_gripper_orientation_euler_right"),
        current_board_position=env.object.my_position,
        current_board_euler=env.object.my_orientation
    )

    cprint(f"[Info]: Move to both sides of the object for pre-contact","white",attrs=["bold"])
    
    e_mx,e_my =  get_xy_move_from_angle(env.object.my_orientation[2], 0.05)  
    left_target_position_x = c_target_position_left[0] + 0.6 * e_mx
    left_target_position_y = c_target_position_left[1] + 0.6 * e_my

    right_target_position_x = c_target_position_right[0] - 0.6 * e_mx
    right_target_position_y = c_target_position_right[1] - 0.6 * e_my

    cprint(f"[Info]: Move to safe distance ","white",attrs=["bold"])
    env.bimanual_franka.Rmpflow_Both_Move(
        left_target_position=np.array([left_target_position_x + 0.05, left_target_position_y ,   0.9]),
        right_target_position=np.array([right_target_position_x - 0.05, right_target_position_y , 0.9]),
        left_target_orientation=c_target_orientation_left,
        right_target_orientation=c_target_orientation_right
    )

    cprint(f"[Info]: test_model_start ","white",attrs=["bold"])
    env.bimanual_franka.Rmpflow_Both_Move(
        left_target_position=np.array([left_target_position_x + 0.05, left_target_position_y , c_target_position_left[2]]),
        right_target_position=np.array([right_target_position_x - 0.05, right_target_position_y , c_target_position_right[2]]),
        left_target_orientation=c_target_orientation_left,
        right_target_orientation=c_target_orientation_right
    )

    new_euler_deg_l = c_target_orientation_left + np.array([0.0, 20, 0.0])
    new_euler_deg_r = c_target_orientation_right + np.array([0.0, 20, 0.0])

    e_mx,e_my =  get_xy_move_from_angle(env.object.my_orientation[2], contact_cfg.get("move_x",0.015))  

    left_euler  = new_euler_deg_l
    right_euler = new_euler_deg_r

    left_euler_4 = euler_angles_to_quat(left_euler[0],left_euler[1],left_euler[2])
    right_euler_4 = euler_angles_to_quat(right_euler[0],right_euler[1],right_euler[2])

    cur_lx = float(left_target_position_x + 0.05 )
    cur_rx = float(right_target_position_x - 0.05 )
    cur_ly = float(left_target_position_y )
    cur_ry = float(right_target_position_y )

    fixed_z = c_target_position_left[2]

    contact_detected = False
    max_iters = 1000

    cprint(f"[Info]: Start inward pushing movement new_euler_deg_r:{new_euler_deg_r}","red",attrs=["bold"])
    for i in range(max_iters):
        cprint(f"[Info]: Inward pushing iteration step {i}","white",attrs=["bold"])
        next_lx = cur_lx - e_mx
        next_rx = cur_rx + e_mx

        if new_euler_deg_r[2] >= 0:
            code = env.bimanual_franka.Rmpflow_Both_Move(
                left_target_position=(np.array([next_lx, cur_ly - e_my, fixed_z])),
                right_target_position=(np.array([next_rx, cur_ry + e_my, fixed_z])),
                left_target_orientation=new_euler_deg_l,
                right_target_orientation=new_euler_deg_r,
            )
        else:
            code = env.bimanual_franka.Rmpflow_Both_Move(
            left_target_position=(np.array([next_lx, cur_ly + e_my, fixed_z])),
            right_target_position=(np.array([next_rx, cur_ry - e_my, fixed_z])),
            left_target_orientation=new_euler_deg_l,
            right_target_orientation=new_euler_deg_r,
            )

        env.world.step()

        if int(code) == int(100):
            cprint("[Info]: Contact detected", "green", attrs=["bold"])
            contact_detected = True
            cprint(f"[Info]: Collect start point information individually","white")

            joint_pos_L = env.bimanual_franka.left_franka.get_joint_positions()
            joint_pos_R = env.bimanual_franka.right_franka.get_joint_positions()
            pos_L, quat_L = env.bimanual_franka.left_franka.get_cur_ee_pos()
            pos_R, quat_R = env.bimanual_franka.right_franka.get_cur_ee_pos()
            ee_pose_L = np.array([*pos_L, *quat_L])
            ee_pose_R = np.array([*pos_R, *quat_R])
            env.saving_data_replay["start_joint_pos_L"] = joint_pos_L
            env.saving_data_replay["start_joint_pos_R"] = joint_pos_R
            env.saving_data_replay["start_ee_pose_L"] = ee_pose_L
            env.saving_data_replay["start_ee_pose_R"] = ee_pose_R
            start_payload = {
                "joint_pos_L": _to_serializable(joint_pos_L),
                "joint_pos_R": _to_serializable(joint_pos_R),
                "ee_pose_L": _to_serializable(ee_pose_L),
                "ee_pose_R": _to_serializable(ee_pose_R),
            }
            env.start_pose_payload = start_payload
            if data_collection_flag:
                env.stop_record()
            for i in range(2):
                env.step()
            break

        cur_lx, cur_rx = next_lx, next_rx

        if (abs(cur_lx - final_left_x)  < 1e-6) and (abs(cur_rx - final_right_x) < 1e-6):
            cprint("✗ No contact detected, reached the set innermost target. Please check object position/threshold.", "red")
            break

    cprint(f"[Info]: Lifting up","green",attrs=["bold"])
    env.bimanual_franka.Rmpflow_Both_Move(
        left_target_position=(np.array([next_lx, cur_ly, fixed_z + 0.02])),
        right_target_position=(np.array([next_rx, cur_ry, fixed_z + 0.02])),
        left_target_orientation=new_euler_deg_l,
        right_target_orientation=new_euler_deg_r,
    )

    obj_pos, _ = env.object.rigid_form.get_world_pose()
    pos_e, _ = env.bimanual_franka.left_franka.get_cur_ee_pos()
    table_pos, _ = env.Table.rigid_form.get_world_pose()
    dist_gripper_to_obj = np.linalg.norm(pos_e - obj_pos)

    success_flag = False

    obj_point = get_point_cloud(env)
    obj_z_min = np.min(obj_point[:, 2])
    cprint(f"object z min:{obj_z_min}", "yellow", attrs=["bold"])   

    if contact_detected:
        if obj_z_min - 0.7603 > 0.0001:
            success_flag = True
            cprint("WearScarf Task Success!", "green")
        else:
            cprint("WearScarf Task Fail!", "red")

    if data_collection_flag:
        for i in range(2):
            env.step()
        judge_point_cloud_pcd, judge_point_cloud_pcd_color = env.judge_camera.get_point_cloud_data_from_segment(
            save_or_not=False,
            sampled_point_num=8192,
            real_time_watch=False
        )    

    if data_collection_flag:
        pos_idx_final = pos_index if pos_index is not None else (getattr(env, "pos_index", None) or 1)
        outcome_flag = "T" if success_flag else "F"
        base_dir = os.path.join("Data", task_name, "train_data", object_name)
        record_dir = os.path.join(base_dir, f"record_{pos_idx_final}_{outcome_flag}")
        os.makedirs(record_dir, exist_ok=True)

        _write_point_cloud_ply3(
            record_dir + f"/initial_env.ply",
            whole_point_cloud_pcd,
            whole_point_cloud_pcd_color,
        )        
        _write_point_cloud_ply3(
            record_dir + f"/judge_env.ply",
            judge_point_cloud_pcd,
            judge_point_cloud_pcd_color,
        )   
        start_payload = getattr(env, "start_pose_payload", None)
        if start_payload:
            _write_pose_txt(os.path.join(record_dir, "start_pos.txt"), start_payload)
        if start_payload:
            _write_pose_txt(os.path.join(record_dir," end_pos.txt"), start_payload)

    if record_video_flag:
        try:
            env.recording_camera.capture = False
            if hasattr(env, "thread_record") and env.thread_record.is_alive():
                env.thread_record.join(timeout=2)
        except Exception as e:
            cprint(f"[video-thread] stop failed: {e}", "red")

    if record_video_flag:
        sub_folder = "success" if success_flag else "fail"
        video_dir = f"Data/{task_name}/Video/{sub_folder}"
        os.makedirs(video_dir, exist_ok=True)

        if pos_index is not None:
            video_path = f"{video_dir}/{object_name}_pos{pos_index}.mp4"
        else:
            video_path = get_unique_filename(f"{video_dir}/{name_slug}", ".mp4")
        
        env.recording_camera.create_mp4(video_path)
        
    for i in range(2000):
        env.step()

    simulation_app.close()
    cprint(f"[Info]: Collection Finished!", "green", attrs=["bold"])
    return success_flag

if __name__=="__main__":
    args = parse_args_record()
    
    pos = np.array([0.0, 0.30, 0.65])
    ori = np.array([90.0, 0.0, 0.0])
    usd_path = None
    env_dx = 0.0
    env_dy = 0.0

    if args.env_random_flag or args.garment_random_flag:
        np.random.seed(int(time.time()))
        if args.env_random_flag:
            env_dx = np.random.uniform(-0.05, 0.1)
            env_dy = np.random.uniform(-0.05, 0.05)
        if args.garment_random_flag:
            x = np.random.uniform(-0.05, 0.05)
            y = np.random.uniform(0.30, 0.40)
            pos = np.array([x,y,0.0])
            ori = np.array([90.0, 0.0, 0.0])
            Base_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
            assets_lists = os.path.join(Base_dir,"Model_HALO/GAM/checkpoints/Scarf/assets_training_list.txt")
            assets_list = []
            with open(assets_lists,"r",encoding='utf-8') as f:
                for line in f:
                    clean_line = line.rstrip('\n')
                    assets_list.append(clean_line)
            usd_path=os.getcwd() + "/" + np.random.choice(assets_list)

    pos_index = None
    _pos_env = os.environ.get("POS_INDEX")
    if _pos_env and _pos_env.isdigit():
        pos_index = int(_pos_env)

    try:
        cprint(
            f"[Args] object_name={args.object_name} data_collection_flag={args.data_collection_flag} record_video_flag={args.record_video_flag} POS_INDEX={pos_index}",
            "yellow",
            attrs=["bold"],
        )
    except Exception:
        pass

    ok = WearScarf(
        pos, ori,
        usd_path,
        env_dx, env_dy,
        args.object_name,
        args.ground_material_usd,
        args.data_collection_flag,
        args.record_video_flag,
        args.object_random_init,
        pos_index,
    )

    if args.data_collection_flag:
        simulation_app.close()
        sys.exit(0 if ok else 1)
    else:
        while simulation_app.is_running():
            simulation_app.update()

simulation_app.close()