mirror of
https://github.com/langbot-app/LangBot.git
synced 2026-06-02 12:05:54 +00:00
* feat(provider): add rerank model management as a core model type * feat(provider): add rerank support to existing requesters and new rerank providers * feat(web): add rerank model management UI and pipeline config * fix(provider): correct rerank support_type after verification - Add rerank to OpenRouter (confirmed /api/v1/rerank endpoint) - Remove rerank from Ollama (no native support, PR #7219 unmerged) - Remove rerank from JiekouAI (no rerank docs found, URL path mismatch) * fix(provider): remove alru_cache from model getters and add rerank param hints * fix: resolve lint errors - Remove unused alru_cache import from modelmgr.py - Remove unused error_message variable in invoke_rerank - Fix prettier formatting in frontend files Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * fix: remove unused exception variable - Change `except Exception as e:` to `except Exception:` since e is not used - Fix prettier formatting in ProviderCard.tsx Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * fix: apply ruff format Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * feat(template): add rerank config fields to default pipeline config Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * chore: remove PR.md Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> * fix(ui): remove duplicate rerank model form in AddModelPopover The form was being rendered twice: once in TabsContent manual mode and again in a separate conditional block for rerank tab. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com> --------- Co-authored-by: Claude Opus 4.7 <noreply@anthropic.com>
148 lines
6.6 KiB
Python
148 lines
6.6 KiB
Python
import quart
|
|
|
|
from ... import group
|
|
|
|
|
|
@group.group_class('models/llm', '/api/v1/provider/models/llm')
|
|
class LLMModelsRouterGroup(group.RouterGroup):
|
|
async def initialize(self) -> None:
|
|
@self.route('', methods=['GET', 'POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _() -> str:
|
|
if quart.request.method == 'GET':
|
|
provider_uuid = quart.request.args.get('provider_uuid')
|
|
if provider_uuid:
|
|
return self.success(
|
|
data={'models': await self.ap.llm_model_service.get_llm_models_by_provider(provider_uuid)}
|
|
)
|
|
return self.success(data={'models': await self.ap.llm_model_service.get_llm_models()})
|
|
elif quart.request.method == 'POST':
|
|
json_data = await quart.request.json
|
|
model_uuid = await self.ap.llm_model_service.create_llm_model(json_data)
|
|
return self.success(data={'uuid': model_uuid})
|
|
|
|
@self.route('/<model_uuid>', methods=['GET', 'PUT', 'DELETE'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
if quart.request.method == 'GET':
|
|
model = await self.ap.llm_model_service.get_llm_model(model_uuid)
|
|
|
|
if model is None:
|
|
return self.http_status(404, -1, 'model not found')
|
|
|
|
return self.success(data={'model': model})
|
|
elif quart.request.method == 'PUT':
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.llm_model_service.update_llm_model(model_uuid, json_data)
|
|
|
|
return self.success()
|
|
elif quart.request.method == 'DELETE':
|
|
await self.ap.llm_model_service.delete_llm_model(model_uuid)
|
|
|
|
return self.success()
|
|
|
|
@self.route('/<model_uuid>/test', methods=['POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.llm_model_service.test_llm_model(model_uuid, json_data)
|
|
|
|
return self.success()
|
|
|
|
|
|
@group.group_class('models/embedding', '/api/v1/provider/models/embedding')
|
|
class EmbeddingModelsRouterGroup(group.RouterGroup):
|
|
async def initialize(self) -> None:
|
|
@self.route('', methods=['GET', 'POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _() -> str:
|
|
if quart.request.method == 'GET':
|
|
provider_uuid = quart.request.args.get('provider_uuid')
|
|
if provider_uuid:
|
|
return self.success(
|
|
data={
|
|
'models': await self.ap.embedding_models_service.get_embedding_models_by_provider(
|
|
provider_uuid
|
|
)
|
|
}
|
|
)
|
|
return self.success(data={'models': await self.ap.embedding_models_service.get_embedding_models()})
|
|
elif quart.request.method == 'POST':
|
|
json_data = await quart.request.json
|
|
model_uuid = await self.ap.embedding_models_service.create_embedding_model(json_data)
|
|
return self.success(data={'uuid': model_uuid})
|
|
|
|
@self.route('/<model_uuid>', methods=['GET', 'PUT', 'DELETE'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
if quart.request.method == 'GET':
|
|
model = await self.ap.embedding_models_service.get_embedding_model(model_uuid)
|
|
|
|
if model is None:
|
|
return self.http_status(404, -1, 'model not found')
|
|
|
|
return self.success(data={'model': model})
|
|
elif quart.request.method == 'PUT':
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.embedding_models_service.update_embedding_model(model_uuid, json_data)
|
|
|
|
return self.success()
|
|
elif quart.request.method == 'DELETE':
|
|
await self.ap.embedding_models_service.delete_embedding_model(model_uuid)
|
|
|
|
return self.success()
|
|
|
|
@self.route('/<model_uuid>/test', methods=['POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.embedding_models_service.test_embedding_model(model_uuid, json_data)
|
|
|
|
return self.success()
|
|
|
|
|
|
@group.group_class('models/rerank', '/api/v1/provider/models/rerank')
|
|
class RerankModelsRouterGroup(group.RouterGroup):
|
|
async def initialize(self) -> None:
|
|
@self.route('', methods=['GET', 'POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _() -> str:
|
|
if quart.request.method == 'GET':
|
|
provider_uuid = quart.request.args.get('provider_uuid')
|
|
if provider_uuid:
|
|
return self.success(
|
|
data={
|
|
'models': await self.ap.rerank_models_service.get_rerank_models_by_provider(provider_uuid)
|
|
}
|
|
)
|
|
return self.success(data={'models': await self.ap.rerank_models_service.get_rerank_models()})
|
|
elif quart.request.method == 'POST':
|
|
json_data = await quart.request.json
|
|
model_uuid = await self.ap.rerank_models_service.create_rerank_model(json_data)
|
|
return self.success(data={'uuid': model_uuid})
|
|
|
|
@self.route('/<model_uuid>', methods=['GET', 'PUT', 'DELETE'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
if quart.request.method == 'GET':
|
|
model = await self.ap.rerank_models_service.get_rerank_model(model_uuid)
|
|
|
|
if model is None:
|
|
return self.http_status(404, -1, 'model not found')
|
|
|
|
return self.success(data={'model': model})
|
|
elif quart.request.method == 'PUT':
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.rerank_models_service.update_rerank_model(model_uuid, json_data)
|
|
|
|
return self.success()
|
|
elif quart.request.method == 'DELETE':
|
|
await self.ap.rerank_models_service.delete_rerank_model(model_uuid)
|
|
|
|
return self.success()
|
|
|
|
@self.route('/<model_uuid>/test', methods=['POST'], auth_type=group.AuthType.USER_TOKEN_OR_API_KEY)
|
|
async def _(model_uuid: str) -> str:
|
|
json_data = await quart.request.json
|
|
|
|
await self.ap.rerank_models_service.test_rerank_model(model_uuid, json_data)
|
|
|
|
return self.success()
|