feat(websockets): implement WebSockets V3 with granular Redis Pub/Sub
- Introduced WS_Message_V3 standardized Pydantic model and WS_Manager_V3. - Implemented /v3/ws/ endpoint with granular Redis routing to solve "noisy neighbor" scaling issues. - Added presence tracking using Redis Sets for group coordination. - Comprehensive test suite added (unit and integration) covering models, manager, and routing logic. - Documentation: Created V3 Frontend WebSocket Guide and Project design spec. - Updated main Frontend API guide and tests README with new standards.
This commit is contained in:
74
tests/unit/test_unit_websockets_v3.py
Normal file
74
tests/unit/test_unit_websockets_v3.py
Normal file
@@ -0,0 +1,74 @@
|
||||
import sys
|
||||
import os
|
||||
import asyncio
|
||||
import unittest
|
||||
from unittest.mock import MagicMock, AsyncMock, patch
|
||||
from datetime import datetime, timezone
|
||||
|
||||
# Add project root to path
|
||||
sys.path.append(os.getcwd())
|
||||
|
||||
# Mock app.config BEFORE imports
|
||||
mock_config = MagicMock()
|
||||
mock_config.settings = MagicMock()
|
||||
mock_config.settings.REDIS = {'server': 'localhost', 'port': 6379}
|
||||
sys.modules["app.config"] = mock_config
|
||||
|
||||
from app.lib_websockets_v3 import WS_Message_V3, WS_Manager_V3
|
||||
|
||||
class TestWSV3Library(unittest.TestCase):
|
||||
|
||||
def test_message_model_validation(self):
|
||||
print("\n--- Testing WS_Message_V3 Validation ---")
|
||||
data = {
|
||||
"msg_type": "cmd",
|
||||
"target": "group",
|
||||
"from_id": "client_abc",
|
||||
"group_id": "group_123",
|
||||
"cmd": "RELOAD",
|
||||
"payload": {"force": True}
|
||||
}
|
||||
msg = WS_Message_V3(**data)
|
||||
self.assertEqual(msg.version, "3")
|
||||
self.assertEqual(msg.cmd, "RELOAD")
|
||||
self.assertTrue(isinstance(msg.sent_at.isoformat(), str))
|
||||
print("✅ Model validation passed.")
|
||||
|
||||
def test_channel_name_generation(self):
|
||||
print("\n--- Testing Channel Name Generation ---")
|
||||
manager = WS_Manager_V3()
|
||||
channels = manager.get_channel_names("client_abc", "group_123")
|
||||
|
||||
self.assertIn("ws:client:client_abc", channels)
|
||||
self.assertIn("ws:group:group_123", channels)
|
||||
self.assertIn("ws:broadcast", channels)
|
||||
print("✅ Channel name generation passed.")
|
||||
|
||||
@patch('redis.asyncio.Redis.from_url')
|
||||
def test_publish_routing(self, mock_redis_factory):
|
||||
print("\n--- Testing Publish Routing ---")
|
||||
mock_redis = AsyncMock()
|
||||
mock_redis_factory.return_value = mock_redis
|
||||
|
||||
manager = WS_Manager_V3()
|
||||
|
||||
async def run_test():
|
||||
# 1. Test Group Routing
|
||||
msg_group = WS_Message_V3(
|
||||
msg_type="msg", target="group", from_id="sender", group_id="target_group"
|
||||
)
|
||||
await manager.publish_message(msg_group)
|
||||
mock_redis.publish.assert_called_with("ws:group:target_group", unittest.mock.ANY)
|
||||
|
||||
# 2. Test Direct Routing
|
||||
msg_direct = WS_Message_V3(
|
||||
msg_type="msg", target="direct", from_id="sender", to_id="target_client"
|
||||
)
|
||||
await manager.publish_message(msg_direct)
|
||||
mock_redis.publish.assert_called_with("ws:client:target_client", unittest.mock.ANY)
|
||||
|
||||
asyncio.run(run_test())
|
||||
print("✅ Publish routing logic passed.")
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
||||
126
tests/unit/test_unit_websockets_v3_router.py
Normal file
126
tests/unit/test_unit_websockets_v3_router.py
Normal file
@@ -0,0 +1,126 @@
|
||||
import sys
|
||||
import os
|
||||
import asyncio
|
||||
import json
|
||||
import unittest
|
||||
from unittest.mock import MagicMock, AsyncMock, patch
|
||||
|
||||
# Add project root to path
|
||||
sys.path.append(os.getcwd())
|
||||
|
||||
# Mock app.config BEFORE imports to prevent attempt to load real settings
|
||||
mock_config = MagicMock()
|
||||
mock_config.settings = MagicMock()
|
||||
mock_config.settings.REDIS = {'server': 'localhost', 'port': 6379}
|
||||
sys.modules["app.config"] = mock_config
|
||||
|
||||
# Mock DB related modules to prevent circular imports or DB connection attempts
|
||||
sys.modules["app.db_sql"] = MagicMock()
|
||||
sys.modules["app.lib_sql_core"] = MagicMock()
|
||||
sys.modules["app.db_connection"] = MagicMock()
|
||||
|
||||
from app.routers.websockets_v3 import v3_ws_endpoint
|
||||
|
||||
class TestWSV3Router(unittest.TestCase):
|
||||
|
||||
@patch('app.routers.websockets_v3.ws_manager_v3')
|
||||
def test_v3_ws_endpoint_logic(self, mock_manager):
|
||||
"""
|
||||
Tests the core logic of the V3 WebSocket endpoint, ensuring
|
||||
Redis subscription and bidirectional message handling are initiated.
|
||||
"""
|
||||
# 1. Setup WebSocket Mock
|
||||
mock_ws = AsyncMock()
|
||||
|
||||
# 2. Setup Redis PubSub Mock
|
||||
mock_pubsub = MagicMock()
|
||||
mock_pubsub.subscribe = AsyncMock()
|
||||
mock_pubsub.unsubscribe = AsyncMock()
|
||||
mock_pubsub.close = AsyncMock()
|
||||
|
||||
mock_message = {
|
||||
'type': 'message',
|
||||
'data': json.dumps({
|
||||
"version": "3",
|
||||
"msg_type": "msg",
|
||||
"target": "group",
|
||||
"from_id": "other_client",
|
||||
"msg": "Hello from Redis",
|
||||
"payload": {},
|
||||
"sent_at": "2026-01-30T12:00:00Z"
|
||||
})
|
||||
}
|
||||
|
||||
# Signal to coordinate loops
|
||||
msg_delivered = asyncio.Event()
|
||||
|
||||
# Counters to break the 'while True' loops in the endpoint
|
||||
get_msg_count = 0
|
||||
recv_json_count = 0
|
||||
|
||||
async def mock_get_message(*args, **kwargs):
|
||||
nonlocal get_msg_count
|
||||
get_msg_count += 1
|
||||
if get_msg_count == 1:
|
||||
msg_delivered.set()
|
||||
return mock_message
|
||||
await asyncio.sleep(0.05)
|
||||
# Raise CancelledError to terminate the loop cleanly
|
||||
raise asyncio.CancelledError("Terminate sender loop")
|
||||
|
||||
mock_pubsub.get_message = mock_get_message
|
||||
|
||||
mock_redis = MagicMock()
|
||||
mock_redis.pubsub.return_value = mock_pubsub
|
||||
|
||||
# 3. Setup Manager Mock
|
||||
mock_manager.get_redis = AsyncMock(return_value=mock_redis)
|
||||
mock_manager.update_presence = AsyncMock()
|
||||
mock_manager.publish_message = AsyncMock()
|
||||
mock_manager.get_channel_names.return_value = ["ws:group:test"]
|
||||
|
||||
# Mock incoming websocket message
|
||||
async def mock_receive_json():
|
||||
nonlocal recv_json_count
|
||||
recv_json_count += 1
|
||||
if recv_json_count == 1:
|
||||
# Wait until the sender loop has processed the Redis message
|
||||
await msg_delivered.wait()
|
||||
return {
|
||||
"msg_type": "msg",
|
||||
"target": "group",
|
||||
"msg": "Client A saying hi"
|
||||
}
|
||||
await asyncio.sleep(0.05)
|
||||
# Raise CancelledError to terminate the loop cleanly
|
||||
raise asyncio.CancelledError("Terminate receiver loop")
|
||||
|
||||
mock_ws.receive_json.side_effect = mock_receive_json
|
||||
|
||||
# 4. Run the endpoint logic
|
||||
async def run_endpoint():
|
||||
try:
|
||||
# Execute endpoint with a short timeout
|
||||
await asyncio.wait_for(
|
||||
v3_ws_endpoint(mock_ws, "test_group", "client_a"),
|
||||
timeout=0.5
|
||||
)
|
||||
except (asyncio.TimeoutError, asyncio.CancelledError):
|
||||
pass
|
||||
except Exception as e:
|
||||
# Suppress our expected loop-termination messages
|
||||
if "Terminate" not in str(e):
|
||||
raise
|
||||
|
||||
asyncio.run(run_endpoint())
|
||||
|
||||
# 5. Verifications
|
||||
mock_ws.accept.assert_called_once()
|
||||
mock_manager.update_presence.assert_any_call("client_a", "test_group", online=True)
|
||||
|
||||
# Verify message from Redis was forwarded to WebSocket
|
||||
mock_ws.send_text.assert_called()
|
||||
print("✅ WebSocket Router unit logic verified.")
|
||||
|
||||
if __name__ == "__main__":
|
||||
unittest.main()
|
||||
Reference in New Issue
Block a user