mirror of
https://github.com/jxxghp/MoviePilot.git
synced 2026-06-04 07:26:46 +00:00
289 lines
12 KiB
Python
289 lines
12 KiB
Python
import asyncio
|
||
import unittest
|
||
from types import ModuleType
|
||
from unittest.mock import AsyncMock, patch
|
||
|
||
from app.testing import stub_modules
|
||
|
||
|
||
def _stub(name: str, **attrs) -> tuple:
|
||
"""构造带指定属性的占位模块,返回 ``(模块名, 模块)`` 供 :func:`stub_modules` 使用。"""
|
||
module = ModuleType(name)
|
||
for key, value in attrs.items():
|
||
setattr(module, key, value)
|
||
return name, module
|
||
|
||
|
||
class _Dummy:
|
||
def __init__(self, *args, **kwargs):
|
||
pass
|
||
|
||
def __getattr__(self, _name):
|
||
return lambda *args, **kwargs: None
|
||
|
||
|
||
class _DummyError(Exception):
|
||
def __init__(self, message="", duration_ms=None):
|
||
super().__init__(message)
|
||
self.duration_ms = duration_ms
|
||
|
||
|
||
# 在 import 期用占位模块替换重依赖/外部模块,import 完由 stub_modules 精确还原,避免污染其它用例
|
||
_STUB_MODULES = dict([
|
||
_stub("pillow_avif"),
|
||
_stub("aiofiles"),
|
||
_stub("psutil"),
|
||
_stub("app.helper.sites", SitesHelper=_Dummy),
|
||
_stub("app.chain.mediaserver", MediaServerChain=_Dummy),
|
||
_stub("app.chain.search", SearchChain=_Dummy),
|
||
_stub("app.chain.system", SystemChain=_Dummy),
|
||
_stub("app.agent.llm", LLMHelper=_Dummy, LLMProviderManager=_Dummy,
|
||
LLMTestError=_DummyError, LLMTestTimeout=_DummyError,
|
||
render_auth_result_html=lambda success, message: message),
|
||
_stub("app.core.event", eventmanager=_Dummy(), Event=_Dummy, EventManager=_Dummy),
|
||
_stub("app.core.metainfo", MetaInfo=_Dummy),
|
||
_stub("app.core.module", ModuleManager=_Dummy),
|
||
_stub("app.core.security", verify_apitoken=_Dummy, verify_resource_token=_Dummy, verify_token=_Dummy),
|
||
_stub("app.db.models", User=_Dummy),
|
||
_stub("app.db.systemconfig_oper", SystemConfigOper=_Dummy),
|
||
_stub("app.db.user_oper", get_current_active_superuser=_Dummy,
|
||
get_current_active_superuser_async=_Dummy, get_current_active_user_async=_Dummy),
|
||
_stub("app.helper.llm", LLMHelper=_Dummy, LLMTestError=_DummyError, LLMTestTimeout=_DummyError),
|
||
_stub("app.helper.mediaserver", MediaServerHelper=_Dummy),
|
||
_stub("app.helper.message", MessageHelper=_Dummy),
|
||
_stub("app.helper.progress", ProgressHelper=_Dummy),
|
||
_stub("app.helper.rule", RuleHelper=_Dummy),
|
||
_stub("app.helper.server", MoviePilotServerHelper=_Dummy),
|
||
_stub("app.helper.system", SystemHelper=_Dummy),
|
||
_stub("app.helper.image", ImageHelper=_Dummy),
|
||
_stub("app.scheduler", Scheduler=_Dummy),
|
||
_stub("app.log", logger=_Dummy(), log_settings=_Dummy(),
|
||
LogConfigModel=type("LogConfigModel", (), {})),
|
||
_stub("app.utils.crypto", HashUtils=_Dummy),
|
||
_stub("app.utils.http", RequestUtils=_Dummy, AsyncRequestUtils=_Dummy),
|
||
_stub("version", APP_VERSION="test"),
|
||
])
|
||
|
||
with stub_modules(_STUB_MODULES):
|
||
from app.api.endpoints import llm as system_endpoint
|
||
|
||
|
||
class LlmTestEndpointTest(unittest.TestCase):
|
||
def test_llm_test_requires_ai_agent_enabled(self):
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", False):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertEqual(resp.message, "请先启用智能助手")
|
||
|
||
def test_llm_test_requires_api_key(self):
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", None
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", "deepseek-chat"):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertEqual(resp.message, "请先配置 LLM API Key")
|
||
self.assertEqual(resp.data["model"], "deepseek-chat")
|
||
|
||
def test_llm_test_requires_model(self):
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-test"
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", ""):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertEqual(resp.message, "请先配置 LLM 模型")
|
||
|
||
def test_llm_test_returns_successful_reply_preview(self):
|
||
llm_test_mock = AsyncMock(
|
||
return_value={
|
||
"provider": "deepseek",
|
||
"model": "deepseek-chat",
|
||
"duration_ms": 321,
|
||
"reply_preview": "OK",
|
||
}
|
||
)
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_PROVIDER", "deepseek"
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", "deepseek-chat"), patch.object(
|
||
system_endpoint.settings, "LLM_THINKING_LEVEL", "max"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-test"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_BASE_URL", "https://api.deepseek.com"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_BASE_URL_PRESET", "deepseek-default"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_USER_AGENT", "MoviePilot-Test/1.0"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_USE_PROXY", True
|
||
), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
llm_test_mock,
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
llm_test_mock.assert_awaited_once_with(
|
||
provider="deepseek",
|
||
model="deepseek-chat",
|
||
thinking_level="max",
|
||
api_key="sk-test",
|
||
base_url="https://api.deepseek.com",
|
||
base_url_preset="deepseek-default",
|
||
user_agent="MoviePilot-Test/1.0",
|
||
use_proxy=True,
|
||
)
|
||
self.assertTrue(resp.success)
|
||
self.assertEqual(resp.data["provider"], "deepseek")
|
||
self.assertEqual(resp.data["model"], "deepseek-chat")
|
||
self.assertEqual(resp.data["duration_ms"], 321)
|
||
self.assertEqual(resp.data["reply_preview"], "OK")
|
||
|
||
def test_llm_test_prefers_request_payload_over_saved_settings(self):
|
||
llm_test_mock = AsyncMock(
|
||
return_value={
|
||
"provider": "openai",
|
||
"model": "gpt-4.1-mini",
|
||
"duration_ms": 123,
|
||
"reply_preview": "OK",
|
||
}
|
||
)
|
||
payload = system_endpoint.LlmTestRequest(
|
||
enabled=True,
|
||
provider="openai",
|
||
model="gpt-4.1-mini",
|
||
thinking_level="high",
|
||
api_key="sk-live",
|
||
base_url="https://example.com/v1",
|
||
base_url_preset="openai-default",
|
||
user_agent="MoviePilot-Custom/1.0",
|
||
use_proxy=False,
|
||
)
|
||
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", False), patch.object(
|
||
system_endpoint.settings, "LLM_PROVIDER", "deepseek"
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", "deepseek-chat"), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-saved"
|
||
), patch.object(
|
||
system_endpoint.settings, "LLM_BASE_URL", "https://api.deepseek.com"
|
||
), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
llm_test_mock,
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(payload=payload, _="token"))
|
||
|
||
llm_test_mock.assert_awaited_once_with(
|
||
provider="openai",
|
||
model="gpt-4.1-mini",
|
||
thinking_level="high",
|
||
api_key="sk-live",
|
||
base_url="https://example.com/v1",
|
||
base_url_preset="openai-default",
|
||
user_agent="MoviePilot-Custom/1.0",
|
||
use_proxy=False,
|
||
)
|
||
self.assertTrue(resp.success)
|
||
self.assertEqual(resp.data["provider"], "openai")
|
||
self.assertEqual(resp.data["model"], "gpt-4.1-mini")
|
||
|
||
def test_llm_test_supports_legacy_thinking_payload(self):
|
||
llm_test_mock = AsyncMock(
|
||
return_value={
|
||
"provider": "deepseek",
|
||
"model": "deepseek-v4-pro",
|
||
"duration_ms": 123,
|
||
"reply_preview": "OK",
|
||
}
|
||
)
|
||
payload = system_endpoint.LlmTestRequest(
|
||
enabled=True,
|
||
provider="deepseek",
|
||
model="deepseek-v4-pro",
|
||
api_key="sk-live",
|
||
base_url="https://api.deepseek.com",
|
||
base_url_preset="deepseek-default",
|
||
user_agent=None,
|
||
use_proxy=None,
|
||
)
|
||
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", False), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
llm_test_mock,
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(payload=payload, _="token"))
|
||
|
||
llm_test_mock.assert_awaited_once_with(
|
||
provider="deepseek",
|
||
model="deepseek-v4-pro",
|
||
thinking_level=None,
|
||
api_key="sk-live",
|
||
base_url="https://api.deepseek.com",
|
||
base_url_preset="deepseek-default",
|
||
user_agent=None,
|
||
use_proxy=None,
|
||
)
|
||
self.assertTrue(resp.success)
|
||
|
||
def test_llm_test_rejects_empty_reply(self):
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_PROVIDER", "deepseek"
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", "deepseek-chat"), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-test"
|
||
), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
AsyncMock(return_value={"provider": "deepseek", "model": "deepseek-chat", "duration_ms": 12}),
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertEqual(resp.message, "模型响应为空")
|
||
self.assertEqual(resp.data["duration_ms"], 12)
|
||
|
||
def test_llm_test_maps_timeout_error(self):
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_PROVIDER", "deepseek"
|
||
), patch.object(system_endpoint.settings, "LLM_MODEL", "deepseek-chat"), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-test"
|
||
), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
AsyncMock(side_effect=TimeoutError("request timed out")),
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertEqual(resp.message, "LLM 调用超时")
|
||
|
||
def test_llm_test_sanitizes_error_message(self):
|
||
raw_error = (
|
||
"request failed api_key=sk-secret "
|
||
"Authorization: Bearer sk-secret "
|
||
"base error sk-secret"
|
||
)
|
||
with patch.object(system_endpoint.settings, "AI_AGENT_ENABLE", True), patch.object(
|
||
system_endpoint.settings, "LLM_API_KEY", "sk-secret"
|
||
), patch.object(system_endpoint.settings, "LLM_PROVIDER", "deepseek"), patch.object(
|
||
system_endpoint.settings, "LLM_MODEL", "deepseek-chat"
|
||
), patch.object(
|
||
system_endpoint.LLMHelper,
|
||
"test_current_settings",
|
||
AsyncMock(side_effect=RuntimeError(raw_error)),
|
||
create=True,
|
||
):
|
||
resp = asyncio.run(system_endpoint.llm_test(_="token"))
|
||
|
||
self.assertFalse(resp.success)
|
||
self.assertNotIn("sk-secret", resp.message)
|
||
self.assertNotIn("Authorization: Bearer", resp.message)
|
||
self.assertIn("***", resp.message)
|