mirror of
https://github.com/ruvnet/RuView
synced 2026-06-18 11:43:19 +00:00
f21daf9aa8
New sub-package `wifi_densepose.client` (no PyO3, no Rust deps):
- ws.SensingClient — asyncio websockets>=12 wrapper for the Rust
sensing-server /ws/sensing endpoint. Yields typed dataclasses
(ConnectionEstablishedMessage, EdgeVitalsMessage, PoseDataMessage)
with raw-payload fallback for forward-compat with unknown types.
Malformed frames log+drop without breaking the stream.
- mqtt.RuViewMqttClient — paho-mqtt v2 wrapper using the explicit
CallbackAPIVersion.VERSION2 API. Per-instance unique client_id by
default (rumqttc memory lesson). MQTT v5-spec-correct topic
wildcard matcher: + as whole-level wildcard, # matches the prefix
itself plus all sub-levels. Auto-resubscribes on reconnect.
Handler exceptions are caught and logged so a misbehaving callback
can't crash the network loop.
- primitives.SemanticPrimitiveListener — typed router for the 10
HA-MIND fused inference outputs from ADR-115 §3.12
(SomeoneSleeping, PossibleDistress, RoomActive, ElderlyInactivity-
Anomaly, MeetingInProgress, BathroomOccupied, FallRiskElevated,
BedExit, NoMovementSafety, MultiRoomTransition). Decodes both
JSON payloads with confidence+explanation AND plain HA state
strings ("ON"/"OFF"/numeric). Pluggable into RuViewMqttClient.
- ha.HABlueprintHelper — read-only parser for the
homeassistant/<kind>/wifi_densepose_<node>/<id>/config payload
family. Aggregator queries: entities_for_node, by_device_class,
nodes. Useful for blueprint authors + dashboard introspection.
Test coverage (63 new tests, 156 total in Python suite):
- test_client_ha — 18 tests (topic+payload parsing, aggregator)
- test_client_primitives — 13 tests (enum coverage, listener routing)
- test_client_mqtt — 17 tests (matcher parametrize, dispatch path,
on_connect, exception isolation) — no broker needed
- test_client_ws — 6 tests including end-to-end against an in-process
websockets.serve() fixture exercising all 4 message types plus a
malformed-frame survival check
Post-bridge wheel size: 238 KB (well under ADR §5.4 5 MB budget).
Refs: docs/adr/ADR-117-pip-wifi-densepose-modernization.md §5.6
Refs: docs/adr/ADR-115-home-assistant-integration.md §3.12
Refs: #785
Co-Authored-By: claude-flow <ruv@ruv.net>
196 lines
6.5 KiB
Python
196 lines
6.5 KiB
Python
"""ADR-117 P4 — End-to-end test for SensingClient against an in-process
|
|
WS server.
|
|
|
|
We spin up a real `websockets.serve()` server in the same event loop,
|
|
send the four message types defined in ADR-115 §1, and assert the
|
|
client decodes them into the right dataclasses. No mocks — the only
|
|
moving part this test does NOT exercise is the actual sensing-server
|
|
binary, but the wire protocol is the contract under test here.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
import json
|
|
from typing import Any
|
|
|
|
import pytest
|
|
import websockets
|
|
|
|
from wifi_densepose.client import (
|
|
ConnectionEstablishedMessage,
|
|
EdgeVitalsMessage,
|
|
PoseDataMessage,
|
|
SensingClient,
|
|
SensingMessage,
|
|
)
|
|
|
|
|
|
# ─── In-process WS server fixture ────────────────────────────────────
|
|
|
|
|
|
_FIXTURE_MESSAGES = [
|
|
{
|
|
"type": "connection_established",
|
|
"node_id": "test-node-001",
|
|
"version": "0.7.4",
|
|
"capabilities": ["edge_vitals", "pose_data"],
|
|
},
|
|
{
|
|
"type": "edge_vitals",
|
|
"node_id": "test-node-001",
|
|
"presence": True,
|
|
"fall_detected": False,
|
|
"motion": 0.21,
|
|
"breathing_rate_bpm": 14.5,
|
|
"heartrate_bpm": 72.3,
|
|
"n_persons": 1,
|
|
"motion_energy": 0.034,
|
|
"presence_score": 0.91,
|
|
"rssi": -42.0,
|
|
},
|
|
{
|
|
"type": "pose_data",
|
|
"node_id": "test-node-001",
|
|
"timestamp": 1700000000.5,
|
|
"persons": [{"id": 1, "keypoints": []}],
|
|
"confidence": 0.88,
|
|
},
|
|
# Unknown type — should NOT crash the stream; should yield a plain
|
|
# SensingMessage.
|
|
{
|
|
"type": "future_message_type_not_yet_modelled",
|
|
"extra": "data",
|
|
},
|
|
]
|
|
|
|
|
|
async def _handler(websocket: Any) -> None:
|
|
for msg in _FIXTURE_MESSAGES:
|
|
await websocket.send(json.dumps(msg))
|
|
# Send one malformed frame to assert the client logs+drops it
|
|
# rather than crashing the stream.
|
|
await websocket.send("{not valid json")
|
|
# And one final "real" message so the test can confirm the stream
|
|
# survived the malformed one.
|
|
await websocket.send(json.dumps({"type": "edge_vitals", "node_id": "post-bad-frame"}))
|
|
|
|
|
|
@pytest.fixture
|
|
async def ws_server() -> Any:
|
|
"""Start a websocket server on a random port; yield the bound URL."""
|
|
server = await websockets.serve(_handler, "127.0.0.1", 0)
|
|
# Get the bound port (host="127.0.0.1" returns one socket).
|
|
port = server.sockets[0].getsockname()[1] # type: ignore[union-attr]
|
|
try:
|
|
yield f"ws://127.0.0.1:{port}/ws/sensing"
|
|
finally:
|
|
server.close()
|
|
await server.wait_closed()
|
|
|
|
|
|
# ─── End-to-end stream test ──────────────────────────────────────────
|
|
|
|
|
|
async def test_sensing_client_decodes_all_message_types(ws_server: str) -> None:
|
|
received: list[SensingMessage] = []
|
|
async with SensingClient(ws_server) as client:
|
|
async for msg in client.stream():
|
|
received.append(msg)
|
|
if len(received) >= len(_FIXTURE_MESSAGES) + 1: # +1 for post-bad-frame
|
|
break
|
|
|
|
# connection_established → typed
|
|
assert isinstance(received[0], ConnectionEstablishedMessage)
|
|
assert received[0].node_id == "test-node-001"
|
|
assert received[0].version == "0.7.4"
|
|
assert "edge_vitals" in received[0].capabilities
|
|
|
|
# edge_vitals → typed with full fields
|
|
assert isinstance(received[1], EdgeVitalsMessage)
|
|
assert received[1].presence is True
|
|
assert received[1].fall_detected is False
|
|
assert received[1].breathing_rate_bpm == 14.5
|
|
assert received[1].heartrate_bpm == 72.3
|
|
assert received[1].n_persons == 1
|
|
assert received[1].rssi == -42.0
|
|
|
|
# pose_data → typed
|
|
assert isinstance(received[2], PoseDataMessage)
|
|
assert received[2].timestamp == 1700000000.5
|
|
assert len(received[2].persons) == 1
|
|
assert received[2].confidence == 0.88
|
|
|
|
# Unknown type → plain SensingMessage (forward-compat)
|
|
assert type(received[3]) is SensingMessage # exact base class
|
|
assert received[3].type == "future_message_type_not_yet_modelled"
|
|
assert received[3].raw["extra"] == "data"
|
|
|
|
# After the malformed frame: the stream should have survived and
|
|
# yielded the post-bad-frame message.
|
|
assert isinstance(received[4], EdgeVitalsMessage)
|
|
assert received[4].node_id == "post-bad-frame"
|
|
|
|
|
|
async def test_sensing_client_recv_one(ws_server: str) -> None:
|
|
async with SensingClient(ws_server) as client:
|
|
msg = await client.recv_one(timeout=2.0)
|
|
assert isinstance(msg, ConnectionEstablishedMessage)
|
|
|
|
|
|
async def test_sensing_client_raises_when_used_without_context() -> None:
|
|
client = SensingClient("ws://127.0.0.1:1/") # never connects
|
|
with pytest.raises(RuntimeError, match="not connected"):
|
|
await client.recv_one(timeout=0.1)
|
|
with pytest.raises(RuntimeError, match="not connected"):
|
|
async for _ in client.stream():
|
|
pass
|
|
|
|
|
|
async def test_sensing_client_close_is_idempotent(ws_server: str) -> None:
|
|
client = SensingClient(ws_server)
|
|
await client.__aenter__()
|
|
await client.close()
|
|
await client.close() # second close is a no-op
|
|
|
|
|
|
def test_sensing_client_decoder_directly() -> None:
|
|
"""The decoder is pure — exercise it without bringing up a WS
|
|
server, so we have a fast unit test for the type mapping."""
|
|
from wifi_densepose.client.ws import _decode
|
|
|
|
msg = _decode(json.dumps({
|
|
"type": "edge_vitals",
|
|
"node_id": "x",
|
|
"presence": True,
|
|
"fall_detected": False,
|
|
"motion": 1.5,
|
|
}))
|
|
assert isinstance(msg, EdgeVitalsMessage)
|
|
assert msg.presence is True
|
|
assert msg.motion == 1.5
|
|
assert msg.breathing_rate_bpm is None # not present → None, not 0.0
|
|
assert msg.heartrate_bpm is None
|
|
assert msg.rssi is None
|
|
|
|
|
|
def test_sensing_client_decoder_handles_None_subfields() -> None:
|
|
"""When the sensing-server explicitly emits null for HR/BR (no
|
|
measurement yet), the client should propagate None, not crash."""
|
|
from wifi_densepose.client.ws import _decode
|
|
|
|
msg = _decode(json.dumps({
|
|
"type": "edge_vitals",
|
|
"node_id": "x",
|
|
"presence": False,
|
|
"fall_detected": False,
|
|
"motion": 0.0,
|
|
"breathing_rate_bpm": None,
|
|
"heartrate_bpm": None,
|
|
"rssi": None,
|
|
}))
|
|
assert isinstance(msg, EdgeVitalsMessage)
|
|
assert msg.breathing_rate_bpm is None
|
|
assert msg.heartrate_bpm is None
|
|
assert msg.rssi is None
|