# visualizer_runner.py

import asyncio
import websockets
import threading
import http.server
import socketserver
import queue
import json
from logger import VSSMLogger

# --- Configuration ---
HTTP_PORT = 8000
WEBSOCKET_PORT = 8765
UI_DIRECTORY = "ui"


class VisualizationRunner:
    """
    Manages the HTTP and WebSocket servers for real-time UI updates.
    This entire class is designed to be run in a background thread.
    """

    def __init__(self, update_queue: queue.Queue):
        self.update_queue = update_queue
        self.http_thread = None
        self.websocket_thread = None
        self.loop = None
        self.clients = set()
        self.logger = VSSMLogger()

    def start_servers(self):
        """Starts both the HTTP and WebSocket servers in separate threads."""
        self.http_thread = threading.Thread(target=self._run_http_server, daemon=True)
        self.http_thread.start()
        self.logger.log(f"HTTP server thread started. UI available at http://localhost:{HTTP_PORT}", "SUCCESS")

        self.websocket_thread = threading.Thread(target=self._run_websocket_server, daemon=True)
        self.websocket_thread.start()
        self.logger.log(f"WebSocket server thread started at ws://localhost:{WEBSOCKET_PORT}", "SUCCESS")

    def _run_http_server(self):
        """Serves the static files for the UI."""

        class Handler(http.server.SimpleHTTPRequestHandler):
            def __init__(self, *args, **kwargs):
                super().__init__(*args, directory=UI_DIRECTORY, **kwargs)

        with socketserver.TCPServer(("", HTTP_PORT), Handler) as httpd:
            self.logger.log(f"Serving UI from '{UI_DIRECTORY}' directory.", "INFO")
            httpd.serve_forever()

    # --- THIS IS THE CORRECTED METHOD ---
    def _run_websocket_server(self):
        """Runs the main asyncio event loop for the WebSocket server."""
        # Create and set the event loop for this thread
        self.loop = asyncio.new_event_loop()
        asyncio.set_event_loop(self.loop)

        # Define an async function to start the server. This is where the
        # `websockets.serve` call needs to be.
        async def start_server_async():
            # The `websockets.serve` function is now called inside an async context,
            # so it can find the running loop.
            return await websockets.serve(
                self._connection_handler, "localhost", WEBSOCKET_PORT
            )

        # Run the async function to start the server and get the server object.
        server = self.loop.run_until_complete(start_server_async())
        self.logger.log("WebSocket server is running inside the event loop.", "DEBUG")

        # Run the event loop forever. It will now manage the server and connections.
        try:
            self.loop.run_forever()
        finally:
            # Clean up when the loop is stopped.
            server.close()
            self.loop.run_until_complete(server.wait_closed())
            self.loop.close()
            self.logger.log("WebSocket event loop has been closed.", "INFO")

    async def _connection_handler(self, websocket, path):
        """Handles a new client connection and starts the update sender."""
        self.logger.log(f"UI client connected from {websocket.remote_address}", "SUCCESS")
        self.clients.add(websocket)
        try:
            await websocket.wait_closed()
        finally:
            self.logger.log(f"UI client disconnected from {websocket.remote_address}", "FAIL")
            self.clients.remove(websocket)

    def broadcast_updates(self):
        """
        Continuously checks the queue for updates and broadcasts them
        to all connected WebSocket clients. This should be run by the VSSM class.
        """
        while True:
            try:
                update = self.update_queue.get()
                if update == 'SHUTDOWN':
                    self.logger.log("Shutdown signal received. Stopping WebSocket event loop.", "INFO")
                    if self.loop and self.loop.is_running():
                        # This is the thread-safe way to stop a loop from another thread.
                        self.loop.call_soon_threadsafe(self.loop.stop)
                    break

                if self.clients and self.loop and self.loop.is_running():
                    asyncio.run_coroutine_threadsafe(self._broadcast(update), self.loop)

            except Exception as e:
                self.logger.log(f"Error in broadcast loop: {e}", "FAIL")
                break

    async def _broadcast(self, message):
        """Asynchronously sends a message to all connected clients."""
        if self.clients:
            tasks = [client.send(message) for client in self.clients]
            await asyncio.gather(*tasks, return_exceptions=True)