mirror of
https://github.com/HKUDS/nanobot.git
synced 2026-05-03 16:25:53 +00:00
The PR stores ref freshness in the metadata sidecar, so the merged main test should assert updated_at there instead of in the refs payload. Made-with: Cursor
889 lines
29 KiB
Python
889 lines
29 KiB
Python
import json
|
||
import time
|
||
|
||
import pytest
|
||
|
||
# Check optional msteams dependencies before running tests
|
||
try:
|
||
from nanobot.channels import msteams
|
||
MSTEAMS_AVAILABLE = getattr(msteams, "MSTEAMS_AVAILABLE", False)
|
||
except ImportError:
|
||
MSTEAMS_AVAILABLE = False
|
||
|
||
if not MSTEAMS_AVAILABLE:
|
||
pytest.skip("MSTeams dependencies not installed (PyJWT, cryptography). Run: pip install nanobot-ai[msteams]", allow_module_level=True)
|
||
|
||
import jwt
|
||
from cryptography.hazmat.primitives.asymmetric import rsa
|
||
|
||
import nanobot.channels.msteams as msteams_module
|
||
from nanobot.bus.events import OutboundMessage
|
||
from nanobot.channels.msteams import ConversationRef, MSTeamsChannel
|
||
|
||
|
||
class DummyBus:
|
||
def __init__(self):
|
||
self.inbound = []
|
||
|
||
async def publish_inbound(self, msg):
|
||
self.inbound.append(msg)
|
||
|
||
|
||
class FakeResponse:
|
||
def __init__(self, payload=None, *, should_raise=False):
|
||
self._payload = payload or {}
|
||
self._should_raise = should_raise
|
||
|
||
def raise_for_status(self):
|
||
if self._should_raise:
|
||
raise RuntimeError("boom")
|
||
return None
|
||
|
||
def json(self):
|
||
return self._payload
|
||
|
||
|
||
class FakeHttpClient:
|
||
def __init__(self, payload=None, *, should_raise=False):
|
||
self.payload = payload or {"access_token": "tok", "expires_in": 3600}
|
||
self.should_raise = should_raise
|
||
self.calls = []
|
||
|
||
async def post(self, url, **kwargs):
|
||
self.calls.append((url, kwargs))
|
||
return FakeResponse(self.payload, should_raise=self.should_raise)
|
||
|
||
async def aclose(self):
|
||
pass
|
||
|
||
|
||
@pytest.fixture
|
||
def make_channel(tmp_path, monkeypatch):
|
||
monkeypatch.setattr("nanobot.channels.msteams.get_workspace_path", lambda: tmp_path)
|
||
|
||
def _make_channel(**config_overrides):
|
||
config = {
|
||
"enabled": True,
|
||
"appId": "app-id",
|
||
"appPassword": "secret",
|
||
"tenantId": "tenant-id",
|
||
"allowFrom": ["*"],
|
||
}
|
||
config.update(config_overrides)
|
||
return MSTeamsChannel(config, DummyBus())
|
||
|
||
return _make_channel
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_handle_activity_personal_message_publishes_and_stores_ref(make_channel, tmp_path):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"type": "message",
|
||
"id": "activity-1",
|
||
"text": "Hello from Teams",
|
||
"serviceUrl": "https://smba.trafficmanager.net/amer/",
|
||
"conversation": {
|
||
"id": "conv-123",
|
||
"conversationType": "personal",
|
||
},
|
||
"from": {
|
||
"id": "29:user-id",
|
||
"aadObjectId": "aad-user-1",
|
||
"name": "Bob",
|
||
},
|
||
"recipient": {
|
||
"id": "28:bot-id",
|
||
"name": "nanobot",
|
||
},
|
||
"channelData": {
|
||
"tenant": {"id": "tenant-id"},
|
||
},
|
||
}
|
||
|
||
await ch._handle_activity(activity)
|
||
|
||
assert len(ch.bus.inbound) == 1
|
||
msg = ch.bus.inbound[0]
|
||
assert msg.channel == "msteams"
|
||
assert msg.sender_id == "aad-user-1"
|
||
assert msg.chat_id == "conv-123"
|
||
assert msg.content == "Hello from Teams"
|
||
assert msg.metadata["msteams"]["conversation_id"] == "conv-123"
|
||
assert "conv-123" in ch._conversation_refs
|
||
|
||
saved = json.loads((tmp_path / "state" / "msteams_conversations.json").read_text(encoding="utf-8"))
|
||
assert saved["conv-123"]["conversation_id"] == "conv-123"
|
||
assert saved["conv-123"]["tenant_id"] == "tenant-id"
|
||
saved_meta = json.loads(
|
||
(tmp_path / "state" / msteams_module.MSTEAMS_REF_META_FILENAME).read_text(encoding="utf-8"),
|
||
)
|
||
assert float(saved_meta["conv-123"]["updated_at"]) > 0
|
||
|
||
|
||
def test_init_prunes_stale_and_unsupported_conversation_refs(make_channel, tmp_path, monkeypatch):
|
||
now = 1_800_000_000.0
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now)
|
||
|
||
state_dir = tmp_path / "state"
|
||
state_dir.mkdir(parents=True, exist_ok=True)
|
||
refs_path = state_dir / "msteams_conversations.json"
|
||
refs_meta_path = state_dir / msteams_module.MSTEAMS_REF_META_FILENAME
|
||
refs_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-valid": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-valid",
|
||
"conversation_type": "personal",
|
||
},
|
||
"conv-webchat": {
|
||
"service_url": "https://webchat.botframework.com/",
|
||
"conversation_id": "conv-webchat",
|
||
"conversation_type": "personal",
|
||
},
|
||
"conv-group": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-group",
|
||
"conversation_type": "channel",
|
||
},
|
||
"conv-stale": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-stale",
|
||
"conversation_type": "personal",
|
||
},
|
||
"conv-missing-ts": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-missing-ts",
|
||
"conversation_type": "personal",
|
||
},
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
refs_meta_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-valid": {"updated_at": now - 60},
|
||
"conv-webchat": {"updated_at": now - 60},
|
||
"conv-group": {"updated_at": now - 60},
|
||
"conv-stale": {"updated_at": now - msteams_module.MSTEAMS_REF_TTL_S - 1},
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
|
||
ch = make_channel()
|
||
|
||
assert set(ch._conversation_refs.keys()) == {"conv-valid", "conv-missing-ts"}
|
||
assert ch._conversation_refs["conv-valid"].conversation_id == "conv-valid"
|
||
assert ch._conversation_refs["conv-missing-ts"].conversation_id == "conv-missing-ts"
|
||
|
||
persisted = json.loads(refs_path.read_text(encoding="utf-8"))
|
||
assert set(persisted.keys()) == {"conv-valid", "conv-missing-ts"}
|
||
|
||
|
||
def test_save_prunes_unsupported_conversation_refs(make_channel, tmp_path, monkeypatch):
|
||
now = 1_800_000_000.0
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now)
|
||
|
||
ch = make_channel()
|
||
ch._conversation_refs = {
|
||
"conv-valid": ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-valid",
|
||
conversation_type="personal",
|
||
updated_at=now,
|
||
),
|
||
"conv-webchat": ConversationRef(
|
||
service_url="https://webchat.botframework.com/",
|
||
conversation_id="conv-webchat",
|
||
conversation_type="personal",
|
||
updated_at=now,
|
||
),
|
||
"conv-group": ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-group",
|
||
conversation_type="groupChat",
|
||
updated_at=now,
|
||
),
|
||
}
|
||
|
||
ch._save_refs()
|
||
|
||
assert set(ch._conversation_refs.keys()) == {"conv-valid"}
|
||
|
||
saved = json.loads((tmp_path / "state" / "msteams_conversations.json").read_text(encoding="utf-8"))
|
||
assert set(saved.keys()) == {"conv-valid"}
|
||
saved_meta = json.loads(
|
||
(tmp_path / "state" / msteams_module.MSTEAMS_REF_META_FILENAME).read_text(encoding="utf-8"),
|
||
)
|
||
assert set(saved_meta.keys()) == {"conv-valid"}
|
||
|
||
|
||
def test_init_respects_prune_toggle_flags(make_channel, tmp_path, monkeypatch):
|
||
now = 1_800_000_000.0
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now)
|
||
|
||
state_dir = tmp_path / "state"
|
||
state_dir.mkdir(parents=True, exist_ok=True)
|
||
refs_path = state_dir / "msteams_conversations.json"
|
||
refs_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-webchat": {
|
||
"service_url": "https://webchat.botframework.com/",
|
||
"conversation_id": "conv-webchat",
|
||
"conversation_type": "personal",
|
||
"updated_at": now - 60,
|
||
},
|
||
"conv-group": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-group",
|
||
"conversation_type": "channel",
|
||
"updated_at": now - 60,
|
||
},
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
|
||
ch = make_channel(pruneWebChatRefs=False, pruneNonPersonalRefs=False)
|
||
|
||
assert set(ch._conversation_refs.keys()) == {"conv-webchat", "conv-group"}
|
||
persisted = json.loads(refs_path.read_text(encoding="utf-8"))
|
||
assert set(persisted.keys()) == {"conv-webchat", "conv-group"}
|
||
|
||
|
||
def test_init_respects_custom_ref_ttl_days(make_channel, tmp_path, monkeypatch):
|
||
now = 1_800_000_000.0
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now)
|
||
|
||
state_dir = tmp_path / "state"
|
||
state_dir.mkdir(parents=True, exist_ok=True)
|
||
refs_path = state_dir / "msteams_conversations.json"
|
||
refs_meta_path = state_dir / msteams_module.MSTEAMS_REF_META_FILENAME
|
||
refs_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-fresh": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-fresh",
|
||
"conversation_type": "personal",
|
||
},
|
||
"conv-old": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-old",
|
||
"conversation_type": "personal",
|
||
},
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
refs_meta_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-fresh": {"updated_at": now - 12 * 60 * 60},
|
||
"conv-old": {"updated_at": now - 10 * 24 * 60 * 60},
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
|
||
ch = make_channel(refTtlDays=1)
|
||
|
||
assert set(ch._conversation_refs.keys()) == {"conv-fresh"}
|
||
persisted = json.loads(refs_path.read_text(encoding="utf-8"))
|
||
assert set(persisted.keys()) == {"conv-fresh"}
|
||
|
||
|
||
def test_init_without_meta_keeps_legacy_refs_alive(make_channel, tmp_path, monkeypatch):
|
||
now = 1_800_000_000.0
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now)
|
||
|
||
state_dir = tmp_path / "state"
|
||
state_dir.mkdir(parents=True, exist_ok=True)
|
||
refs_path = state_dir / "msteams_conversations.json"
|
||
refs_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-legacy": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-legacy",
|
||
"conversation_type": "personal",
|
||
}
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
|
||
ch = make_channel(refTtlDays=1)
|
||
|
||
assert set(ch._conversation_refs.keys()) == {"conv-legacy"}
|
||
assert ch._conversation_refs["conv-legacy"].updated_at == now
|
||
assert not (state_dir / msteams_module.MSTEAMS_REF_META_FILENAME).exists()
|
||
|
||
|
||
def test_save_uses_atomic_replace_and_keeps_existing_file_on_replace_error(make_channel, tmp_path, monkeypatch):
|
||
ch = make_channel()
|
||
refs_path = tmp_path / "state" / "msteams_conversations.json"
|
||
refs_path.write_text(
|
||
json.dumps(
|
||
{
|
||
"conv-old": {
|
||
"service_url": "https://smba.trafficmanager.net/amer/",
|
||
"conversation_id": "conv-old",
|
||
"conversation_type": "personal",
|
||
"updated_at": 1_700_000_000.0,
|
||
}
|
||
},
|
||
indent=2,
|
||
),
|
||
encoding="utf-8",
|
||
)
|
||
|
||
ch._conversation_refs = {
|
||
"conv-new": ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-new",
|
||
conversation_type="personal",
|
||
updated_at=1_800_000_000.0,
|
||
)
|
||
}
|
||
|
||
def _raise_replace(_src, _dst):
|
||
raise OSError("replace failed")
|
||
|
||
monkeypatch.setattr(msteams_module.os, "replace", _raise_replace)
|
||
ch._save_refs()
|
||
|
||
persisted = json.loads(refs_path.read_text(encoding="utf-8"))
|
||
assert set(persisted.keys()) == {"conv-old"}
|
||
tmp_files = list((tmp_path / "state").glob("msteams_conversations.json.*.tmp"))
|
||
assert tmp_files == []
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_handle_activity_ignores_group_messages(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"type": "message",
|
||
"id": "activity-2",
|
||
"text": "Hello group",
|
||
"serviceUrl": "https://smba.trafficmanager.net/amer/",
|
||
"conversation": {
|
||
"id": "conv-group",
|
||
"conversationType": "channel",
|
||
},
|
||
"from": {
|
||
"id": "29:user-id",
|
||
"aadObjectId": "aad-user-1",
|
||
"name": "Bob",
|
||
},
|
||
"recipient": {
|
||
"id": "28:bot-id",
|
||
"name": "nanobot",
|
||
},
|
||
}
|
||
|
||
await ch._handle_activity(activity)
|
||
|
||
assert ch.bus.inbound == []
|
||
assert ch._conversation_refs == {}
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_handle_activity_denied_sender_does_not_store_ref(make_channel, tmp_path):
|
||
ch = make_channel(allowFrom=["allowed-user"])
|
||
|
||
activity = {
|
||
"type": "message",
|
||
"id": "activity-denied",
|
||
"text": "Hello from denied user",
|
||
"serviceUrl": "https://smba.trafficmanager.net/amer/",
|
||
"conversation": {
|
||
"id": "conv-denied",
|
||
"conversationType": "personal",
|
||
},
|
||
"from": {
|
||
"id": "29:user-id",
|
||
"aadObjectId": "aad-user-1",
|
||
"name": "Bob",
|
||
},
|
||
"recipient": {
|
||
"id": "28:bot-id",
|
||
"name": "nanobot",
|
||
},
|
||
"channelData": {
|
||
"tenant": {"id": "tenant-id"},
|
||
},
|
||
}
|
||
|
||
await ch._handle_activity(activity)
|
||
|
||
assert ch.bus.inbound == []
|
||
assert ch._conversation_refs == {}
|
||
assert not (tmp_path / "state" / "msteams_conversations.json").exists()
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_handle_activity_mention_only_uses_default_response(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"type": "message",
|
||
"id": "activity-3",
|
||
"text": "<at>Nanobot</at>",
|
||
"serviceUrl": "https://smba.trafficmanager.net/amer/",
|
||
"conversation": {
|
||
"id": "conv-empty",
|
||
"conversationType": "personal",
|
||
},
|
||
"from": {
|
||
"id": "29:user-id",
|
||
"aadObjectId": "aad-user-1",
|
||
"name": "Bob",
|
||
},
|
||
"recipient": {
|
||
"id": "28:bot-id",
|
||
"name": "nanobot",
|
||
},
|
||
}
|
||
|
||
await ch._handle_activity(activity)
|
||
|
||
assert len(ch.bus.inbound) == 1
|
||
assert ch.bus.inbound[0].content == "Hi — what can I help with?"
|
||
assert "conv-empty" in ch._conversation_refs
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_handle_activity_mention_only_ignores_when_response_disabled(make_channel):
|
||
ch = make_channel(mentionOnlyResponse=" ")
|
||
|
||
activity = {
|
||
"type": "message",
|
||
"id": "activity-4",
|
||
"text": "<at>Nanobot</at>",
|
||
"serviceUrl": "https://smba.trafficmanager.net/amer/",
|
||
"conversation": {
|
||
"id": "conv-empty-disabled",
|
||
"conversationType": "personal",
|
||
},
|
||
"from": {
|
||
"id": "29:user-id",
|
||
"aadObjectId": "aad-user-1",
|
||
"name": "Bob",
|
||
},
|
||
"recipient": {
|
||
"id": "28:bot-id",
|
||
"name": "nanobot",
|
||
},
|
||
}
|
||
|
||
await ch._handle_activity(activity)
|
||
|
||
assert ch.bus.inbound == []
|
||
assert ch._conversation_refs == {}
|
||
|
||
|
||
def test_strip_possible_bot_mention_removes_generic_at_tags(make_channel):
|
||
ch = make_channel()
|
||
|
||
assert ch._strip_possible_bot_mention("<at>Nanobot</at> hello") == "hello"
|
||
assert ch._strip_possible_bot_mention("hi <at>Some Bot</at> there") == "hi there"
|
||
|
||
|
||
def test_sanitize_inbound_text_keeps_normal_inline_message(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": "<at>Nanobot</at> normal inline message",
|
||
"channelData": {},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == "normal inline message"
|
||
|
||
|
||
def test_sanitize_inbound_text_normalizes_nbsp_entities(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": "Hello from Teams",
|
||
"channelData": {},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == "Hello from Teams"
|
||
|
||
|
||
def test_sanitize_inbound_text_normalizes_reply_wrapper_without_reply_metadata(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": "Reply wrapper \r\nQuoted prior message\r\n\r\nThis is a reply with quote test",
|
||
"channelData": {},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == (
|
||
"User is replying to: Quoted prior message\n"
|
||
"User reply: This is a reply with quote test"
|
||
)
|
||
|
||
|
||
def test_sanitize_inbound_text_structures_reply_quote_prefix(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": "Replying to Bob Smith\nactual reply text",
|
||
"replyToId": "parent-activity",
|
||
"channelData": {"messageType": "reply"},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == "User is replying to: Bob Smith\nUser reply: actual reply text"
|
||
|
||
|
||
def test_sanitize_inbound_text_structures_live_reply_wrapper_shape(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": "Reply wrapper Got it. I’ll watch for the exact text reply with quote test and then inspect that turn specifically. Reply with quote test",
|
||
"replyToId": "parent-activity",
|
||
"channelData": {"messageType": "reply"},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == (
|
||
"User is replying to: Got it. I’ll watch for the exact text reply with quote test and then inspect that turn specifically.\n"
|
||
"User reply: Reply with quote test"
|
||
)
|
||
|
||
|
||
def test_normalize_teams_reply_quote_leaves_plain_text_test_phrase_untouched(make_channel):
|
||
ch = make_channel()
|
||
|
||
text = "Normal message ending with Reply with quote test"
|
||
|
||
assert ch._normalize_teams_reply_quote(text) == text
|
||
|
||
|
||
def test_sanitize_inbound_text_structures_multiline_reply_wrapper_shape(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": (
|
||
"Reply wrapper\r\n"
|
||
"Understood — then the restart already happened, and the new Teams quote normalization should now be live. "
|
||
"Next best step: • send one more real reply-with-quote message in Teams • I&rsquo…\r\n"
|
||
"\r\n"
|
||
"This is a reply with quote"
|
||
),
|
||
"replyToId": "parent-activity",
|
||
"channelData": {"messageType": "reply"},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == (
|
||
"User is replying to: Understood — then the restart already happened, and the new Teams quote normalization should now be live. "
|
||
"Next best step: • send one more real reply-with-quote message in Teams • I’…\n"
|
||
"User reply: This is a reply with quote"
|
||
)
|
||
|
||
|
||
def test_sanitize_inbound_text_structures_exact_live_crlf_reply_wrapper_shape(make_channel):
|
||
ch = make_channel()
|
||
|
||
activity = {
|
||
"text": (
|
||
"Reply wrapper \r\n"
|
||
"Please send one real reply-with-quote message in Teams. That single test should be enough now: "
|
||
"• I’ll check the new MSTeams sanitized inbound text ... log • and compare it to the prompt…\r\n"
|
||
"\r\n"
|
||
"This is a reply with quote test"
|
||
),
|
||
"replyToId": "parent-activity",
|
||
"channelData": {"messageType": "reply"},
|
||
}
|
||
|
||
assert ch._sanitize_inbound_text(activity) == (
|
||
"User is replying to: Please send one real reply-with-quote message in Teams. That single test should be enough now: "
|
||
"• I’ll check the new MSTeams sanitized inbound text ... log • and compare it to the prompt…\n"
|
||
"User reply: This is a reply with quote test"
|
||
)
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_get_access_token_uses_configured_tenant(make_channel):
|
||
ch = make_channel(tenantId="tenant-123")
|
||
fake_http = FakeHttpClient()
|
||
ch._http = fake_http
|
||
|
||
token = await ch._get_access_token()
|
||
|
||
assert token == "tok"
|
||
assert len(fake_http.calls) == 1
|
||
url, kwargs = fake_http.calls[0]
|
||
assert url == "https://login.microsoftonline.com/tenant-123/oauth2/v2.0/token"
|
||
assert kwargs["data"]["client_id"] == "app-id"
|
||
assert kwargs["data"]["client_secret"] == "secret"
|
||
assert kwargs["data"]["scope"] == "https://api.botframework.com/.default"
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_posts_to_conversation_with_reply_to_id_when_reply_in_thread_enabled(make_channel):
|
||
ch = make_channel(replyInThread=True)
|
||
fake_http = FakeHttpClient()
|
||
ch._http = fake_http
|
||
ch._token = "tok"
|
||
ch._token_expires_at = 9999999999
|
||
ch._conversation_refs["conv-123"] = ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-123",
|
||
activity_id="activity-1",
|
||
)
|
||
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="conv-123", content="Reply text"))
|
||
|
||
assert len(fake_http.calls) == 1
|
||
url, kwargs = fake_http.calls[0]
|
||
assert url == "https://smba.trafficmanager.net/amer/v3/conversations/conv-123/activities"
|
||
assert kwargs["headers"]["Authorization"] == "Bearer tok"
|
||
assert kwargs["json"]["text"] == "Reply text"
|
||
assert kwargs["json"]["replyToId"] == "activity-1"
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_success_refreshes_updated_at_and_persists_meta(make_channel, tmp_path, monkeypatch):
|
||
now = {"value": 1_800_000_000.0}
|
||
monkeypatch.setattr(msteams_module.time, "time", lambda: now["value"])
|
||
|
||
ch = make_channel(refTouchIntervalS=0)
|
||
fake_http = FakeHttpClient()
|
||
ch._http = fake_http
|
||
ch._token = "tok"
|
||
ch._token_expires_at = 9_999_999_999
|
||
ch._conversation_refs["conv-123"] = ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-123",
|
||
activity_id="activity-1",
|
||
updated_at=now["value"] - 100,
|
||
)
|
||
|
||
now["value"] += 5
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="conv-123", content="Reply text"))
|
||
|
||
assert ch._conversation_refs["conv-123"].updated_at == now["value"]
|
||
saved_meta = json.loads(
|
||
(tmp_path / "state" / msteams_module.MSTEAMS_REF_META_FILENAME).read_text(encoding="utf-8"),
|
||
)
|
||
assert saved_meta["conv-123"]["updated_at"] == now["value"]
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_posts_to_conversation_when_thread_reply_disabled(make_channel):
|
||
ch = make_channel(replyInThread=False)
|
||
fake_http = FakeHttpClient()
|
||
ch._http = fake_http
|
||
ch._token = "tok"
|
||
ch._token_expires_at = 9999999999
|
||
ch._conversation_refs["conv-123"] = ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-123",
|
||
activity_id="activity-1",
|
||
)
|
||
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="conv-123", content="Reply text"))
|
||
|
||
assert len(fake_http.calls) == 1
|
||
url, kwargs = fake_http.calls[0]
|
||
assert url == "https://smba.trafficmanager.net/amer/v3/conversations/conv-123/activities"
|
||
assert kwargs["headers"]["Authorization"] == "Bearer tok"
|
||
assert kwargs["json"]["text"] == "Reply text"
|
||
assert "replyToId" not in kwargs["json"]
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_posts_to_conversation_when_thread_reply_enabled_but_no_activity_id(make_channel):
|
||
ch = make_channel(replyInThread=True)
|
||
fake_http = FakeHttpClient()
|
||
ch._http = fake_http
|
||
ch._token = "tok"
|
||
ch._token_expires_at = 9999999999
|
||
ch._conversation_refs["conv-123"] = ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-123",
|
||
activity_id=None,
|
||
)
|
||
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="conv-123", content="Reply text"))
|
||
|
||
assert len(fake_http.calls) == 1
|
||
url, kwargs = fake_http.calls[0]
|
||
assert url == "https://smba.trafficmanager.net/amer/v3/conversations/conv-123/activities"
|
||
assert kwargs["headers"]["Authorization"] == "Bearer tok"
|
||
assert kwargs["json"]["text"] == "Reply text"
|
||
assert "replyToId" not in kwargs["json"]
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_raises_when_conversation_ref_missing(make_channel):
|
||
ch = make_channel()
|
||
ch._http = FakeHttpClient()
|
||
|
||
with pytest.raises(RuntimeError, match="conversation ref not found"):
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="missing", content="Reply text"))
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_send_raises_delivery_failures_for_retry(make_channel):
|
||
ch = make_channel()
|
||
ch._http = FakeHttpClient(should_raise=True)
|
||
ch._token = "tok"
|
||
ch._token_expires_at = 9999999999
|
||
ch._conversation_refs["conv-123"] = ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="conv-123",
|
||
activity_id="activity-1",
|
||
)
|
||
|
||
with pytest.raises(RuntimeError, match="boom"):
|
||
await ch.send(OutboundMessage(channel="msteams", chat_id="conv-123", content="Reply text"))
|
||
|
||
|
||
def _make_test_rsa_jwk(kid: str = "test-kid"):
|
||
private_key = rsa.generate_private_key(public_exponent=65537, key_size=2048)
|
||
public_key = private_key.public_key()
|
||
jwk = json.loads(jwt.algorithms.RSAAlgorithm.to_jwk(public_key))
|
||
jwk["kid"] = kid
|
||
jwk["use"] = "sig"
|
||
jwk["kty"] = "RSA"
|
||
jwk["alg"] = "RS256"
|
||
return private_key, jwk
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_validate_inbound_auth_accepts_observed_botframework_shape(make_channel):
|
||
ch = make_channel(validateInboundAuth=True)
|
||
|
||
private_key, jwk = _make_test_rsa_jwk()
|
||
ch._botframework_jwks = {"keys": [jwk]}
|
||
ch._botframework_jwks_expires_at = 9999999999
|
||
|
||
service_url = "https://smba.trafficmanager.net/amer/tenant/"
|
||
token = jwt.encode(
|
||
{
|
||
"iss": "https://api.botframework.com",
|
||
"aud": "app-id",
|
||
"serviceurl": service_url,
|
||
"nbf": 1700000000,
|
||
"exp": 4100000000,
|
||
},
|
||
private_key,
|
||
algorithm="RS256",
|
||
headers={"kid": jwk["kid"]},
|
||
)
|
||
|
||
await ch._validate_inbound_auth(
|
||
f"Bearer {token}",
|
||
{"serviceUrl": service_url},
|
||
)
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_validate_inbound_auth_rejects_service_url_mismatch(make_channel):
|
||
ch = make_channel(validateInboundAuth=True)
|
||
|
||
private_key, jwk = _make_test_rsa_jwk()
|
||
ch._botframework_jwks = {"keys": [jwk]}
|
||
ch._botframework_jwks_expires_at = 9999999999
|
||
|
||
token = jwt.encode(
|
||
{
|
||
"iss": "https://api.botframework.com",
|
||
"aud": "app-id",
|
||
"serviceurl": "https://smba.trafficmanager.net/amer/tenant-a/",
|
||
"nbf": 1700000000,
|
||
"exp": 4100000000,
|
||
},
|
||
private_key,
|
||
algorithm="RS256",
|
||
headers={"kid": jwk["kid"]},
|
||
)
|
||
|
||
with pytest.raises(ValueError, match="serviceUrl claim mismatch"):
|
||
await ch._validate_inbound_auth(
|
||
f"Bearer {token}",
|
||
{"serviceUrl": "https://smba.trafficmanager.net/amer/tenant-b/"},
|
||
)
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_validate_inbound_auth_rejects_missing_bearer_token(make_channel):
|
||
ch = make_channel(validateInboundAuth=True)
|
||
|
||
with pytest.raises(ValueError, match="missing bearer token"):
|
||
await ch._validate_inbound_auth("", {"serviceUrl": "https://smba.trafficmanager.net/amer/tenant/"})
|
||
|
||
|
||
@pytest.mark.asyncio
|
||
async def test_start_logs_install_hint_when_pyjwt_missing(make_channel, monkeypatch):
|
||
ch = make_channel()
|
||
errors = []
|
||
monkeypatch.setattr(msteams_module, "MSTEAMS_AVAILABLE", False)
|
||
monkeypatch.setattr(msteams_module.logger, "error", lambda message, *args: errors.append(message.format(*args)))
|
||
|
||
await ch.start()
|
||
|
||
assert errors == ["PyJWT not installed. Run: pip install nanobot-ai[msteams]"]
|
||
|
||
|
||
def test_save_refs_prunes_webchat_and_stale_refs(make_channel):
|
||
ch = make_channel()
|
||
now = time.time()
|
||
ch._conversation_refs = {
|
||
"teams-good": ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="teams-good",
|
||
conversation_type="personal",
|
||
updated_at=now,
|
||
),
|
||
"webchat-bad": ConversationRef(
|
||
service_url="https://webchat.botframework.com/",
|
||
conversation_id="webchat-bad",
|
||
conversation_type=None,
|
||
updated_at=now,
|
||
),
|
||
"teams-stale": ConversationRef(
|
||
service_url="https://smba.trafficmanager.net/amer/",
|
||
conversation_id="teams-stale",
|
||
conversation_type="personal",
|
||
updated_at=now - (31 * 24 * 60 * 60),
|
||
),
|
||
}
|
||
|
||
ch._save_refs()
|
||
|
||
assert set(ch._conversation_refs) == {"teams-good"}
|
||
saved = json.loads(ch._refs_path.read_text(encoding="utf-8"))
|
||
assert set(saved) == {"teams-good"}
|
||
saved_meta = json.loads(ch._refs_meta_path.read_text(encoding="utf-8"))
|
||
assert saved_meta["teams-good"]["updated_at"] == pytest.approx(now)
|
||
|
||
|
||
def test_msteams_default_config_includes_restart_notify_fields():
|
||
cfg = MSTeamsChannel.default_config()
|
||
|
||
assert cfg["validateInboundAuth"] is True
|
||
assert cfg["refTtlDays"] == msteams_module.MSTEAMS_REF_TTL_DAYS
|
||
assert cfg["pruneWebChatRefs"] is True
|
||
assert cfg["pruneNonPersonalRefs"] is True
|
||
assert cfg["refTouchIntervalS"] == msteams_module.MSTEAMS_REF_TOUCH_INTERVAL_S
|
||
assert "restartNotifyEnabled" not in cfg
|
||
assert "restartNotifyPreMessage" not in cfg
|
||
assert "restartNotifyPostMessage" not in cfg
|