luckyrobots 0.1.68__py3-none-any.whl → 0.1.69__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (57) hide show
  1. luckyrobots/__init__.py +23 -12
  2. luckyrobots/client.py +800 -0
  3. luckyrobots/config/robots.yaml +231 -71
  4. luckyrobots/engine/__init__.py +23 -0
  5. luckyrobots/{utils → engine}/check_updates.py +108 -48
  6. luckyrobots/{utils → engine}/download.py +61 -39
  7. luckyrobots/engine/manager.py +427 -0
  8. luckyrobots/grpc/__init__.py +6 -0
  9. luckyrobots/grpc/generated/__init__.py +18 -0
  10. luckyrobots/grpc/generated/agent_pb2.py +61 -0
  11. luckyrobots/grpc/generated/agent_pb2_grpc.py +255 -0
  12. luckyrobots/grpc/generated/camera_pb2.py +45 -0
  13. luckyrobots/grpc/generated/camera_pb2_grpc.py +155 -0
  14. luckyrobots/grpc/generated/common_pb2.py +39 -0
  15. luckyrobots/grpc/generated/common_pb2_grpc.py +27 -0
  16. luckyrobots/grpc/generated/media_pb2.py +35 -0
  17. luckyrobots/grpc/generated/media_pb2_grpc.py +27 -0
  18. luckyrobots/grpc/generated/mujoco_pb2.py +47 -0
  19. luckyrobots/grpc/generated/mujoco_pb2_grpc.py +248 -0
  20. luckyrobots/grpc/generated/scene_pb2.py +54 -0
  21. luckyrobots/grpc/generated/scene_pb2_grpc.py +248 -0
  22. luckyrobots/grpc/generated/telemetry_pb2.py +43 -0
  23. luckyrobots/grpc/generated/telemetry_pb2_grpc.py +154 -0
  24. luckyrobots/grpc/generated/viewport_pb2.py +48 -0
  25. luckyrobots/grpc/generated/viewport_pb2_grpc.py +155 -0
  26. luckyrobots/grpc/proto/agent.proto +152 -0
  27. luckyrobots/grpc/proto/camera.proto +41 -0
  28. luckyrobots/grpc/proto/common.proto +36 -0
  29. luckyrobots/grpc/proto/hazel_rpc.proto +32 -0
  30. luckyrobots/grpc/proto/media.proto +26 -0
  31. luckyrobots/grpc/proto/mujoco.proto +64 -0
  32. luckyrobots/grpc/proto/scene.proto +70 -0
  33. luckyrobots/grpc/proto/telemetry.proto +43 -0
  34. luckyrobots/grpc/proto/viewport.proto +45 -0
  35. luckyrobots/luckyrobots.py +193 -0
  36. luckyrobots/models/__init__.py +13 -0
  37. luckyrobots/models/camera.py +97 -0
  38. luckyrobots/models/observation.py +135 -0
  39. luckyrobots/{utils/helpers.py → utils.py} +75 -40
  40. luckyrobots-0.1.69.dist-info/METADATA +262 -0
  41. luckyrobots-0.1.69.dist-info/RECORD +44 -0
  42. {luckyrobots-0.1.68.dist-info → luckyrobots-0.1.69.dist-info}/WHEEL +1 -1
  43. luckyrobots/core/luckyrobots.py +0 -628
  44. luckyrobots/core/manager.py +0 -236
  45. luckyrobots/core/models.py +0 -68
  46. luckyrobots/core/node.py +0 -273
  47. luckyrobots/message/__init__.py +0 -18
  48. luckyrobots/message/pubsub.py +0 -145
  49. luckyrobots/message/srv/client.py +0 -81
  50. luckyrobots/message/srv/service.py +0 -135
  51. luckyrobots/message/srv/types.py +0 -83
  52. luckyrobots/message/transporter.py +0 -427
  53. luckyrobots/utils/event_loop.py +0 -94
  54. luckyrobots/utils/sim_manager.py +0 -413
  55. luckyrobots-0.1.68.dist-info/METADATA +0 -253
  56. luckyrobots-0.1.68.dist-info/RECORD +0 -24
  57. {luckyrobots-0.1.68.dist-info → luckyrobots-0.1.69.dist-info}/licenses/LICENSE +0 -0
@@ -1,628 +0,0 @@
1
- import cv2
2
- import json
3
- import msgpack
4
- import asyncio
5
- import logging
6
- import secrets
7
- import platform
8
- import signal
9
- import threading
10
- import time
11
-
12
- from typing import Dict
13
-
14
- import uvicorn
15
- from fastapi import FastAPI, WebSocket, WebSocketDisconnect
16
- from websocket import create_connection
17
-
18
- from .manager import Manager
19
- from ..message.transporter import MessageType, TransportMessage
20
- from ..message.srv.types import Reset, Step
21
- from ..utils.sim_manager import launch_luckyworld, stop_luckyworld
22
- from ..core.models import ObservationModel
23
- from .node import Node
24
- from ..utils.event_loop import (
25
- get_event_loop,
26
- initialize_event_loop,
27
- )
28
- from ..utils.helpers import (
29
- validate_params,
30
- get_robot_config,
31
- )
32
-
33
- logging.basicConfig(
34
- level=logging.INFO, format="%(asctime)s - %(name)s - %(levelname)s - %(message)s"
35
- )
36
- logger = logging.getLogger("luckyrobots")
37
-
38
- app = FastAPI()
39
- manager = Manager()
40
-
41
-
42
- class LuckyRobots(Node):
43
- """Main LuckyRobots node for managing robot communication and control"""
44
-
45
- def __init__(self, host: str = "localhost", port: int = 3000) -> None:
46
- self.host = host
47
- self.port = port
48
- self.robot_client = None
49
- self.world_client = None
50
- self._pending_resets = {}
51
- self._pending_steps = {}
52
- self._running = False
53
- self._nodes: Dict[str, Node] = {}
54
- self._shutdown_event = threading.Event()
55
-
56
- initialize_event_loop()
57
-
58
- self._start_websocket_server()
59
-
60
- super().__init__("lucky_robots_manager", "", self.host, self.port)
61
- app.lucky_robots = self
62
-
63
- def _is_websocket_server_running(self) -> bool:
64
- """Check if the websocket server is already running"""
65
- try:
66
- ws_url = f"ws://{self.host}:{self.port}/nodes"
67
- ws = create_connection(ws_url, timeout=1)
68
- ws.close()
69
- return True
70
- except Exception as e:
71
- return False
72
-
73
- def _start_websocket_server(self) -> None:
74
- """Start the websocket server in a separate thread using uvicorn"""
75
-
76
- if self._is_websocket_server_running():
77
- logger.warning(
78
- f"WebSocket server already running on {self.host}:{self.port}"
79
- )
80
- return
81
-
82
- def run_server():
83
- logging.getLogger("uvicorn.access").setLevel(logging.WARNING)
84
- config = uvicorn.Config(
85
- app, host=self.host, port=self.port, log_level="warning"
86
- )
87
- self._server = uvicorn.Server(config)
88
- self._server.run()
89
-
90
- self._server_thread = threading.Thread(target=run_server, daemon=True)
91
- self._server_thread.start()
92
-
93
- logger.info(f"Starting WebSocket server on {self.host}:{self.port}")
94
-
95
- # Wait for the server to start
96
- timeout = 10.0
97
- start_time = time.perf_counter()
98
-
99
- while time.perf_counter() - start_time < timeout:
100
- if self._is_websocket_server_running():
101
- logger.info(f"WebSocket server ready on {self.host}:{self.port}")
102
- return
103
- time.sleep(0.1)
104
-
105
- logger.error(f"WebSocket server failed to start within {timeout} seconds")
106
- raise RuntimeError(
107
- f"WebSocket server failed to start on {self.host}:{self.port}"
108
- )
109
-
110
- @staticmethod
111
- def get_robot_config(robot: str = None) -> dict:
112
- """Get the configuration for the LuckyRobots node"""
113
- return get_robot_config(robot)
114
-
115
- def register_node(self, node: Node) -> None:
116
- """Register a node with the LuckyRobots node"""
117
- self._nodes[node.full_name] = node
118
- logger.info(f"Registered node: {node.full_name}")
119
-
120
- async def _setup_async(self):
121
- """Setup the LuckyRobots node asynchronously"""
122
- self.reset_service = await self.create_service(
123
- Reset, "/reset", self.handle_reset
124
- )
125
- self.step_service = await self.create_service(Step, "/step", self.handle_step)
126
-
127
- def _register_cleanup_handlers(self) -> None:
128
- """Register cleanup handlers for the LuckyRobots node to handle Ctrl+C"""
129
-
130
- def sigint_handler(signum, frame):
131
- logger.info("Ctrl+C pressed. Shutting down...")
132
- self.shutdown()
133
-
134
- signal.signal(signal.SIGINT, sigint_handler)
135
-
136
- def start(
137
- self,
138
- scene: str,
139
- robot: str,
140
- debug: bool = False,
141
- task: str = None,
142
- executable_path: str = None,
143
- observation_type: str = "pixels_agent_pos",
144
- headless: bool = False,
145
- ) -> None:
146
- """Start the LuckyRobots node"""
147
- if self._running:
148
- logger.warning("LuckyRobots is already running")
149
- return
150
-
151
- validate_params(scene, robot, task, observation_type)
152
- self.process_cameras = "pixels" in observation_type
153
-
154
- success = launch_luckyworld(
155
- scene=scene,
156
- robot=robot,
157
- debug=debug,
158
- task=task,
159
- executable_path=executable_path,
160
- headless=headless,
161
- )
162
- if not success:
163
- logger.error("Failed to launch LuckyWorld")
164
- self.shutdown()
165
- raise RuntimeError(
166
- "Failed to launch LuckyWorld. Look through the log for more information."
167
- )
168
-
169
- self._register_cleanup_handlers()
170
-
171
- # Start all registered nodes
172
- failed_nodes = []
173
- for node in self._nodes.values():
174
- try:
175
- node.start()
176
- logger.info(f"Started node: {node.full_name}")
177
- except Exception as e:
178
- logger.error(f"Error starting node {node.full_name}: {e}")
179
- failed_nodes.append(node.full_name)
180
-
181
- if failed_nodes:
182
- logger.error(f"Failed to start nodes: {', '.join(failed_nodes)}")
183
- # Continue anyway - some nodes might be optional
184
-
185
- super().start()
186
-
187
- self._running = True
188
-
189
- def _display_welcome_message(self) -> None:
190
- """Display the welcome message for the LuckyRobots node in the terminal"""
191
-
192
- # Create the complete message in one go
193
- stars = "*" * 60
194
-
195
- welcome_lines = [
196
- stars,
197
- " ",
198
- " ",
199
- "▄▄▌ ▄• ▄▌ ▄▄· ▄ •▄ ▄· ▄▌▄▄▄ ▄▄▄▄· ▄▄▄▄▄.▄▄ · ",
200
- "██• █▪██▌▐█ ▌▪█▌▄▌▪▐█▪██▌▀▄ █·▪ ▐█ ▀█▪▪ •██ ▐█ ▀. ",
201
- "██▪ █▌▐█▌██ ▄▄▐▀▀▄·▐█▌▐█▪▐▀▀▄ ▄█▀▄ ▐█▀▀█▄ ▄█▀▄ ▐█.▪▄▀▀▀█▄",
202
- "▐█▌▐▌▐█▄█▌▐███▌▐█.█▌ ▐█▀·.▐█•█▌▐█▌.▐▌██▄▪▐█▐█▌.▐▌ ▐█▌·▐█▄▪▐█",
203
- ".▀▀▀ ▀▀▀ ·▀▀▀ ·▀ ▀ ▀ • .▀ ▀ ▀█▄▀▪·▀▀▀▀ ▀█▄▀▪ ▀▀▀ ▀▀▀▀ ",
204
- " ",
205
- " ",
206
- ]
207
-
208
- # Add macOS instructions if needed
209
- if platform.system() == "Darwin":
210
- welcome_lines.extend(
211
- [
212
- stars,
213
- "For macOS users:",
214
- "Please be patient. The application may take up to a minute to open on its first launch.",
215
- "If the application doesn't appear, please follow these steps:",
216
- "1. Open System Settings",
217
- "2. Navigate to Privacy & Security",
218
- "3. Scroll down and click 'Allow' next to the 'luckyrobots' app",
219
- stars,
220
- ]
221
- )
222
-
223
- # Add final messages
224
- welcome_lines.extend(
225
- [
226
- "Lucky Robots application started successfully.",
227
- "To move the robot: Choose a level and tick the HTTP checkbox.",
228
- "To receive camera feed: Choose a level and tick the Capture checkbox.",
229
- stars,
230
- "", # Empty line at the end
231
- ]
232
- )
233
-
234
- # Single print statement - cannot be interrupted!
235
- print("\n".join(welcome_lines), flush=True)
236
-
237
- def wait_for_world_client(self, timeout: float = 120.0) -> bool:
238
- """Wait for the world client to connect to the websocket server"""
239
- start_time = time.perf_counter()
240
-
241
- logger.info(f"Waiting for world client to connect (timeout: {timeout}s)")
242
- while not self.world_client and time.perf_counter() - start_time < timeout:
243
- time.sleep(0.5)
244
-
245
- if self.world_client:
246
- logger.info("World client connected successfully")
247
- return True
248
- else:
249
- logger.error(f"No world client connected after {timeout} seconds")
250
- self.shutdown()
251
- raise RuntimeError(
252
- f"World client connection timeout after {timeout} seconds"
253
- )
254
-
255
- async def handle_reset(self, request: Reset.Request) -> Reset.Response:
256
- """Handle the reset request by forwarding to the world client"""
257
- if self.world_client is None:
258
- logger.error("No world client connection available")
259
- self.shutdown()
260
- raise
261
-
262
- request_id = secrets.token_hex(4)
263
- shared_loop = get_event_loop()
264
- response_future = shared_loop.create_future()
265
- self._pending_resets[request_id] = response_future
266
-
267
- seed = getattr(request, "seed", None)
268
- options = getattr(request, "options", None)
269
- request_data = {
270
- "request_type": "reset",
271
- "request_id": request_id,
272
- "seed": seed,
273
- "options": options,
274
- }
275
-
276
- try:
277
- await self.world_client.send_text(json.dumps(request_data))
278
- response_data = await asyncio.wait_for(response_future, timeout=30.0)
279
-
280
- # Wait for reset animation to finish in Lucky World
281
- time.sleep(1)
282
-
283
- observation = ObservationModel(**response_data["Observation"])
284
- if self.process_cameras:
285
- observation.process_all_cameras()
286
-
287
- return Reset.Response(
288
- success=True,
289
- message="Reset request processed",
290
- request_type=response_data["RequestType"],
291
- request_id=response_data["RequestID"],
292
- time_stamp=response_data["TimeStamp"],
293
- observation=observation,
294
- info=response_data["Info"],
295
- )
296
- except Exception as e:
297
- self._pending_resets.pop(request_id, None)
298
- logger.error(f"Error processing reset request: {e}")
299
- self.shutdown()
300
- raise
301
-
302
- async def handle_step(self, request: Step.Request) -> Step.Response:
303
- """Handle the step request by forwarding to the world client"""
304
- if self.world_client is None:
305
- logger.error("No world client connection available")
306
- self.shutdown()
307
- raise
308
-
309
- request_id = secrets.token_hex(4)
310
- shared_loop = get_event_loop()
311
- response_future = shared_loop.create_future()
312
- self._pending_steps[request_id] = response_future
313
-
314
- self._pending_steps[request_id] = response_future
315
-
316
- request_data = {
317
- "request_type": "step",
318
- "request_id": request_id,
319
- "actuator_values": request.actuator_values,
320
- }
321
-
322
- try:
323
- await self.world_client.send_text(json.dumps(request_data))
324
- response_data = await asyncio.wait_for(response_future, timeout=30.0)
325
-
326
- observation = ObservationModel(**response_data["Observation"])
327
- if self.process_cameras:
328
- observation.process_all_cameras()
329
-
330
- return Step.Response(
331
- success=True,
332
- message="Step request processed",
333
- request_type=response_data["RequestType"],
334
- request_id=response_data["RequestID"],
335
- time_stamp=response_data["TimeStamp"],
336
- observation=observation,
337
- info=response_data["Info"],
338
- )
339
- except Exception as e:
340
- self._pending_steps.pop(request_id, None)
341
- logger.error(f"Error processing step request: {e}")
342
- self.shutdown()
343
- raise
344
-
345
- def spin(self) -> None:
346
- """Spin the LuckyRobots node to keep it running"""
347
- if not self._running:
348
- logger.warning("LuckyRobots is not running")
349
- return
350
-
351
- self._display_welcome_message()
352
- logger.info("LuckyRobots spinning")
353
-
354
- try:
355
- self._shutdown_event.wait()
356
- except KeyboardInterrupt:
357
- logger.info("Keyboard interrupt received. Shutting down...")
358
- self.shutdown()
359
-
360
- logger.info("LuckyRobots stopped spinning")
361
-
362
- def _stop_websocket_server(self) -> None:
363
- """Stop the WebSocket server if it's running"""
364
- if hasattr(self, "_server") and self._server is not None:
365
- try:
366
- # Stop the uvicorn server
367
- self._server.should_exit = True
368
-
369
- # Wait for the server thread to terminate
370
- if (
371
- hasattr(self, "_server_thread")
372
- and self._server_thread
373
- and self._server_thread.is_alive()
374
- ):
375
- self._server_thread.join(timeout=5.0)
376
-
377
- if self._server_thread.is_alive():
378
- logger.warning(
379
- "WebSocket server thread did not terminate within timeout"
380
- )
381
- else:
382
- logger.info("WebSocket server thread terminated successfully")
383
-
384
- self._server = None
385
- self._server_thread = None
386
-
387
- except Exception as e:
388
- logger.error(f"Error stopping WebSocket server: {e}")
389
- else:
390
- logger.info("No WebSocket server instance found")
391
-
392
- def _cleanup_camera_windows(self) -> None:
393
- """Clean up all OpenCV windows and reset tracking"""
394
- try:
395
- # Only cleanup if we're in the main thread to avoid Qt warnings
396
- if threading.current_thread() == threading.main_thread():
397
- cv2.destroyAllWindows()
398
- cv2.waitKey(1)
399
- else:
400
- # If not in main thread, just skip OpenCV cleanup
401
- # The windows will close when the main thread exits anyway
402
- pass
403
- except Exception:
404
- # Ignore any errors during cleanup
405
- pass
406
-
407
- def shutdown(self) -> None:
408
- """Shutdown the LuckyRobots node and clean up resources"""
409
- if not self._running:
410
- logger.info("LuckyRobots already shut down")
411
- return
412
-
413
- logger.info("Starting LuckyRobots shutdown sequence")
414
- self._running = False
415
-
416
- self._cancel_pending_operations()
417
- self._shutdown_nodes()
418
- self._stop_luckyworld()
419
- self._shutdown_transport()
420
- self._stop_websocket_server()
421
- self._cleanup_resources()
422
- self._shutdown_event.set()
423
-
424
- logger.info("LuckyRobots shutdown complete")
425
-
426
- exit(0)
427
-
428
- def _cancel_pending_operations(self) -> None:
429
- """Cancel all pending reset and step operations"""
430
- logger.info("Cancelling pending operations")
431
-
432
- # Cancel pending resets
433
- for _, future in self._pending_resets.items():
434
- if not future.done():
435
- future.cancel()
436
- self._pending_resets.clear()
437
-
438
- # Cancel pending steps
439
- for _, future in self._pending_steps.items():
440
- if not future.done():
441
- future.cancel()
442
- self._pending_steps.clear()
443
-
444
- def _shutdown_nodes(self) -> None:
445
- """Shutdown all registered nodes with timeout"""
446
- if not self._nodes:
447
- return
448
-
449
- logger.info(f"Shutting down {len(self._nodes)} registered nodes")
450
-
451
- failed_nodes = []
452
- for node_name, node in self._nodes.items():
453
- try:
454
- logger.debug(f"Shutting down node: {node_name}")
455
- node.shutdown()
456
- logger.debug(f"Node {node_name} shut down successfully")
457
- except Exception as e:
458
- logger.error(f"Error shutting down node {node_name}: {e}")
459
- failed_nodes.append(node_name)
460
-
461
- if failed_nodes:
462
- logger.warning(f"Failed to shutdown nodes: {', '.join(failed_nodes)}")
463
- else:
464
- logger.info("All nodes shut down successfully")
465
-
466
- self._nodes.clear()
467
-
468
- def _stop_luckyworld(self) -> None:
469
- """Stop the LuckyWorld executable with error handling"""
470
- try:
471
- stop_luckyworld()
472
- except Exception as e:
473
- logger.error(f"Error stopping LuckyWorld executable: {e}")
474
- # Don't raise - continue with shutdown even if LuckyWorld fails to stop
475
-
476
- def _shutdown_transport(self) -> None:
477
- """Shutdown the transport layer with timeout"""
478
- try:
479
- logger.info("Shutting down transport layer...")
480
- super().shutdown()
481
- logger.info("Transport layer shut down")
482
- except Exception as e:
483
- logger.error(f"Error shutting down transport layer: {e}")
484
-
485
- def _cleanup_resources(self) -> None:
486
- """Clean up all remaining resources"""
487
-
488
- # Cleanup camera windows
489
- self._cleanup_camera_windows()
490
-
491
- # Close WebSocket connections
492
- if hasattr(self, "world_client") and self.world_client:
493
- try:
494
- self.world_client.close()
495
- self.world_client = None
496
- except Exception as e:
497
- logger.debug(f"Error closing world client: {e}")
498
-
499
- # Clear any remaining state
500
- self.robot_client = None
501
- self.world_client = None
502
-
503
- logger.info("Resource cleanup complete")
504
-
505
-
506
- @app.websocket("/nodes")
507
- async def nodes_endpoint(websocket: WebSocket) -> None:
508
- """WebSocket endpoint for node communication"""
509
- await websocket.accept()
510
- node_name = None
511
-
512
- try:
513
- # Wait for the first message, which should be NODE_ANNOUNCE
514
- message = await websocket.receive_bytes()
515
- message_data = msgpack.unpackb(message)
516
- message = TransportMessage(**message_data)
517
-
518
- if message.msg_type != MessageType.NODE_ANNOUNCE:
519
- logger.warning(
520
- f"First message from node should be NODE_ANNOUNCE, got {message.msg_type}"
521
- )
522
- await websocket.close(4000, "First message must be NODE_ANNOUNCE")
523
- return
524
-
525
- # Register the node
526
- node_name = message.node_name
527
- await manager.register_node(node_name, websocket)
528
-
529
- # Message processing loop
530
- while True:
531
- try:
532
- message = await websocket.receive_bytes()
533
- message_data = msgpack.unpackb(message)
534
- message = TransportMessage(**message_data)
535
-
536
- # Process message based on type
537
- handlers = {
538
- MessageType.SUBSCRIBE: lambda: manager.subscribe(
539
- node_name, message.topic_or_service
540
- ),
541
- MessageType.UNSUBSCRIBE: lambda: manager.unsubscribe(
542
- node_name, message.topic_or_service
543
- ),
544
- MessageType.SERVICE_REGISTER: lambda: manager.register_service(
545
- node_name, message.topic_or_service
546
- ),
547
- MessageType.SERVICE_UNREGISTER: lambda: manager.unregister_service(
548
- node_name, message.topic_or_service
549
- ),
550
- MessageType.NODE_SHUTDOWN: lambda: None, # Will break the loop
551
- }
552
-
553
- if message.msg_type in handlers:
554
- if message.msg_type == MessageType.NODE_SHUTDOWN:
555
- break
556
- await handlers[message.msg_type]()
557
- else:
558
- await manager.route_message(message)
559
-
560
- except msgpack.UnpackValueError:
561
- logger.error(f"Received invalid msgpack from {node_name}")
562
- except Exception as e:
563
- logger.error(f"Error processing message from {node_name}: {e}")
564
-
565
- except WebSocketDisconnect:
566
- logger.info(f"Node {node_name} disconnected")
567
-
568
-
569
- @app.websocket("/world")
570
- async def world_endpoint(websocket: WebSocket) -> None:
571
- """WebSocket endpoint for world client communication"""
572
- await websocket.accept()
573
-
574
- if hasattr(app, "lucky_robots"):
575
- app.lucky_robots.world_client = websocket
576
-
577
- lucky_robots = app.lucky_robots
578
- pending_resets = lucky_robots._pending_resets
579
- pending_steps = lucky_robots._pending_steps
580
-
581
- try:
582
- while True:
583
- try:
584
- message_bytes = await websocket.receive_bytes()
585
- message_data = msgpack.unpackb(message_bytes)
586
- request_type = message_data.get("RequestType")
587
- request_id = message_data.get("RequestID")
588
- shared_loop = get_event_loop()
589
-
590
- if request_type == "reset_response":
591
- future = pending_resets.get(request_id)
592
- shared_loop.call_soon_threadsafe(
593
- lambda: future.set_result(message_data)
594
- if future is not None and not future.done()
595
- else None
596
- )
597
- shared_loop.call_soon_threadsafe(
598
- lambda: pending_resets.pop(request_id, None)
599
- )
600
- elif request_type == "step_response":
601
- future = pending_steps.get(request_id)
602
- shared_loop.call_soon_threadsafe(
603
- lambda: future.set_result(message_data)
604
- if future is not None and not future.done()
605
- else None
606
- )
607
- shared_loop.call_soon_threadsafe(
608
- lambda: pending_resets.pop(request_id, None)
609
- if request_type == "reset_response"
610
- else pending_steps.pop(request_id, None)
611
- )
612
- else:
613
- logger.warning(f"Unhandled message type: {request_type}")
614
-
615
- except WebSocketDisconnect as e:
616
- logger.info(f"WebSocket disconnected. Code: {e.code}")
617
- break
618
- except Exception as e:
619
- logger.error(f"Message processing error: {type(e).__name__}: {e}")
620
- break
621
-
622
- except WebSocketDisconnect:
623
- pass
624
- except Exception as e:
625
- logger.error(f"Critical error in world_endpoint: {type(e).__name__}: {e}")
626
- finally:
627
- if hasattr(app, "lucky_robots") and app.lucky_robots.world_client == websocket:
628
- app.lucky_robots.world_client = None