mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-02 03:55:55 +00:00
274 lines
9.9 KiB
Python
274 lines
9.9 KiB
Python
from __future__ import annotations
|
|
|
|
from types import SimpleNamespace
|
|
from unittest.mock import AsyncMock, Mock
|
|
|
|
import pytest
|
|
import langbot_plugin.api.entities.builtin.pipeline.query as pipeline_query
|
|
import langbot_plugin.api.entities.builtin.platform.entities as platform_entities
|
|
import langbot_plugin.api.entities.builtin.platform.events as platform_events
|
|
import langbot_plugin.api.entities.builtin.platform.message as platform_message
|
|
import langbot_plugin.api.entities.builtin.provider.session as provider_session
|
|
|
|
from langbot.pkg.api.http.service.model import _runtime_model_data
|
|
from langbot.pkg.api.http.service.provider import ModelProviderService
|
|
from langbot.pkg.entity.persistence import model as persistence_model
|
|
from langbot.pkg.pipeline.preproc.preproc import PreProcessor
|
|
from langbot.pkg.provider.modelmgr import requester
|
|
from langbot.pkg.provider.modelmgr.modelmgr import ModelManager
|
|
from langbot.pkg.provider.modelmgr.requesters.chatcmpl import OpenAIChatCompletions
|
|
from langbot.pkg.provider.modelmgr.requesters.modelscopechatcmpl import ModelScopeChatCompletions
|
|
from langbot.pkg.provider.modelmgr.token import TokenManager
|
|
from langbot.pkg.provider.runners.localagent import LocalAgentRunner
|
|
|
|
|
|
def test_runtime_llm_model_data_preserves_uuid_after_update_payload_uuid_removed():
|
|
update_payload = {
|
|
'name': 'Qwen3.5-27B',
|
|
'provider_uuid': 'provider-uuid',
|
|
'abilities': [],
|
|
'extra_args': {},
|
|
}
|
|
|
|
runtime_entity = persistence_model.LLMModel(**_runtime_model_data('model-uuid', update_payload))
|
|
|
|
assert runtime_entity.uuid == 'model-uuid'
|
|
assert runtime_entity.name == 'Qwen3.5-27B'
|
|
|
|
|
|
def test_runtime_embedding_model_data_preserves_uuid_after_update_payload_uuid_removed():
|
|
update_payload = {
|
|
'name': 'embedding-model',
|
|
'provider_uuid': 'provider-uuid',
|
|
'extra_args': {},
|
|
}
|
|
|
|
runtime_entity = persistence_model.EmbeddingModel(**_runtime_model_data('embedding-uuid', update_payload))
|
|
|
|
assert runtime_entity.uuid == 'embedding-uuid'
|
|
assert runtime_entity.name == 'embedding-model'
|
|
|
|
|
|
def test_runtime_rerank_model_data_preserves_uuid_after_update_payload_uuid_removed():
|
|
update_payload = {
|
|
'name': 'rerank-model',
|
|
'provider_uuid': 'provider-uuid',
|
|
'extra_args': {},
|
|
}
|
|
|
|
runtime_entity = persistence_model.RerankModel(**_runtime_model_data('rerank-uuid', update_payload))
|
|
|
|
assert runtime_entity.uuid == 'rerank-uuid'
|
|
assert runtime_entity.name == 'rerank-model'
|
|
|
|
|
|
def test_normalize_space_provider_api_keys_filters_blank_values():
|
|
assert ModelProviderService._normalize_api_keys('space-key') == ['space-key']
|
|
assert ModelProviderService._normalize_api_keys(' trimmed-key ') == ['trimmed-key']
|
|
assert ModelProviderService._normalize_api_keys('') == []
|
|
assert ModelProviderService._normalize_api_keys(' ') == []
|
|
assert ModelProviderService._normalize_api_keys(None) == []
|
|
assert ModelProviderService._normalize_api_keys([' first-key ', '', 'first-key', 'second-key']) == [
|
|
'first-key',
|
|
'second-key',
|
|
]
|
|
|
|
|
|
def test_token_manager_filters_blank_and_duplicate_tokens():
|
|
token_mgr = TokenManager('provider-uuid', [' first-key ', '', 'first-key', 'second-key', ' '])
|
|
|
|
assert token_mgr.tokens == ['first-key', 'second-key']
|
|
assert token_mgr.get_token() == 'first-key'
|
|
|
|
|
|
def test_token_manager_next_token_ignores_empty_token_list():
|
|
token_mgr = TokenManager('provider-uuid', [])
|
|
|
|
token_mgr.next_token()
|
|
|
|
assert token_mgr.get_token() == ''
|
|
assert token_mgr.using_token_index == 0
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_openai_requester_initialize_uses_placeholder_api_key(monkeypatch):
|
|
captured_kwargs = {}
|
|
|
|
def fake_client(**kwargs):
|
|
captured_kwargs.update(kwargs)
|
|
return SimpleNamespace(**kwargs)
|
|
|
|
monkeypatch.setattr('langbot.pkg.provider.modelmgr.requesters.chatcmpl.openai.AsyncClient', fake_client)
|
|
monkeypatch.setattr('langbot.pkg.provider.modelmgr.requesters.chatcmpl.httpx.AsyncClient', fake_client)
|
|
|
|
requester_inst = OpenAIChatCompletions(ap=SimpleNamespace(), config={})
|
|
await requester_inst.initialize()
|
|
|
|
assert captured_kwargs['api_key'] == OpenAIChatCompletions.init_api_key
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_modelscope_requester_initialize_uses_placeholder_api_key(monkeypatch):
|
|
captured_kwargs = {}
|
|
|
|
def fake_client(**kwargs):
|
|
captured_kwargs.update(kwargs)
|
|
return SimpleNamespace(**kwargs)
|
|
|
|
monkeypatch.setattr('langbot.pkg.provider.modelmgr.requesters.modelscopechatcmpl.openai.AsyncClient', fake_client)
|
|
monkeypatch.setattr('langbot.pkg.provider.modelmgr.requesters.modelscopechatcmpl.httpx.AsyncClient', fake_client)
|
|
|
|
requester_inst = ModelScopeChatCompletions(ap=SimpleNamespace(), config={})
|
|
await requester_inst.initialize()
|
|
|
|
assert captured_kwargs['api_key'] == ModelScopeChatCompletions.init_api_key
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_openai_embedding_call_overrides_placeholder_api_key():
|
|
captured_request = {}
|
|
|
|
async def fake_create(**kwargs):
|
|
captured_request['api_key'] = fake_client.api_key
|
|
captured_request['kwargs'] = kwargs
|
|
return SimpleNamespace(
|
|
data=[SimpleNamespace(embedding=[0.1, 0.2])],
|
|
usage=SimpleNamespace(prompt_tokens=3, total_tokens=3),
|
|
)
|
|
|
|
fake_client = SimpleNamespace(
|
|
api_key=OpenAIChatCompletions.init_api_key,
|
|
embeddings=SimpleNamespace(create=fake_create),
|
|
)
|
|
|
|
requester_inst = OpenAIChatCompletions(ap=SimpleNamespace(), config={})
|
|
requester_inst.client = fake_client
|
|
|
|
embeddings, usage_info = await requester_inst.invoke_embedding(
|
|
model=requester.RuntimeEmbeddingModel(
|
|
model_entity=SimpleNamespace(name='text-embedding-3-small', extra_args={}),
|
|
provider=SimpleNamespace(token_mgr=TokenManager('provider-uuid', [' runtime-key ', '', 'runtime-key'])),
|
|
),
|
|
input_text=['hello'],
|
|
)
|
|
|
|
assert captured_request['api_key'] == 'runtime-key'
|
|
assert captured_request['kwargs']['model'] == 'text-embedding-3-small'
|
|
assert embeddings == [[0.1, 0.2]]
|
|
assert usage_info == {'prompt_tokens': 3, 'total_tokens': 3}
|
|
|
|
|
|
@pytest.mark.asyncio
|
|
async def test_updated_llm_model_is_immediately_usable_by_local_agent_pipeline():
|
|
from langbot.pkg.api.http.service.model import LLMModelsService
|
|
|
|
model_uuid = 'qwen-model-uuid'
|
|
provider_uuid = 'ollama-provider-uuid'
|
|
|
|
ap = SimpleNamespace()
|
|
ap.logger = Mock()
|
|
ap.persistence_mgr = SimpleNamespace(execute_async=AsyncMock())
|
|
ap.tool_mgr = SimpleNamespace(get_all_tools=AsyncMock(return_value=[]))
|
|
ap.plugin_connector = SimpleNamespace(
|
|
emit_event=AsyncMock(return_value=SimpleNamespace(event=SimpleNamespace(default_prompt=[], prompt=[])))
|
|
)
|
|
|
|
ap.model_mgr = ModelManager(ap)
|
|
runtime_provider = Mock()
|
|
ap.model_mgr.provider_dict = {provider_uuid: runtime_provider}
|
|
ap.model_mgr.llm_models = [
|
|
requester.RuntimeLLMModel(
|
|
model_entity=persistence_model.LLMModel(
|
|
uuid=model_uuid,
|
|
name='old-qwen-name',
|
|
provider_uuid=provider_uuid,
|
|
abilities=[],
|
|
extra_args={},
|
|
),
|
|
provider=runtime_provider,
|
|
)
|
|
]
|
|
|
|
await LLMModelsService(ap).update_llm_model(
|
|
model_uuid,
|
|
{
|
|
'name': 'Qwen3.5-27B',
|
|
'provider_uuid': provider_uuid,
|
|
'abilities': [],
|
|
'extra_args': {},
|
|
},
|
|
)
|
|
|
|
runtime_model = await ap.model_mgr.get_model_by_uuid(model_uuid)
|
|
assert runtime_model.model_entity.uuid == model_uuid
|
|
assert runtime_model.model_entity.name == 'Qwen3.5-27B'
|
|
|
|
session = SimpleNamespace(
|
|
launcher_type=provider_session.LauncherTypes.PERSON,
|
|
launcher_id=12345,
|
|
)
|
|
conversation = SimpleNamespace(
|
|
uuid='conversation-uuid',
|
|
create_time=None,
|
|
update_time=None,
|
|
prompt=SimpleNamespace(messages=[], copy=Mock(return_value=SimpleNamespace(messages=[]))),
|
|
messages=[],
|
|
)
|
|
ap.sess_mgr = SimpleNamespace(
|
|
get_session=AsyncMock(return_value=session),
|
|
get_conversation=AsyncMock(return_value=conversation),
|
|
)
|
|
|
|
message_chain = platform_message.MessageChain([platform_message.Plain(text='hello')])
|
|
sender = platform_entities.Friend(id=12345, nickname='Tester', remark=None)
|
|
message_event = platform_events.FriendMessage(
|
|
type='FriendMessage',
|
|
sender=sender,
|
|
message_chain=message_chain,
|
|
time=1710000000,
|
|
)
|
|
pipeline_config = {
|
|
'ai': {
|
|
'runner': {'runner': 'local-agent'},
|
|
'local-agent': {
|
|
'model': {'primary': model_uuid, 'fallbacks': []},
|
|
'prompt': [],
|
|
'knowledge-bases': [],
|
|
},
|
|
},
|
|
'trigger': {'misc': {'combine-quote-message': False}},
|
|
'output': {'misc': {'remove-think': False}},
|
|
}
|
|
query = pipeline_query.Query.model_construct(
|
|
query_id='query-id',
|
|
launcher_type=provider_session.LauncherTypes.PERSON,
|
|
launcher_id=12345,
|
|
sender_id=12345,
|
|
message_chain=message_chain,
|
|
message_event=message_event,
|
|
adapter=AsyncMock(),
|
|
pipeline_uuid='pipeline-uuid',
|
|
bot_uuid='bot-uuid',
|
|
pipeline_config=pipeline_config,
|
|
session=None,
|
|
prompt=None,
|
|
messages=[],
|
|
user_message=None,
|
|
use_funcs=[],
|
|
use_llm_model_uuid=None,
|
|
variables={},
|
|
resp_messages=[],
|
|
resp_message_chain=None,
|
|
current_stage_name=None,
|
|
)
|
|
|
|
result = await PreProcessor(ap).process(query, 'PreProcessor')
|
|
processed_query = result.new_query
|
|
|
|
assert processed_query.use_llm_model_uuid == model_uuid
|
|
|
|
runner = SimpleNamespace(ap=ap, pipeline_config=pipeline_config)
|
|
candidates = await LocalAgentRunner._get_model_candidates(runner, processed_query)
|
|
|
|
assert [model.model_entity.uuid for model in candidates] == [model_uuid]
|