mirror of
https://github.com/zvx-echo6/meshai.git
synced 2026-05-21 15:14:45 +02:00
Adds dedicated renderer classes per channel type:
- MeshRenderer produces 1+ chunks <=200 chars with (k/N) counters
when the payload overflows. Reuses the toggle-label vocabulary
from the digest. Mesh channels skip re-chunking when the payload
already carries chunk_index metadata (digest path).
- EmailRenderer produces {subject, body} with structured context
lines. Plain text only; HTML body is a future polish.
- WebhookRenderer produces a JSON-serializable dict with stable
schema_version 1.0. Optional fields omitted (not nulled) for
compactness. Designed for reuse by Phase 2.6.5's MQTT event
publisher.
- All four channel implementations (MeshBroadcast, MeshDM, Email,
Webhook) now call their renderer in deliver() before transport.
- New renderer tests cover each renderer in isolation; new channel
integration tests confirm channels actually call their renderer.
Renderers are pure functions of the payload - no network, no
state, fully testable without mocking I/O. The future MQTT
publisher will instantiate WebhookRenderer directly.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
214 lines
6.5 KiB
Python
214 lines
6.5 KiB
Python
"""Tests for channel-renderer integration (Phase 2.5b)."""
|
|
|
|
import asyncio
|
|
import time
|
|
from unittest.mock import MagicMock, AsyncMock, patch
|
|
|
|
import pytest
|
|
|
|
from meshai.notifications.events import NotificationPayload
|
|
from meshai.notifications.channels import (
|
|
MeshBroadcastChannel,
|
|
MeshDMChannel,
|
|
EmailChannel,
|
|
WebhookChannel,
|
|
)
|
|
|
|
|
|
# ============================================================
|
|
# MESH CHANNEL RENDERING TESTS
|
|
# ============================================================
|
|
|
|
def test_mesh_channel_uses_mesh_renderer():
|
|
"""MeshBroadcastChannel renders long messages to multiple chunks."""
|
|
mock_connector = MagicMock()
|
|
|
|
channel = MeshBroadcastChannel(
|
|
connector=mock_connector,
|
|
channel_index=0,
|
|
)
|
|
|
|
# Build a long message that will require chunking
|
|
long_message = "This is a very long alert message that exceeds the character limit. " * 5
|
|
|
|
payload = NotificationPayload(
|
|
message=long_message,
|
|
category="weather_warning",
|
|
severity="priority",
|
|
timestamp=time.time(),
|
|
event_type="weather_warning",
|
|
)
|
|
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Should have called send_message multiple times (once per chunk)
|
|
assert mock_connector.send_message.call_count >= 2
|
|
|
|
# Each call's text should be <= 200 chars
|
|
for call in mock_connector.send_message.call_args_list:
|
|
text = call.kwargs.get("text", call.args[0] if call.args else "")
|
|
assert len(text) <= 200
|
|
|
|
|
|
def test_mesh_channel_uses_payload_message_directly_when_chunk_metadata_set():
|
|
"""Pre-chunked payloads (from digest) skip re-rendering."""
|
|
mock_connector = MagicMock()
|
|
|
|
channel = MeshBroadcastChannel(
|
|
connector=mock_connector,
|
|
channel_index=0,
|
|
)
|
|
|
|
# Payload with chunk metadata set (from digest scheduler)
|
|
payload = NotificationPayload(
|
|
message="pre-chunked text",
|
|
category="digest",
|
|
severity="routine",
|
|
timestamp=time.time(),
|
|
chunk_index=1,
|
|
chunk_total=3,
|
|
)
|
|
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Should have called send_message exactly once
|
|
assert mock_connector.send_message.call_count == 1
|
|
# Should use the message directly
|
|
call = mock_connector.send_message.call_args
|
|
text = call.kwargs.get("text", call.args[0] if call.args else "")
|
|
assert text == "pre-chunked text"
|
|
|
|
|
|
def test_mesh_dm_channel_uses_mesh_renderer():
|
|
"""MeshDMChannel renders long messages to chunks for each recipient."""
|
|
mock_connector = MagicMock()
|
|
|
|
channel = MeshDMChannel(
|
|
connector=mock_connector,
|
|
node_ids=["!node1", "!node2"],
|
|
)
|
|
|
|
long_message = "This is a long DM message that should be chunked. " * 4
|
|
|
|
payload = NotificationPayload(
|
|
message=long_message,
|
|
category="test",
|
|
severity="routine",
|
|
timestamp=time.time(),
|
|
)
|
|
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Should have called send_message multiple times
|
|
# (chunks * nodes)
|
|
assert mock_connector.send_message.call_count >= 2
|
|
|
|
|
|
def test_mesh_dm_channel_uses_payload_message_directly_when_chunk_metadata_set():
|
|
"""Pre-chunked DM payloads skip re-rendering."""
|
|
mock_connector = MagicMock()
|
|
|
|
channel = MeshDMChannel(
|
|
connector=mock_connector,
|
|
node_ids=["!node1"],
|
|
)
|
|
|
|
payload = NotificationPayload(
|
|
message="pre-chunked DM",
|
|
category="digest",
|
|
severity="routine",
|
|
timestamp=time.time(),
|
|
chunk_index=2,
|
|
chunk_total=5,
|
|
)
|
|
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Should use message directly, once per node
|
|
assert mock_connector.send_message.call_count == 1
|
|
call = mock_connector.send_message.call_args
|
|
text = call.kwargs.get("text", call.args[0] if call.args else "")
|
|
assert text == "pre-chunked DM"
|
|
|
|
|
|
# ============================================================
|
|
# EMAIL CHANNEL RENDERING TESTS
|
|
# ============================================================
|
|
|
|
def test_email_channel_uses_email_renderer():
|
|
"""EmailChannel uses renderer for subject and body."""
|
|
channel = EmailChannel(
|
|
smtp_host="localhost",
|
|
smtp_port=25,
|
|
smtp_user="",
|
|
smtp_password="",
|
|
smtp_tls=False,
|
|
from_address="test@example.com",
|
|
recipients=["user@example.com"],
|
|
)
|
|
|
|
payload = NotificationPayload(
|
|
message="Test alert message",
|
|
category="weather_warning",
|
|
severity="immediate",
|
|
timestamp=time.time(),
|
|
event_type="weather_warning",
|
|
)
|
|
|
|
# Mock the _send_email method
|
|
with patch.object(channel, "_send_email") as mock_send:
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Should have been called with renderer output
|
|
mock_send.assert_called_once()
|
|
call_args = mock_send.call_args
|
|
subject = call_args.args[0]
|
|
body = call_args.args[1]
|
|
|
|
# Renderer format checks
|
|
assert "[MeshAI]" in subject
|
|
assert "IMMEDIATE" in subject
|
|
assert "Test alert message" in body
|
|
assert "Severity:" in body
|
|
|
|
|
|
# ============================================================
|
|
# WEBHOOK CHANNEL RENDERING TESTS
|
|
# ============================================================
|
|
|
|
def test_webhook_channel_uses_webhook_renderer():
|
|
"""WebhookChannel uses renderer for JSON payload."""
|
|
channel = WebhookChannel(
|
|
url="https://example.com/webhook",
|
|
headers={},
|
|
)
|
|
|
|
payload = NotificationPayload(
|
|
message="Test webhook message",
|
|
category="test",
|
|
severity="priority",
|
|
timestamp=time.time(),
|
|
event_type="battery_warning",
|
|
)
|
|
|
|
# Mock httpx
|
|
with patch("meshai.notifications.channels.httpx.AsyncClient") as mock_client_class:
|
|
mock_client = MagicMock()
|
|
mock_response = MagicMock()
|
|
mock_response.status_code = 200
|
|
mock_client.post = AsyncMock(return_value=mock_response)
|
|
mock_client.__aenter__ = AsyncMock(return_value=mock_client)
|
|
mock_client.__aexit__ = AsyncMock(return_value=None)
|
|
mock_client_class.return_value = mock_client
|
|
|
|
asyncio.run(channel.deliver(payload, None))
|
|
|
|
# Check the POST was called
|
|
mock_client.post.assert_called_once()
|
|
call_kwargs = mock_client.post.call_args.kwargs
|
|
|
|
# Should have JSON payload with schema_version
|
|
json_payload = call_kwargs.get("json", {})
|
|
assert "schema_version" in json_payload
|
|
assert json_payload["schema_version"] == "1.0"
|
|
assert json_payload["message"] == "Test webhook message"
|