import os
import uuid
import time
import base64
import signal
import logging
import threading
import traceback
import numpy as np
from PIL import Image
from io import BytesIO
from queue import Queue
from concurrent.futures import ThreadPoolExecutor
from flask import Flask, render_template, request, jsonify
from flask_socketio import SocketIO, emit
from dataclasses import dataclass
from typing import List, Optional


def convert_numpy_values(obj):
    """Convert numpy values to native Python types for JSON serialization"""
    if isinstance(obj, np.ndarray):
        return [convert_numpy_values(item) for item in obj]
    elif isinstance(obj, list):
        return [convert_numpy_values(item) for item in obj]
    elif isinstance(obj, dict):
        return {key: convert_numpy_values(value) for key, value in obj.items()}
    elif hasattr(obj, "item") and hasattr(obj, "dtype"):  # Check if it's a numpy scalar
        return obj.item()  # Convert to native Python type
    else:
        return obj


@dataclass
class CameraState:
    """Stores the state of a camera for web server display and control"""

    position: List[float]
    orientation: List[float]  # Quaternion [x, y, z, w]
    movement_direction: List[float]  # [forward/back, right/left, up/down]
    roll_input: float
    tracking_robot: Optional[str] = None
    track_height: float = 1.0
    # Mouse/rotation control
    pending_mouse_pitch: float = 0.0
    pending_mouse_yaw: float = 0.0
    bird_view_mode: bool = False

    def to_dict(self):
        """Convert to a dictionary for JSON serialization"""
        # Convert all values to ensure JSON serializable
        state_dict = {
            "position": self.position,
            "orientation": self.orientation,
            "movement_direction": self.movement_direction,
            "roll_input": self.roll_input,
            "tracking_robot": self.tracking_robot,
            "track_height": self.track_height,
            "pending_mouse_pitch": self.pending_mouse_pitch,
            "pending_mouse_yaw": self.pending_mouse_yaw,
            "bird_view_mode": self.bird_view_mode,
        }

        # Ensure all values are JSON serializable
        return convert_numpy_values(state_dict)

    @classmethod
    def from_dict(cls, data):
        """Create a CameraState from a dictionary"""
        # Convert any potential NumPy values to standard Python types
        data = convert_numpy_values(data)

        return cls(
            position=data.get("position", [0.0, 0.0, 1.0]),
            orientation=data.get("orientation", [0.0, 0.0, 0.0, 1.0]),
            movement_direction=data.get("movement_direction", [0.0, 0.0, 0.0]),
            roll_input=data.get("roll_input", 0.0),
            tracking_robot=data.get("tracking_robot"),
            track_height=data.get("track_height", 1.0),
            pending_mouse_pitch=data.get("pending_mouse_pitch", 0.0),
            pending_mouse_yaw=data.get("pending_mouse_yaw", 0.0),
            bird_view_mode=data.get("bird_view_mode", False),
        )


class SimulationWebServer:
    """Web server for displaying camera feeds from multiple simulations"""

    def __init__(self, host="0.0.0.0", port=8080, column_count=3):
        # Get the absolute path of the current directory
        self.base_dir = os.path.abspath(os.path.dirname(__file__))

        # Set up templates and static directories with absolute paths
        self.templates_dir = os.path.join(self.base_dir, os.pardir, "web", "templates")
        self.static_dir = os.path.join(self.base_dir, os.pardir, "web", "static")

        # Create directories
        os.makedirs(self.templates_dir, exist_ok=True)
        os.makedirs(self.static_dir, exist_ok=True)

        # Configure logging
        logging.basicConfig(level=logging.INFO)
        self.logger = logging.getLogger(__name__)

        # Initialize Flask with the specific template folder
        self.app = Flask(
            __name__, template_folder=self.templates_dir, static_folder=self.static_dir
        )

        # Continue with initialization
        self.socketio = SocketIO(
            self.app, cors_allowed_origins="*", async_mode="threading"
        )
        self.host = host
        self.port = port
        self.column_count = column_count
        self.simulations = {}  # Stores information about connected simulations
        self.camera_states = {}  # Stores camera states for each simulation
        self.heartbeat_timeout = (
            10  # seconds before considering a simulation disconnected
        )
        self.stale_check_timer = None
        self.setup_routes()
        self.setup_socketio()
        self.server_thread = None

        self.current_title = "Rise Simulation Viewer"

    def setup_routes(self):
        """Set up the Flask routes"""

        @self.app.route("/")
        def index():
            # Log information about template rendering
            self.logger.info(f"Rendering index.html from {self.templates_dir}")
            try:
                return render_template("index.html", title=self.current_title)
            except Exception as e:
                self.logger.error(f"Error rendering template: {e}")
                traceback.print_exc()
                return f"Error rendering template: {e}", 500

        @self.app.route("/api/simulations", methods=["GET"])
        def get_simulations():
            return jsonify(list(self.simulations.values()))

        @self.app.route("/api/simulation/<sim_id>", methods=["GET"])
        def get_simulation(sim_id):
            if sim_id in self.simulations:
                return jsonify(self.simulations[sim_id])
            return jsonify({"error": "Simulation not found"}), 404

    def setup_socketio(self):
        """Set up the Socket.IO event handlers"""

        @self.socketio.on("connect")
        def handle_viewer_connect():
            """Handle new web viewer client connections"""
            client_id = request.sid
            self.logger.info(f"Viewer client connected: {client_id}")

            # Send the current state of all simulations to the new client
            for sim_id, sim_info in self.simulations.items():
                self.logger.info(
                    f"Sending simulation {sim_id} state to new viewer client {client_id}"
                )

                # Make sure to include the robot list in the simulation update
                emit("viewer_simulation_update", sim_info)

                # Also send an explicit robot list update to ensure it's received
                if "robots" in sim_info and sim_info["robots"]:
                    self.logger.info(
                        f"Sending robot list for {sim_id} to new viewer client {client_id}: {sim_info['robots']}"
                    )
                    emit(
                        "viewer_robot_list",
                        {"sim_id": sim_id, "robots": sim_info["robots"]},
                    )

        @self.socketio.on("disconnect")
        def handle_viewer_disconnect():
            """Handle web viewer client disconnections"""
            self.logger.info(f"Viewer client disconnected: {request.sid}")

        @self.socketio.on("client_heartbeat")
        def handle_client_heartbeat(data):
            """Handle heartbeat signals from simulation clients"""
            sim_id = data.get("sim_id")
            if sim_id and sim_id in self.simulations:
                self.simulations[sim_id]["last_heartbeat"] = time.time()
                self.logger.debug(f"Received heartbeat from simulation client {sim_id}")
                return {"success": True}
            return {"success": False, "error": "Invalid simulation ID"}

        @self.socketio.on("client_disconnect")
        def handle_client_disconnect(data):
            """Handle explicit simulation client disconnection"""
            sim_id = data.get("sim_id")
            if sim_id:
                self.logger.info(f"Simulation client {sim_id} explicitly disconnected")
                self.remove_simulation(sim_id)
                return {"success": True}
            return {"success": False, "error": "Invalid simulation ID"}

        @self.socketio.on("client_register")
        def handle_client_register(data):
            """Register a new simulation client with the server"""
            sim_id = data.get("sim_id", str(uuid.uuid4()))

            if sim_id not in self.simulations:
                self.simulations[sim_id] = {
                    "sim_id": sim_id,
                    "sim_name": data.get("sim_name", f"Simulation {sim_id}"),
                    "timestamp": time.time(),
                    "last_heartbeat": time.time(),  # Initialize heartbeat timestamp
                    "robots": [],
                }

                # Initialize camera state for this simulation
                self.camera_states[sim_id] = CameraState(
                    position=[0.0, 0.0, 1.0],
                    orientation=[0.0, 0.0, 0.0, 1.0],
                    movement_direction=[0.0, 0.0, 0.0],
                    roll_input=0.0,
                )

                self.logger.info(f"Registered new simulation client: {sim_id}")
            else:
                # Update heartbeat for existing simulation
                self.simulations[sim_id]["last_heartbeat"] = time.time()

            self.simulations[sim_id]["timestamp"] = time.time()

            # Update camera state if provided
            if "camera_state" in data:
                self.camera_states[sim_id] = CameraState.from_dict(data["camera_state"])

            # Update robots list if provided
            if "robots" in data:
                self.simulations[sim_id]["robots"] = data["robots"]
                self.logger.info(f"Updated robot list for {sim_id}: {data['robots']}")
                # Broadcast robot list update to all viewer clients
                emit(
                    "viewer_robot_list",
                    {"sim_id": sim_id, "robots": data["robots"]},
                    broadcast=True,
                )

            # Broadcast to all viewer clients
            self.simulations[sim_id]["camera_state"] = self.camera_states[
                sim_id
            ].to_dict()
            emit("viewer_simulation_update", self.simulations[sim_id], broadcast=True)

            return {"success": True, "sim_id": sim_id}

        @self.socketio.on("client_update_camera_feed")
        def handle_client_update_camera_feed(data):
            """Update a simulation camera feed from a simulation client"""
            sim_id = data.get("sim_id")
            if not sim_id or sim_id not in self.simulations:
                return {"success": False, "error": "Invalid simulation ID"}

            # Update timestamp and heartbeat
            self.simulations[sim_id]["timestamp"] = time.time()
            self.simulations[sim_id]["last_heartbeat"] = time.time()

            # Update camera state if provided
            if "camera_state" in data:
                self.camera_states[sim_id] = CameraState.from_dict(data["camera_state"])
                self.simulations[sim_id]["camera_state"] = self.camera_states[
                    sim_id
                ].to_dict()
                emit(
                    "viewer_simulation_update", self.simulations[sim_id], broadcast=True
                )

            # Process the main camera feed
            main_camera = data.get("main_camera")
            if main_camera:
                # Broadcast the main camera feed to all viewer clients
                emit(
                    "viewer_main_camera_feed",
                    {
                        "sim_id": sim_id,
                        "sim_name": self.simulations[sim_id].get(
                            "sim_name", f"Simulation {sim_id}"
                        ),
                        "image_data": main_camera.get("image_data"),
                        "width": main_camera.get("width"),
                        "height": main_camera.get("height"),
                        "timestamp": time.time(),
                        "simulation_time": data.get("simulation_time", 0.0),
                    },
                    broadcast=True,
                )

            # Process robot camera feeds
            robot_cameras = data.get("robot_cameras", {})
            for robot_name, cameras in robot_cameras.items():
                # Add robot to the simulation's robot list if not already present
                if "robots" not in self.simulations[sim_id]:
                    self.simulations[sim_id]["robots"] = []
                if robot_name not in self.simulations[sim_id]["robots"]:
                    self.simulations[sim_id]["robots"].append(robot_name)
                    # Broadcast updated robot list
                    emit(
                        "viewer_robot_list",
                        {
                            "sim_id": sim_id,
                            "robots": self.simulations[sim_id]["robots"],
                        },
                        broadcast=True,
                    )

                # Send each camera feed
                for camera_name, camera_data in cameras.items():
                    emit(
                        "viewer_robot_camera_feed",
                        {
                            "sim_id": sim_id,
                            "sim_name": self.simulations[sim_id].get(
                                "sim_name", f"Simulation {sim_id}"
                            ),
                            "robot_name": robot_name,
                            "camera_name": camera_name,
                            "image_data": camera_data.get("image_data"),
                            "width": camera_data.get("width"),
                            "height": camera_data.get("height"),
                            "timestamp": time.time(),
                            "simulation_time": data.get("simulation_time", 0.0),
                        },
                        broadcast=True,
                    )

            return {"success": True}

        @self.socketio.on("viewer_camera_control")
        def handle_viewer_camera_control(data):
            """Handle camera control inputs from viewer clients"""
            sim_id = data.get("sim_id")
            if not sim_id or sim_id not in self.simulations:
                return {"success": False, "error": "Invalid simulation ID"}

            action = data.get("action")
            camera_state = self.camera_states[sim_id]

            if action == "movement":
                # Store previous value for logging
                prev_value = camera_state.movement_direction.copy()
                camera_state.movement_direction = data.get("movement", [0.0, 0.0, 0.0])

                # Only disable bird's eye view mode if there's actual movement
                if any(x != 0 for x in camera_state.movement_direction):
                    camera_state.bird_view_mode = False

                self.logger.debug(
                    f"Movement direction changed from {prev_value} to {camera_state.movement_direction}"
                )

            elif action == "roll":
                camera_state.roll_input = data.get("roll", 0.0)

                # Only disable bird's eye view mode if there's actual roll input
                if camera_state.roll_input != 0:
                    camera_state.bird_view_mode = False

                self.logger.debug(f"Roll input set to {camera_state.roll_input}")

            elif action == "rotate":
                camera_state.pending_mouse_pitch = data.get("pitch", 0.0)
                camera_state.pending_mouse_yaw = data.get("yaw", 0.0)

                # Only disable bird's eye view mode if there's actual rotation
                if (
                    camera_state.pending_mouse_pitch != 0
                    or camera_state.pending_mouse_yaw != 0
                ):
                    camera_state.bird_view_mode = False

                self.logger.debug(
                    f"Rotation set to pitch={camera_state.pending_mouse_pitch}, yaw={camera_state.pending_mouse_yaw}"
                )

            # Update the simulation with the new camera state
            self.simulations[sim_id]["camera_state"] = camera_state.to_dict()

            # Send the updated state to the specific simulation client
            timestamp = time.time()
            emit(
                "client_camera_state_update",
                {
                    "sim_id": sim_id,
                    "camera_state": camera_state.to_dict(),
                    "timestamp": timestamp,
                },
                broadcast=True,  # Send to all clients, including the simulation client
            )

            return {"success": True}

        @self.socketio.on("viewer_set_tracking")
        def handle_viewer_set_tracking(data):
            """Handle robot tracking requests from viewer clients"""
            sim_id = data.get("sim_id")
            if not sim_id or sim_id not in self.simulations:
                return {"success": False, "error": "Invalid simulation ID"}

            tracking_robot = data.get("tracking_robot")
            camera_state = self.camera_states[sim_id]
            camera_state.bird_view_mode = False

            # Update tracking robot
            camera_state.tracking_robot = tracking_robot
            self.logger.info(f"Tracking robot set to: {tracking_robot}")

            # If tracking is enabled, reset movement and rotation inputs
            if tracking_robot:
                camera_state.movement_direction = [0.0, 0.0, 0.0]
                camera_state.roll_input = 0.0
                camera_state.pending_mouse_pitch = 0.0
                camera_state.pending_mouse_yaw = 0.0

            # Update the simulation with the new camera state
            self.simulations[sim_id]["camera_state"] = camera_state.to_dict()

            # Broadcast the updated simulation state to viewer clients
            emit("viewer_simulation_update", self.simulations[sim_id], broadcast=True)

            # Also send specific camera state update to ensure simulation client gets it
            timestamp = time.time()
            emit(
                "client_camera_state_update",
                {
                    "sim_id": sim_id,
                    "camera_state": camera_state.to_dict(),
                    "timestamp": timestamp,
                },
                broadcast=True,
            )

            return {"success": True}

        @self.socketio.on("viewer_set_bird_view")
        def handle_viewer_set_bird_view(data):
            """Handle bird's eye view requests from viewer clients"""
            sim_id = data.get("sim_id")
            if not sim_id or sim_id not in self.simulations:
                return {"success": False, "error": "Invalid simulation ID"}

            robot_name = data.get("robot_name")
            if not robot_name:
                return {"success": False, "error": "No robot specified"}

            camera_state = self.camera_states[sim_id]

            # Calculate bird's eye view position based on robot index
            # Robot names have format: "bot_e_{epoch}_ro_{robot_id}_idx{i}"
            robot_index = 0
            if "_idx" in robot_name:
                try:
                    robot_index = int(robot_name.split("_idx")[-1])
                except (ValueError, IndexError):
                    robot_index = 0

            # Calculate Y position based on robot spacing (10 units apart)
            # Reference position from env.py: [1.0, 0.0, 10.0]
            bird_view_position = [1.0, 10 * robot_index, 10.0]

            # Bird's eye view orientation (looking down)
            # Quaternion for looking straight down: [0, 0.7071, 0, 0.7071]
            bird_view_orientation = [0, 0.7071, 0, 0.7071]

            # Update camera state
            camera_state.position = bird_view_position
            camera_state.orientation = bird_view_orientation
            camera_state.tracking_robot = None  # Disable tracking
            camera_state.movement_direction = [0.0, 0.0, 0.0]
            camera_state.roll_input = 0.0
            camera_state.pending_mouse_pitch = 0.0
            camera_state.pending_mouse_yaw = 0.0
            camera_state.bird_view_mode = True
            self.logger.info(
                f"Set bird's eye view for robot {robot_name} at position {bird_view_position}"
            )

            # Update the simulation with the new camera state
            self.simulations[sim_id]["camera_state"] = camera_state.to_dict()

            # Broadcast the updated simulation state to viewer clients
            emit("viewer_simulation_update", self.simulations[sim_id], broadcast=True)

            # Also send specific camera state update to ensure simulation client gets it
            timestamp = time.time()
            emit(
                "client_camera_state_update",
                {
                    "sim_id": sim_id,
                    "camera_state": camera_state.to_dict(),
                    "timestamp": timestamp,
                },
                broadcast=True,
            )

            return {"success": True}

        @self.socketio.on("client_update_title")
        def handle_client_update_title(data):
            """Handle title update requests"""
            new_title = data.get("title")
            if new_title:
                self.current_title = new_title
                self.logger.info(f"Updating title to {new_title}")
                emit("title_update", {"title": new_title}, broadcast=True)
            return {"success": True}

    def remove_simulation(self, sim_id):
        """Remove a simulation and notify web clients"""
        if sim_id in self.simulations:
            self.logger.info(f"Removing simulation: {sim_id}")
            del self.simulations[sim_id]
            if sim_id in self.camera_states:
                del self.camera_states[sim_id]

            # Notify web clients to remove this simulation from UI
            self.socketio.emit("viewer_simulation_removed", {"sim_id": sim_id})

    def check_stale_simulations(self):
        """Check for and remove simulations that haven't sent heartbeats recently"""
        current_time = time.time()

        for sim_id, sim_info in list(self.simulations.items()):
            last_heartbeat = sim_info.get(
                "last_heartbeat", sim_info.get("timestamp", 0)
            )
            if current_time - last_heartbeat > self.heartbeat_timeout:
                self.logger.info(
                    f"Simulation {sim_id} timed out (no heartbeat for >{self.heartbeat_timeout}s)"
                )
                self.remove_simulation(sim_id)

    def start(self):
        """Start the web server in a separate thread"""

        def run_server():
            self.logger.info(f"Starting web server on {self.host}:{self.port}")
            try:
                self.socketio.run(
                    self.app,
                    host=self.host,
                    port=self.port,
                    debug=False,
                    allow_unsafe_werkzeug=True,
                )
            except Exception as e:
                self.logger.error(f"Error running web server: {e}")
                traceback.print_exc()

        # Start the server in a daemon thread
        self.server_thread = threading.Thread(target=run_server)
        self.server_thread.daemon = True
        self.server_thread.start()
        print(f"Web server started at http://{self.host}:{self.port}")

        # Start periodic check for stale simulations
        def schedule_check():
            self.check_stale_simulations()
            self.stale_check_timer = threading.Timer(5.0, schedule_check)
            self.stale_check_timer.daemon = True
            self.stale_check_timer.start()

        # Start the first check after 5 seconds
        self.stale_check_timer = threading.Timer(5.0, schedule_check)
        self.stale_check_timer.daemon = True
        self.stale_check_timer.start()

    def stop(self):
        """Stop the web server"""
        # Cancel the stale check timer if it's running
        if self.stale_check_timer:
            self.stale_check_timer.cancel()

        # This will trigger a KeyboardInterrupt in the Flask thread
        if self.server_thread and self.server_thread.is_alive():
            os.kill(os.getpid(), signal.SIGINT)
            self.server_thread.join(timeout=2)
            print("Web server stopped")


class CameraWebClient:
    """Client for connecting simulations to the web server with async capabilities"""

    def __init__(
        self,
        server_url="XXXX",
        sim_id=None,
        sim_name=None,
        max_queue_size=10,
        worker_threads=2,
    ):
        self.server_url = server_url
        self.sim_id = sim_id or str(uuid.uuid4())
        self.sim_name = sim_name or f"Simulation {self.sim_id}"
        self.socketio = None
        self.connected = False
        self.camera_state = CameraState(
            position=[0.0, 0.0, 1.0],
            orientation=[0.0, 0.0, 0.0, 1.0],
            movement_direction=[0.0, 0.0, 0.0],
            roll_input=0.0,
        )
        self._last_state_update = 0  # Timestamp of last state update from server

        # Async support
        self.update_queue = Queue(maxsize=max_queue_size)
        self.worker_thread = None
        self.should_stop = False
        self.executor = ThreadPoolExecutor(max_workers=worker_threads)

        # Heartbeat support
        self.heartbeat_thread = None
        self.heartbeat_interval = 3.0  # seconds between heartbeats

    def start_heartbeat(self):
        """Start sending regular heartbeat signals to the server"""

        def send_heartbeat():
            while self.connected:
                try:
                    if self.socketio and self.connected:
                        self.socketio.emit("client_heartbeat", {"sim_id": self.sim_id})
                    time.sleep(self.heartbeat_interval)
                except Exception as e:
                    print(f"Error sending heartbeat: {e}")
                    self.connected = False
                    break

        # Start the heartbeat in a daemon thread
        self.heartbeat_thread = threading.Thread(target=send_heartbeat)
        self.heartbeat_thread.daemon = True
        self.heartbeat_thread.start()

    def connect(self):
        """Connect to the web server"""
        from socketio import Client

        try:
            self.socketio = Client()
            self.socketio.connect(self.server_url)
            self.connected = True

            # Register with the server
            camera_state_dict = convert_numpy_values(self.camera_state.to_dict())

            response = self.socketio.call(
                "client_register",
                {
                    "sim_id": self.sim_id,
                    "sim_name": self.sim_name,
                    "camera_state": camera_state_dict,
                },
            )

            if response.get("success", False):
                print(f"Connected to server as simulation client {self.sim_id}")

                # Start sending heartbeat signals
                self.start_heartbeat()

                # Start the async worker
                self._start_worker()

                # Set up event handlers
                @self.socketio.on("disconnect")
                def on_disconnect():
                    self.connected = False
                    print("Disconnected from server")

                # Add handler for camera control updates from the server
                @self.socketio.on("client_camera_state_update")
                def on_client_camera_state_update(data):
                    if data.get("sim_id") == self.sim_id:
                        try:
                            # Update local camera state from server
                            new_state = data.get("camera_state", {})
                            # Only apply the update if it's newer than our last update
                            timestamp = data.get("timestamp", 0)
                            if timestamp > self._last_state_update:
                                self._last_state_update = timestamp
                                self.camera_state = CameraState.from_dict(new_state)
                                # print(
                                #     f"Camera state updated from server: {self.camera_state.movement_direction}"
                                # )
                        except Exception as e:
                            print(f"Error updating camera state: {e}")
                            traceback.print_exc()

                return True
            else:
                print(
                    f"Failed to register with server: {response.get('error', 'Unknown error')}"
                )
                self.socketio.disconnect()
                self.connected = False
                return False

        except Exception as e:
            print(f"Error connecting to server: {e}")
            traceback.print_exc()
            self.connected = False
            return False

    def update_title(self, title):
        if self.connected:
            # print(f"Updating title to {title}")
            self.socketio.emit("client_update_title", {"title": title})

    def update_camera_feed(
        self,
        simulation_time,
        image,
        robot_cameras=None,
        camera_state=None,
        blocking=False,
    ):
        """
        Update the camera feed with a new image and optional robot camera feeds

        Args:
            simulation_time: Current simulation time in seconds
            image: Main camera image data (numpy array or PIL image)
            robot_cameras: Optional dictionary mapping robot names to camera images
                           Format: {robot_name: {camera_name: image_data}}
            camera_state: Optional updated camera state
            blocking: If True, process immediately and wait for result.
                     If False (default), queue the update and return immediately.

        Returns:
            bool: Success status (or queue status if non-blocking)
        """
        if not self.connected:
            return False

        # If blocking, process update immediately
        if blocking:
            return self._process_camera_update(
                simulation_time, image, robot_cameras, camera_state
            )

        # Otherwise, use the async queue
        try:
            # If queue is full, remove oldest item
            if self.update_queue.full():
                try:
                    # Non-blocking get
                    self.update_queue.get_nowait()
                    self.update_queue.task_done()
                    print("Warning: Camera update queue full, dropping oldest frame")
                except:
                    pass

            # Add to queue
            self.update_queue.put(
                (simulation_time, image, robot_cameras, camera_state), block=False
            )
            return True

        except Exception as e:
            print(f"Error queueing camera update: {e}")
            return False

    def update_robot_list(self, robots):
        """Update the list of robots available for tracking"""
        if not self.connected:
            return False

        try:
            # Ensure robot list is JSON serializable (in case it contains numpy values)
            robot_list = convert_numpy_values(robots)

            response = self.socketio.call(
                "client_register",
                {
                    "sim_id": self.sim_id,
                    "sim_name": self.sim_name,
                    "robots": robot_list,
                },
            )

            return response.get("success", False)

        except Exception as e:
            print(f"Error updating robot list: {e}")
            return False

    def get_camera_state(self):
        """Get the current camera state from the server"""
        # This returns the potentially updated state from server
        return self.camera_state

    def get_queue_size(self):
        """Get the current number of pending updates in the queue"""
        return self.update_queue.qsize()

    def disconnect(self):
        """Disconnect from the web server"""
        # Signal worker to stop
        self.should_stop = True

        # Wait for queue to empty with timeout
        try:
            self.update_queue.join(timeout=2.0)
        except:
            pass

        # Shutdown executor
        self.executor.shutdown(wait=False)

        if self.socketio and self.connected:
            try:
                # Notify server about disconnection
                self.socketio.emit("client_disconnect", {"sim_id": self.sim_id})
                # Wait a brief moment for the event to be sent
                time.sleep(0.2)
                # Then disconnect
                self.socketio.disconnect()
            except Exception as e:
                print(f"Error during disconnect: {e}")
            finally:
                self.connected = False
                print("Disconnected from server")

    def _process_camera_update(
        self, simulation_time, image, robot_cameras=None, camera_state=None
    ):
        """Internal method to process and send camera update to server"""
        if not self.connected:
            return False

        try:
            # Process main camera image
            main_camera_data = self._prepare_image_data(image)

            # Process robot camera images if provided
            robot_camera_data = {}
            if robot_cameras:
                for robot_name, cameras in robot_cameras.items():
                    robot_camera_data[robot_name] = {}
                    for camera_name, camera_image in cameras.items():
                        robot_camera_data[robot_name][camera_name] = (
                            self._prepare_image_data(camera_image)
                        )

            # Update camera state if provided
            if camera_state:
                self.camera_state = camera_state

            # Create a JSON-serializable camera state
            camera_state_dict = convert_numpy_values(self.camera_state.to_dict())

            # Send to server
            response = self.socketio.call(
                "client_update_camera_feed",
                {
                    "sim_id": self.sim_id,
                    "main_camera": main_camera_data,
                    "robot_cameras": robot_camera_data,
                    "camera_state": camera_state_dict,
                    "simulation_time": simulation_time,
                },
            )

            return response.get("success", False)

        except Exception as e:
            print(f"Error updating camera feed: {e}")
            traceback.print_exc()
            return False

    @staticmethod
    def _prepare_image_data(image):
        """Convert an image to base64-encoded JPEG format with dimensions"""
        if isinstance(image, np.ndarray):
            # Convert numpy array to PIL Image
            pil_img = Image.fromarray(image)
            buffer = BytesIO()
            pil_img.save(buffer, format="JPEG", quality=80)
            img_data = base64.b64encode(buffer.getvalue()).decode("utf-8")
            width, height = pil_img.size
        else:
            # Assume it's already a PIL Image
            buffer = BytesIO()
            image.save(buffer, format="JPEG", quality=80)
            img_data = base64.b64encode(buffer.getvalue()).decode("utf-8")
            width, height = image.size

        return {"image_data": img_data, "width": width, "height": height}

    def _start_worker(self):
        """Start the background worker thread to process the update queue"""

        def process_queue():
            while not self.should_stop:
                try:
                    # Get item from queue with timeout to allow checking should_stop periodically
                    try:
                        item = self.update_queue.get(timeout=1.0)
                    except:
                        # Queue empty or timeout, continue loop
                        continue

                    # Process the update
                    try:
                        # Unpack the item
                        simulation_time, image, robot_cameras, camera_state = item

                        # Process the update directly
                        self._process_camera_update(
                            simulation_time, image, robot_cameras, camera_state
                        )
                    except Exception as e:
                        print(f"Error processing camera update: {e}")
                        traceback.print_exc()
                    finally:
                        # Mark task as done regardless of success/failure
                        self.update_queue.task_done()

                except Exception as e:
                    print(f"Error in worker thread: {e}")
                    traceback.print_exc()

        # Start the background worker
        self.worker_thread = threading.Thread(target=process_queue)
        self.worker_thread.daemon = True
        self.worker_thread.start()
