362 lines
13 KiB
Python
362 lines
13 KiB
Python
|
|
"""Tests for the BlueBubbles iMessage gateway adapter."""
|
||
|
|
import pytest
|
||
|
|
|
||
|
|
from gateway.config import Platform, PlatformConfig
|
||
|
|
|
||
|
|
|
||
|
|
def _make_adapter(monkeypatch, **extra):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_PASSWORD", "secret")
|
||
|
|
from gateway.platforms.bluebubbles import BlueBubblesAdapter
|
||
|
|
|
||
|
|
cfg = PlatformConfig(
|
||
|
|
enabled=True,
|
||
|
|
extra={
|
||
|
|
"server_url": "http://localhost:1234",
|
||
|
|
"password": "secret",
|
||
|
|
**extra,
|
||
|
|
},
|
||
|
|
)
|
||
|
|
return BlueBubblesAdapter(cfg)
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesPlatformEnum:
|
||
|
|
def test_bluebubbles_enum_exists(self):
|
||
|
|
assert Platform.BLUEBUBBLES.value == "bluebubbles"
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesConfigLoading:
|
||
|
|
def test_apply_env_overrides_bluebubbles(self, monkeypatch):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_PASSWORD", "secret")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_WEBHOOK_PORT", "9999")
|
||
|
|
from gateway.config import GatewayConfig, _apply_env_overrides
|
||
|
|
|
||
|
|
config = GatewayConfig()
|
||
|
|
_apply_env_overrides(config)
|
||
|
|
assert Platform.BLUEBUBBLES in config.platforms
|
||
|
|
bc = config.platforms[Platform.BLUEBUBBLES]
|
||
|
|
assert bc.enabled is True
|
||
|
|
assert bc.extra["server_url"] == "http://localhost:1234"
|
||
|
|
assert bc.extra["password"] == "secret"
|
||
|
|
assert bc.extra["webhook_port"] == 9999
|
||
|
|
|
||
|
|
def test_connected_platforms_includes_bluebubbles(self, monkeypatch):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_PASSWORD", "secret")
|
||
|
|
from gateway.config import GatewayConfig, _apply_env_overrides
|
||
|
|
|
||
|
|
config = GatewayConfig()
|
||
|
|
_apply_env_overrides(config)
|
||
|
|
assert Platform.BLUEBUBBLES in config.get_connected_platforms()
|
||
|
|
|
||
|
|
def test_home_channel_set_from_env(self, monkeypatch):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_PASSWORD", "secret")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_HOME_CHANNEL", "user@example.com")
|
||
|
|
from gateway.config import GatewayConfig, _apply_env_overrides
|
||
|
|
|
||
|
|
config = GatewayConfig()
|
||
|
|
_apply_env_overrides(config)
|
||
|
|
hc = config.platforms[Platform.BLUEBUBBLES].home_channel
|
||
|
|
assert hc is not None
|
||
|
|
assert hc.chat_id == "user@example.com"
|
||
|
|
|
||
|
|
def test_not_connected_without_password(self, monkeypatch):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.delenv("BLUEBUBBLES_PASSWORD", raising=False)
|
||
|
|
from gateway.config import GatewayConfig, _apply_env_overrides
|
||
|
|
|
||
|
|
config = GatewayConfig()
|
||
|
|
_apply_env_overrides(config)
|
||
|
|
assert Platform.BLUEBUBBLES not in config.get_connected_platforms()
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesHelpers:
|
||
|
|
def test_check_requirements(self, monkeypatch):
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_SERVER_URL", "http://localhost:1234")
|
||
|
|
monkeypatch.setenv("BLUEBUBBLES_PASSWORD", "secret")
|
||
|
|
from gateway.platforms.bluebubbles import check_bluebubbles_requirements
|
||
|
|
|
||
|
|
assert check_bluebubbles_requirements() is True
|
||
|
|
|
||
|
|
def test_format_message_strips_markdown(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
assert adapter.format_message("**Hello** `world`") == "Hello world"
|
||
|
|
|
||
|
|
def test_strip_markdown_headers(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
assert adapter.format_message("## Heading\ntext") == "Heading\ntext"
|
||
|
|
|
||
|
|
def test_strip_markdown_links(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
assert adapter.format_message("[click here](http://example.com)") == "click here"
|
||
|
|
|
||
|
|
def test_init_normalizes_webhook_path(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch, webhook_path="bluebubbles-webhook")
|
||
|
|
assert adapter.webhook_path == "/bluebubbles-webhook"
|
||
|
|
|
||
|
|
def test_init_preserves_leading_slash(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch, webhook_path="/my-hook")
|
||
|
|
assert adapter.webhook_path == "/my-hook"
|
||
|
|
|
||
|
|
def test_server_url_normalized(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch, server_url="http://localhost:1234/")
|
||
|
|
assert adapter.server_url == "http://localhost:1234"
|
||
|
|
|
||
|
|
def test_server_url_adds_scheme(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch, server_url="localhost:1234")
|
||
|
|
assert adapter.server_url == "http://localhost:1234"
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesWebhookParsing:
|
||
|
|
def test_webhook_prefers_chat_guid_over_message_guid(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
payload = {
|
||
|
|
"guid": "MESSAGE-GUID",
|
||
|
|
"chatGuid": "iMessage;-;user@example.com",
|
||
|
|
"chatIdentifier": "user@example.com",
|
||
|
|
}
|
||
|
|
record = adapter._extract_payload_record(payload) or {}
|
||
|
|
chat_guid = adapter._value(
|
||
|
|
record.get("chatGuid"),
|
||
|
|
payload.get("chatGuid"),
|
||
|
|
record.get("chat_guid"),
|
||
|
|
payload.get("chat_guid"),
|
||
|
|
payload.get("guid"),
|
||
|
|
)
|
||
|
|
assert chat_guid == "iMessage;-;user@example.com"
|
||
|
|
|
||
|
|
def test_webhook_can_fall_back_to_sender_when_chat_fields_missing(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
payload = {
|
||
|
|
"data": {
|
||
|
|
"guid": "MESSAGE-GUID",
|
||
|
|
"text": "hello",
|
||
|
|
"handle": {"address": "user@example.com"},
|
||
|
|
"isFromMe": False,
|
||
|
|
}
|
||
|
|
}
|
||
|
|
record = adapter._extract_payload_record(payload) or {}
|
||
|
|
chat_guid = adapter._value(
|
||
|
|
record.get("chatGuid"),
|
||
|
|
payload.get("chatGuid"),
|
||
|
|
record.get("chat_guid"),
|
||
|
|
payload.get("chat_guid"),
|
||
|
|
payload.get("guid"),
|
||
|
|
)
|
||
|
|
chat_identifier = adapter._value(
|
||
|
|
record.get("chatIdentifier"),
|
||
|
|
record.get("identifier"),
|
||
|
|
payload.get("chatIdentifier"),
|
||
|
|
payload.get("identifier"),
|
||
|
|
)
|
||
|
|
sender = (
|
||
|
|
adapter._value(
|
||
|
|
record.get("handle", {}).get("address")
|
||
|
|
if isinstance(record.get("handle"), dict)
|
||
|
|
else None,
|
||
|
|
record.get("sender"),
|
||
|
|
record.get("from"),
|
||
|
|
record.get("address"),
|
||
|
|
)
|
||
|
|
or chat_identifier
|
||
|
|
or chat_guid
|
||
|
|
)
|
||
|
|
if not (chat_guid or chat_identifier) and sender:
|
||
|
|
chat_identifier = sender
|
||
|
|
assert chat_identifier == "user@example.com"
|
||
|
|
|
||
|
|
def test_extract_payload_record_accepts_list_data(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
payload = {
|
||
|
|
"type": "new-message",
|
||
|
|
"data": [
|
||
|
|
{
|
||
|
|
"text": "hello",
|
||
|
|
"chatGuid": "iMessage;-;user@example.com",
|
||
|
|
"chatIdentifier": "user@example.com",
|
||
|
|
}
|
||
|
|
],
|
||
|
|
}
|
||
|
|
record = adapter._extract_payload_record(payload)
|
||
|
|
assert record == payload["data"][0]
|
||
|
|
|
||
|
|
def test_extract_payload_record_dict_data(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
payload = {"data": {"text": "hello", "chatGuid": "iMessage;-;+1234"}}
|
||
|
|
record = adapter._extract_payload_record(payload)
|
||
|
|
assert record["text"] == "hello"
|
||
|
|
|
||
|
|
def test_extract_payload_record_fallback_to_message(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
payload = {"message": {"text": "hello"}}
|
||
|
|
record = adapter._extract_payload_record(payload)
|
||
|
|
assert record["text"] == "hello"
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesGuidResolution:
|
||
|
|
def test_raw_guid_returned_as_is(self, monkeypatch):
|
||
|
|
"""If target already contains ';' it's a raw GUID — return unchanged."""
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
import asyncio
|
||
|
|
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._resolve_chat_guid("iMessage;-;user@example.com")
|
||
|
|
)
|
||
|
|
assert result == "iMessage;-;user@example.com"
|
||
|
|
|
||
|
|
def test_empty_target_returns_none(self, monkeypatch):
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
import asyncio
|
||
|
|
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._resolve_chat_guid("")
|
||
|
|
)
|
||
|
|
assert result is None
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesToolsetIntegration:
|
||
|
|
def test_toolset_exists(self):
|
||
|
|
from toolsets import TOOLSETS
|
||
|
|
|
||
|
|
assert "hermes-bluebubbles" in TOOLSETS
|
||
|
|
|
||
|
|
def test_toolset_in_gateway_composite(self):
|
||
|
|
from toolsets import TOOLSETS
|
||
|
|
|
||
|
|
gateway = TOOLSETS["hermes-gateway"]
|
||
|
|
assert "hermes-bluebubbles" in gateway["includes"]
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesPromptHint:
|
||
|
|
def test_platform_hint_exists(self):
|
||
|
|
from agent.prompt_builder import PLATFORM_HINTS
|
||
|
|
|
||
|
|
assert "bluebubbles" in PLATFORM_HINTS
|
||
|
|
hint = PLATFORM_HINTS["bluebubbles"]
|
||
|
|
assert "iMessage" in hint
|
||
|
|
assert "plain text" in hint
|
||
|
|
|
||
|
|
|
||
|
|
class TestBlueBubblesAttachmentDownload:
|
||
|
|
"""Verify _download_attachment routes to the correct cache helper."""
|
||
|
|
|
||
|
|
def test_download_image_uses_image_cache(self, monkeypatch):
|
||
|
|
"""Image MIME routes to cache_image_from_bytes."""
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
import asyncio
|
||
|
|
import httpx
|
||
|
|
|
||
|
|
# Mock the HTTP client response
|
||
|
|
class MockResponse:
|
||
|
|
status_code = 200
|
||
|
|
content = b"\x89PNG\r\n\x1a\n"
|
||
|
|
|
||
|
|
def raise_for_status(self):
|
||
|
|
pass
|
||
|
|
|
||
|
|
async def mock_get(*args, **kwargs):
|
||
|
|
return MockResponse()
|
||
|
|
|
||
|
|
adapter.client = type("MockClient", (), {"get": mock_get})()
|
||
|
|
|
||
|
|
cached_path = None
|
||
|
|
|
||
|
|
def mock_cache_image(data, ext):
|
||
|
|
nonlocal cached_path
|
||
|
|
cached_path = f"/tmp/test_image{ext}"
|
||
|
|
return cached_path
|
||
|
|
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"gateway.platforms.bluebubbles.cache_image_from_bytes",
|
||
|
|
mock_cache_image,
|
||
|
|
)
|
||
|
|
|
||
|
|
att_meta = {"mimeType": "image/png", "transferName": "photo.png"}
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._download_attachment("att-guid-123", att_meta)
|
||
|
|
)
|
||
|
|
assert result == "/tmp/test_image.png"
|
||
|
|
|
||
|
|
def test_download_audio_uses_audio_cache(self, monkeypatch):
|
||
|
|
"""Audio MIME routes to cache_audio_from_bytes."""
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
import asyncio
|
||
|
|
|
||
|
|
class MockResponse:
|
||
|
|
status_code = 200
|
||
|
|
content = b"fake-audio-data"
|
||
|
|
|
||
|
|
def raise_for_status(self):
|
||
|
|
pass
|
||
|
|
|
||
|
|
async def mock_get(*args, **kwargs):
|
||
|
|
return MockResponse()
|
||
|
|
|
||
|
|
adapter.client = type("MockClient", (), {"get": mock_get})()
|
||
|
|
|
||
|
|
cached_path = None
|
||
|
|
|
||
|
|
def mock_cache_audio(data, ext):
|
||
|
|
nonlocal cached_path
|
||
|
|
cached_path = f"/tmp/test_audio{ext}"
|
||
|
|
return cached_path
|
||
|
|
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"gateway.platforms.bluebubbles.cache_audio_from_bytes",
|
||
|
|
mock_cache_audio,
|
||
|
|
)
|
||
|
|
|
||
|
|
att_meta = {"mimeType": "audio/mpeg", "transferName": "voice.mp3"}
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._download_attachment("att-guid-456", att_meta)
|
||
|
|
)
|
||
|
|
assert result == "/tmp/test_audio.mp3"
|
||
|
|
|
||
|
|
def test_download_document_uses_document_cache(self, monkeypatch):
|
||
|
|
"""Non-image/audio MIME routes to cache_document_from_bytes."""
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
import asyncio
|
||
|
|
|
||
|
|
class MockResponse:
|
||
|
|
status_code = 200
|
||
|
|
content = b"fake-doc-data"
|
||
|
|
|
||
|
|
def raise_for_status(self):
|
||
|
|
pass
|
||
|
|
|
||
|
|
async def mock_get(*args, **kwargs):
|
||
|
|
return MockResponse()
|
||
|
|
|
||
|
|
adapter.client = type("MockClient", (), {"get": mock_get})()
|
||
|
|
|
||
|
|
cached_path = None
|
||
|
|
|
||
|
|
def mock_cache_doc(data, filename):
|
||
|
|
nonlocal cached_path
|
||
|
|
cached_path = f"/tmp/{filename}"
|
||
|
|
return cached_path
|
||
|
|
|
||
|
|
monkeypatch.setattr(
|
||
|
|
"gateway.platforms.bluebubbles.cache_document_from_bytes",
|
||
|
|
mock_cache_doc,
|
||
|
|
)
|
||
|
|
|
||
|
|
att_meta = {"mimeType": "application/pdf", "transferName": "report.pdf"}
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._download_attachment("att-guid-789", att_meta)
|
||
|
|
)
|
||
|
|
assert result == "/tmp/report.pdf"
|
||
|
|
|
||
|
|
def test_download_returns_none_without_client(self, monkeypatch):
|
||
|
|
"""No client → returns None gracefully."""
|
||
|
|
adapter = _make_adapter(monkeypatch)
|
||
|
|
adapter.client = None
|
||
|
|
import asyncio
|
||
|
|
|
||
|
|
result = asyncio.get_event_loop().run_until_complete(
|
||
|
|
adapter._download_attachment("att-guid", {"mimeType": "image/png"})
|
||
|
|
)
|
||
|
|
assert result is None
|