mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-04-25 00:51:20 +00:00
- Remove duplicate _setup_feishu() definition (old 3-line version left behind by cherry-pick — Python picked the new one but dead code remained) - Remove misleading 'Disable direct messages' DM option — the Feishu adapter has no DM policy mechanism, so 'disable' produced identical env vars to 'pairing'. Users who chose 'disable' would still see pairing prompts. Reduced to 3 options: pairing, allow-all, allowlist. - Fix test_probe_returns_bot_info_on_success and test_probe_returns_none_on_failure: patch FEISHU_AVAILABLE=True so probe_bot() takes the SDK path when lark_oapi is not installed
438 lines
17 KiB
Python
438 lines
17 KiB
Python
"""Tests for gateway.platforms.feishu — Feishu scan-to-create registration."""
|
|
|
|
import json
|
|
from unittest.mock import patch, MagicMock
|
|
import pytest
|
|
|
|
|
|
def _mock_urlopen(response_data, status=200):
|
|
"""Create a mock for urllib.request.urlopen that returns JSON response_data."""
|
|
mock_response = MagicMock()
|
|
mock_response.read.return_value = json.dumps(response_data).encode("utf-8")
|
|
mock_response.status = status
|
|
mock_response.__enter__ = lambda s: s
|
|
mock_response.__exit__ = MagicMock(return_value=False)
|
|
return mock_response
|
|
|
|
|
|
class TestPostRegistration:
|
|
"""Tests for the low-level HTTP helper."""
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_post_registration_returns_parsed_json(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _post_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({"nonce": "abc", "supported_auth_methods": ["client_secret"]})
|
|
result = _post_registration("https://accounts.feishu.cn", {"action": "init"})
|
|
assert result["nonce"] == "abc"
|
|
assert "client_secret" in result["supported_auth_methods"]
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_post_registration_sends_form_encoded_body(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _post_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({})
|
|
_post_registration("https://accounts.feishu.cn", {"action": "init", "key": "val"})
|
|
call_args = mock_urlopen_fn.call_args
|
|
request = call_args[0][0]
|
|
body = request.data.decode("utf-8")
|
|
assert "action=init" in body
|
|
assert "key=val" in body
|
|
assert request.get_header("Content-type") == "application/x-www-form-urlencoded"
|
|
|
|
|
|
class TestInitRegistration:
|
|
"""Tests for the init step."""
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_init_succeeds_when_client_secret_supported(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _init_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"nonce": "abc",
|
|
"supported_auth_methods": ["client_secret"],
|
|
})
|
|
_init_registration("feishu")
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_init_raises_when_client_secret_not_supported(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _init_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"nonce": "abc",
|
|
"supported_auth_methods": ["other_method"],
|
|
})
|
|
with pytest.raises(RuntimeError, match="client_secret"):
|
|
_init_registration("feishu")
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_init_uses_lark_url_for_lark_domain(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _init_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"nonce": "abc",
|
|
"supported_auth_methods": ["client_secret"],
|
|
})
|
|
_init_registration("lark")
|
|
call_args = mock_urlopen_fn.call_args
|
|
request = call_args[0][0]
|
|
assert "larksuite.com" in request.full_url
|
|
|
|
|
|
class TestBeginRegistration:
|
|
"""Tests for the begin step."""
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_begin_returns_device_code_and_qr_url(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _begin_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"device_code": "dc_123",
|
|
"verification_uri_complete": "https://accounts.feishu.cn/qr/abc",
|
|
"user_code": "ABCD-1234",
|
|
"interval": 5,
|
|
"expire_in": 600,
|
|
})
|
|
result = _begin_registration("feishu")
|
|
assert result["device_code"] == "dc_123"
|
|
assert "qr_url" in result
|
|
assert "accounts.feishu.cn" in result["qr_url"]
|
|
assert result["user_code"] == "ABCD-1234"
|
|
assert result["interval"] == 5
|
|
assert result["expire_in"] == 600
|
|
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_begin_sends_correct_archetype(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import _begin_registration
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"device_code": "dc_123",
|
|
"verification_uri_complete": "https://example.com/qr",
|
|
"user_code": "X",
|
|
"interval": 5,
|
|
"expire_in": 600,
|
|
})
|
|
_begin_registration("feishu")
|
|
request = mock_urlopen_fn.call_args[0][0]
|
|
body = request.data.decode("utf-8")
|
|
assert "archetype=PersonalAgent" in body
|
|
assert "auth_method=client_secret" in body
|
|
|
|
|
|
class TestPollRegistration:
|
|
"""Tests for the poll step."""
|
|
|
|
@patch("gateway.platforms.feishu.time")
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_poll_returns_credentials_on_success(self, mock_urlopen_fn, mock_time):
|
|
from gateway.platforms.feishu import _poll_registration
|
|
|
|
mock_time.time.side_effect = [0, 1]
|
|
mock_time.sleep = MagicMock()
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"client_id": "cli_app123",
|
|
"client_secret": "secret456",
|
|
"user_info": {"open_id": "ou_owner", "tenant_brand": "feishu"},
|
|
})
|
|
result = _poll_registration(
|
|
device_code="dc_123", interval=1, expire_in=60, domain="feishu"
|
|
)
|
|
assert result is not None
|
|
assert result["app_id"] == "cli_app123"
|
|
assert result["app_secret"] == "secret456"
|
|
assert result["domain"] == "feishu"
|
|
assert result["open_id"] == "ou_owner"
|
|
|
|
@patch("gateway.platforms.feishu.time")
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_poll_switches_domain_on_lark_tenant_brand(self, mock_urlopen_fn, mock_time):
|
|
from gateway.platforms.feishu import _poll_registration
|
|
|
|
mock_time.time.side_effect = [0, 1, 2]
|
|
mock_time.sleep = MagicMock()
|
|
|
|
pending_resp = _mock_urlopen({
|
|
"error": "authorization_pending",
|
|
"user_info": {"tenant_brand": "lark"},
|
|
})
|
|
success_resp = _mock_urlopen({
|
|
"client_id": "cli_lark",
|
|
"client_secret": "secret_lark",
|
|
"user_info": {"open_id": "ou_lark", "tenant_brand": "lark"},
|
|
})
|
|
mock_urlopen_fn.side_effect = [pending_resp, success_resp]
|
|
|
|
result = _poll_registration(
|
|
device_code="dc_123", interval=0, expire_in=60, domain="feishu"
|
|
)
|
|
assert result is not None
|
|
assert result["domain"] == "lark"
|
|
|
|
@patch("gateway.platforms.feishu.time")
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_poll_success_with_lark_brand_in_same_response(self, mock_urlopen_fn, mock_time):
|
|
"""Credentials and lark tenant_brand in one response must not be discarded."""
|
|
from gateway.platforms.feishu import _poll_registration
|
|
|
|
mock_time.time.side_effect = [0, 1]
|
|
mock_time.sleep = MagicMock()
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"client_id": "cli_lark_direct",
|
|
"client_secret": "secret_lark_direct",
|
|
"user_info": {"open_id": "ou_lark_direct", "tenant_brand": "lark"},
|
|
})
|
|
result = _poll_registration(
|
|
device_code="dc_123", interval=1, expire_in=60, domain="feishu"
|
|
)
|
|
assert result is not None
|
|
assert result["app_id"] == "cli_lark_direct"
|
|
assert result["domain"] == "lark"
|
|
assert result["open_id"] == "ou_lark_direct"
|
|
|
|
@patch("gateway.platforms.feishu.time")
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_poll_returns_none_on_access_denied(self, mock_urlopen_fn, mock_time):
|
|
from gateway.platforms.feishu import _poll_registration
|
|
|
|
mock_time.time.side_effect = [0, 1]
|
|
mock_time.sleep = MagicMock()
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"error": "access_denied",
|
|
})
|
|
result = _poll_registration(
|
|
device_code="dc_123", interval=1, expire_in=60, domain="feishu"
|
|
)
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu.time")
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_poll_returns_none_on_timeout(self, mock_urlopen_fn, mock_time):
|
|
from gateway.platforms.feishu import _poll_registration
|
|
|
|
mock_time.time.side_effect = [0, 999]
|
|
mock_time.sleep = MagicMock()
|
|
|
|
mock_urlopen_fn.return_value = _mock_urlopen({
|
|
"error": "authorization_pending",
|
|
})
|
|
result = _poll_registration(
|
|
device_code="dc_123", interval=1, expire_in=1, domain="feishu"
|
|
)
|
|
assert result is None
|
|
|
|
|
|
class TestRenderQr:
|
|
"""Tests for QR code terminal rendering."""
|
|
|
|
@patch("gateway.platforms.feishu._qrcode_mod", create=True)
|
|
def test_render_qr_returns_true_on_success(self, mock_qrcode_mod):
|
|
from gateway.platforms.feishu import _render_qr
|
|
|
|
mock_qr = MagicMock()
|
|
mock_qrcode_mod.QRCode.return_value = mock_qr
|
|
assert _render_qr("https://example.com/qr") is True
|
|
mock_qr.add_data.assert_called_once_with("https://example.com/qr")
|
|
mock_qr.make.assert_called_once_with(fit=True)
|
|
mock_qr.print_ascii.assert_called_once()
|
|
|
|
def test_render_qr_returns_false_when_qrcode_missing(self):
|
|
from gateway.platforms.feishu import _render_qr
|
|
|
|
with patch("gateway.platforms.feishu._qrcode_mod", None):
|
|
assert _render_qr("https://example.com/qr") is False
|
|
|
|
|
|
class TestProbeBot:
|
|
"""Tests for bot connectivity verification."""
|
|
|
|
@patch("gateway.platforms.feishu.FEISHU_AVAILABLE", True)
|
|
def test_probe_returns_bot_info_on_success(self):
|
|
from gateway.platforms.feishu import probe_bot
|
|
|
|
with patch("gateway.platforms.feishu._probe_bot_sdk") as mock_sdk:
|
|
mock_sdk.return_value = {"bot_name": "TestBot", "bot_open_id": "ou_bot123"}
|
|
result = probe_bot("cli_app", "secret", "feishu")
|
|
|
|
assert result is not None
|
|
assert result["bot_name"] == "TestBot"
|
|
assert result["bot_open_id"] == "ou_bot123"
|
|
|
|
@patch("gateway.platforms.feishu.FEISHU_AVAILABLE", True)
|
|
def test_probe_returns_none_on_failure(self):
|
|
from gateway.platforms.feishu import probe_bot
|
|
|
|
with patch("gateway.platforms.feishu._probe_bot_sdk") as mock_sdk:
|
|
mock_sdk.return_value = None
|
|
result = probe_bot("bad_id", "bad_secret", "feishu")
|
|
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu.FEISHU_AVAILABLE", False)
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_http_fallback_when_sdk_unavailable(self, mock_urlopen_fn):
|
|
"""Without lark_oapi, probe falls back to raw HTTP."""
|
|
from gateway.platforms.feishu import probe_bot
|
|
|
|
token_resp = _mock_urlopen({"code": 0, "tenant_access_token": "t-123"})
|
|
bot_resp = _mock_urlopen({"code": 0, "bot": {"bot_name": "HttpBot", "open_id": "ou_http"}})
|
|
mock_urlopen_fn.side_effect = [token_resp, bot_resp]
|
|
|
|
result = probe_bot("cli_app", "secret", "feishu")
|
|
assert result is not None
|
|
assert result["bot_name"] == "HttpBot"
|
|
|
|
@patch("gateway.platforms.feishu.FEISHU_AVAILABLE", False)
|
|
@patch("gateway.platforms.feishu.urlopen")
|
|
def test_http_fallback_returns_none_on_network_error(self, mock_urlopen_fn):
|
|
from gateway.platforms.feishu import probe_bot
|
|
from urllib.error import URLError
|
|
|
|
mock_urlopen_fn.side_effect = URLError("connection refused")
|
|
result = probe_bot("cli_app", "secret", "feishu")
|
|
assert result is None
|
|
|
|
|
|
class TestQrRegister:
|
|
"""Tests for the public qr_register entry point."""
|
|
|
|
@patch("gateway.platforms.feishu.probe_bot")
|
|
@patch("gateway.platforms.feishu._render_qr")
|
|
@patch("gateway.platforms.feishu._poll_registration")
|
|
@patch("gateway.platforms.feishu._begin_registration")
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_success_flow(
|
|
self, mock_init, mock_begin, mock_poll, mock_render, mock_probe
|
|
):
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_begin.return_value = {
|
|
"device_code": "dc_123",
|
|
"qr_url": "https://example.com/qr",
|
|
"user_code": "ABCD",
|
|
"interval": 1,
|
|
"expire_in": 60,
|
|
}
|
|
mock_poll.return_value = {
|
|
"app_id": "cli_app",
|
|
"app_secret": "secret",
|
|
"domain": "feishu",
|
|
"open_id": "ou_owner",
|
|
}
|
|
mock_probe.return_value = {"bot_name": "MyBot", "bot_open_id": "ou_bot"}
|
|
|
|
result = qr_register()
|
|
assert result is not None
|
|
assert result["app_id"] == "cli_app"
|
|
assert result["app_secret"] == "secret"
|
|
assert result["bot_name"] == "MyBot"
|
|
mock_init.assert_called_once()
|
|
mock_render.assert_called_once()
|
|
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_returns_none_on_init_failure(self, mock_init):
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_init.side_effect = RuntimeError("not supported")
|
|
result = qr_register()
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu._render_qr")
|
|
@patch("gateway.platforms.feishu._poll_registration")
|
|
@patch("gateway.platforms.feishu._begin_registration")
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_returns_none_on_poll_failure(
|
|
self, mock_init, mock_begin, mock_poll, mock_render
|
|
):
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_begin.return_value = {
|
|
"device_code": "dc_123",
|
|
"qr_url": "https://example.com/qr",
|
|
"user_code": "ABCD",
|
|
"interval": 1,
|
|
"expire_in": 60,
|
|
}
|
|
mock_poll.return_value = None
|
|
|
|
result = qr_register()
|
|
assert result is None
|
|
|
|
# -- Contract: expected errors → None, unexpected errors → propagate --
|
|
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_returns_none_on_network_error(self, mock_init):
|
|
"""URLError (network down) is an expected failure → None."""
|
|
from gateway.platforms.feishu import qr_register
|
|
from urllib.error import URLError
|
|
|
|
mock_init.side_effect = URLError("DNS resolution failed")
|
|
result = qr_register()
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_returns_none_on_json_error(self, mock_init):
|
|
"""Malformed server response is an expected failure → None."""
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_init.side_effect = json.JSONDecodeError("bad json", "", 0)
|
|
result = qr_register()
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_propagates_unexpected_errors(self, mock_init):
|
|
"""Bugs (e.g. AttributeError) must not be swallowed — they propagate."""
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_init.side_effect = AttributeError("some internal bug")
|
|
with pytest.raises(AttributeError, match="some internal bug"):
|
|
qr_register()
|
|
|
|
# -- Negative paths: partial/malformed server responses --
|
|
|
|
@patch("gateway.platforms.feishu._render_qr")
|
|
@patch("gateway.platforms.feishu._begin_registration")
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_returns_none_when_begin_missing_device_code(
|
|
self, mock_init, mock_begin, mock_render
|
|
):
|
|
"""Server returns begin response without device_code → RuntimeError → None."""
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_begin.side_effect = RuntimeError("Feishu registration did not return a device_code")
|
|
result = qr_register()
|
|
assert result is None
|
|
|
|
@patch("gateway.platforms.feishu.probe_bot")
|
|
@patch("gateway.platforms.feishu._render_qr")
|
|
@patch("gateway.platforms.feishu._poll_registration")
|
|
@patch("gateway.platforms.feishu._begin_registration")
|
|
@patch("gateway.platforms.feishu._init_registration")
|
|
def test_qr_register_succeeds_even_when_probe_fails(
|
|
self, mock_init, mock_begin, mock_poll, mock_render, mock_probe
|
|
):
|
|
"""Registration succeeds but probe fails → result with bot_name=None."""
|
|
from gateway.platforms.feishu import qr_register
|
|
|
|
mock_begin.return_value = {
|
|
"device_code": "dc_123",
|
|
"qr_url": "https://example.com/qr",
|
|
"user_code": "ABCD",
|
|
"interval": 1,
|
|
"expire_in": 60,
|
|
}
|
|
mock_poll.return_value = {
|
|
"app_id": "cli_app",
|
|
"app_secret": "secret",
|
|
"domain": "feishu",
|
|
"open_id": "ou_owner",
|
|
}
|
|
mock_probe.return_value = None # probe failed
|
|
|
|
result = qr_register()
|
|
assert result is not None
|
|
assert result["app_id"] == "cli_app"
|
|
assert result["bot_name"] is None
|
|
assert result["bot_open_id"] is None
|