| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562 | 
							- import logging
 - from typing import Optional
 - 
 - from core.entities.model_entities import ModelStatus, ModelWithProviderEntity, ProviderModelWithStatusEntity
 - from core.model_runtime.entities.model_entities import ModelType, ParameterRule
 - from core.model_runtime.model_providers.model_provider_factory import ModelProviderFactory
 - from core.provider_manager import ProviderManager
 - from models.provider import ProviderType
 - from services.entities.model_provider_entities import (
 -     CustomConfigurationResponse,
 -     CustomConfigurationStatus,
 -     DefaultModelResponse,
 -     ModelWithProviderEntityResponse,
 -     ProviderResponse,
 -     ProviderWithModelsResponse,
 -     SimpleProviderEntityResponse,
 -     SystemConfigurationResponse,
 - )
 - from services.errors.app_model_config import ProviderNotFoundError
 - 
 - logger = logging.getLogger(__name__)
 - 
 - 
 - class ModelProviderService:
 -     """
 -     Model Provider Service
 -     """
 - 
 -     def __init__(self) -> None:
 -         self.provider_manager = ProviderManager()
 - 
 -     def _get_provider_configuration(self, tenant_id: str, provider: str):
 -         """
 -         Get provider configuration or raise exception if not found.
 - 
 -         Args:
 -             tenant_id: Workspace identifier
 -             provider: Provider name
 - 
 -         Returns:
 -             Provider configuration instance
 - 
 -         Raises:
 -             ProviderNotFoundError: If provider doesn't exist
 -         """
 -         # Get all provider configurations of the current workspace
 -         provider_configurations = self.provider_manager.get_configurations(tenant_id)
 -         provider_configuration = provider_configurations.get(provider)
 - 
 -         if not provider_configuration:
 -             raise ProviderNotFoundError(f"Provider {provider} does not exist.")
 - 
 -         return provider_configuration
 - 
 -     def get_provider_list(self, tenant_id: str, model_type: Optional[str] = None) -> list[ProviderResponse]:
 -         """
 -         get provider list.
 - 
 -         :param tenant_id: workspace id
 -         :param model_type: model type
 -         :return:
 -         """
 -         # Get all provider configurations of the current workspace
 -         provider_configurations = self.provider_manager.get_configurations(tenant_id)
 - 
 -         provider_responses = []
 -         for provider_configuration in provider_configurations.values():
 -             if model_type:
 -                 model_type_entity = ModelType.value_of(model_type)
 -                 if model_type_entity not in provider_configuration.provider.supported_model_types:
 -                     continue
 - 
 -             provider_config = provider_configuration.custom_configuration.provider
 -             model_config = provider_configuration.custom_configuration.models
 - 
 -             provider_response = ProviderResponse(
 -                 tenant_id=tenant_id,
 -                 provider=provider_configuration.provider.provider,
 -                 label=provider_configuration.provider.label,
 -                 description=provider_configuration.provider.description,
 -                 icon_small=provider_configuration.provider.icon_small,
 -                 icon_large=provider_configuration.provider.icon_large,
 -                 background=provider_configuration.provider.background,
 -                 help=provider_configuration.provider.help,
 -                 supported_model_types=provider_configuration.provider.supported_model_types,
 -                 configurate_methods=provider_configuration.provider.configurate_methods,
 -                 provider_credential_schema=provider_configuration.provider.provider_credential_schema,
 -                 model_credential_schema=provider_configuration.provider.model_credential_schema,
 -                 preferred_provider_type=provider_configuration.preferred_provider_type,
 -                 custom_configuration=CustomConfigurationResponse(
 -                     status=CustomConfigurationStatus.ACTIVE
 -                     if provider_configuration.is_custom_configuration_available()
 -                     else CustomConfigurationStatus.NO_CONFIGURE,
 -                     current_credential_id=getattr(provider_config, "current_credential_id", None),
 -                     current_credential_name=getattr(provider_config, "current_credential_name", None),
 -                     available_credentials=getattr(provider_config, "available_credentials", []),
 -                     custom_models=model_config,
 -                 ),
 -                 system_configuration=SystemConfigurationResponse(
 -                     enabled=provider_configuration.system_configuration.enabled,
 -                     current_quota_type=provider_configuration.system_configuration.current_quota_type,
 -                     quota_configurations=provider_configuration.system_configuration.quota_configurations,
 -                 ),
 -             )
 - 
 -             provider_responses.append(provider_response)
 - 
 -         return provider_responses
 - 
 -     def get_models_by_provider(self, tenant_id: str, provider: str) -> list[ModelWithProviderEntityResponse]:
 -         """
 -         get provider models.
 -         For the model provider page,
 -         only supports passing in a single provider to query the list of supported models.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :return:
 -         """
 -         # Get all provider configurations of the current workspace
 -         provider_configurations = self.provider_manager.get_configurations(tenant_id)
 - 
 -         # Get provider available models
 -         return [
 -             ModelWithProviderEntityResponse(tenant_id=tenant_id, model=model)
 -             for model in provider_configurations.get_models(provider=provider)
 -         ]
 - 
 -     def get_provider_credential(
 -         self, tenant_id: str, provider: str, credential_id: Optional[str] = None
 -     ) -> Optional[dict]:
 -         """
 -         get provider credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credential_id: credential id, if not provided, return current used credentials
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         return provider_configuration.get_provider_credential(credential_id=credential_id)  # type: ignore
 - 
 -     def validate_provider_credentials(self, tenant_id: str, provider: str, credentials: dict) -> None:
 -         """
 -         validate provider credentials before saving.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credentials: provider credentials dict
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.validate_provider_credentials(credentials)
 - 
 -     def create_provider_credential(
 -         self, tenant_id: str, provider: str, credentials: dict, credential_name: str
 -     ) -> None:
 -         """
 -         Create and save new provider credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credentials: provider credentials dict
 -         :param credential_name: credential name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.create_provider_credential(credentials, credential_name)
 - 
 -     def update_provider_credential(
 -         self,
 -         tenant_id: str,
 -         provider: str,
 -         credentials: dict,
 -         credential_id: str,
 -         credential_name: str,
 -     ) -> None:
 -         """
 -         update a saved provider credential (by credential_id).
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credentials: provider credentials dict
 -         :param credential_id: credential id
 -         :param credential_name: credential name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.update_provider_credential(
 -             credential_id=credential_id,
 -             credentials=credentials,
 -             credential_name=credential_name,
 -         )
 - 
 -     def remove_provider_credential(self, tenant_id: str, provider: str, credential_id: str) -> None:
 -         """
 -         remove a saved provider credential (by credential_id).
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credential_id: credential id
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.delete_provider_credential(credential_id=credential_id)
 - 
 -     def switch_active_provider_credential(self, tenant_id: str, provider: str, credential_id: str) -> None:
 -         """
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param credential_id: credential id
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.switch_active_provider_credential(credential_id=credential_id)
 - 
 -     def get_model_credential(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credential_id: str | None
 -     ) -> Optional[dict]:
 -         """
 -         Retrieve model-specific credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credential_id: Optional credential ID, uses current if not provided
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         return provider_configuration.get_custom_model_credential(  # type: ignore
 -             model_type=ModelType.value_of(model_type), model=model, credential_id=credential_id
 -         )
 - 
 -     def validate_model_credentials(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credentials: dict
 -     ) -> None:
 -         """
 -         validate model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credentials: model credentials dict
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.validate_custom_model_credentials(
 -             model_type=ModelType.value_of(model_type), model=model, credentials=credentials
 -         )
 - 
 -     def create_model_credential(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credentials: dict, credential_name: str
 -     ) -> None:
 -         """
 -         create and save model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credentials: model credentials dict
 -         :param credential_name: credential name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.create_custom_model_credential(
 -             model_type=ModelType.value_of(model_type),
 -             model=model,
 -             credentials=credentials,
 -             credential_name=credential_name,
 -         )
 - 
 -     def update_model_credential(
 -         self,
 -         tenant_id: str,
 -         provider: str,
 -         model_type: str,
 -         model: str,
 -         credentials: dict,
 -         credential_id: str,
 -         credential_name: str,
 -     ) -> None:
 -         """
 -         update model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credentials: model credentials dict
 -         :param credential_id: credential id
 -         :param credential_name: credential name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.update_custom_model_credential(
 -             model_type=ModelType.value_of(model_type),
 -             model=model,
 -             credentials=credentials,
 -             credential_id=credential_id,
 -             credential_name=credential_name,
 -         )
 - 
 -     def remove_model_credential(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credential_id: str
 -     ) -> None:
 -         """
 -         remove model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credential_id: credential id
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.delete_custom_model_credential(
 -             model_type=ModelType.value_of(model_type), model=model, credential_id=credential_id
 -         )
 - 
 -     def switch_active_custom_model_credential(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credential_id: str
 -     ) -> None:
 -         """
 -         switch model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credential_id: credential id
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.switch_custom_model_credential(
 -             model_type=ModelType.value_of(model_type), model=model, credential_id=credential_id
 -         )
 - 
 -     def add_model_credential_to_model_list(
 -         self, tenant_id: str, provider: str, model_type: str, model: str, credential_id: str
 -     ) -> None:
 -         """
 -         add model credentials to model list.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :param credential_id: credential id
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.add_model_credential_to_model(
 -             model_type=ModelType.value_of(model_type), model=model, credential_id=credential_id
 -         )
 - 
 -     def remove_model(self, tenant_id: str, provider: str, model_type: str, model: str) -> None:
 -         """
 -         remove model credentials.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model_type: model type
 -         :param model: model name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.delete_custom_model(model_type=ModelType.value_of(model_type), model=model)
 - 
 -     def get_models_by_model_type(self, tenant_id: str, model_type: str) -> list[ProviderWithModelsResponse]:
 -         """
 -         get models by model type.
 - 
 -         :param tenant_id: workspace id
 -         :param model_type: model type
 -         :return:
 -         """
 -         # Get all provider configurations of the current workspace
 -         provider_configurations = self.provider_manager.get_configurations(tenant_id)
 - 
 -         # Get provider available models
 -         models = provider_configurations.get_models(model_type=ModelType.value_of(model_type))
 - 
 -         # Group models by provider
 -         provider_models: dict[str, list[ModelWithProviderEntity]] = {}
 -         for model in models:
 -             if model.provider.provider not in provider_models:
 -                 provider_models[model.provider.provider] = []
 - 
 -             if model.deprecated:
 -                 continue
 - 
 -             if model.status != ModelStatus.ACTIVE:
 -                 continue
 - 
 -             provider_models[model.provider.provider].append(model)
 - 
 -         # convert to ProviderWithModelsResponse list
 -         providers_with_models: list[ProviderWithModelsResponse] = []
 -         for provider, models in provider_models.items():
 -             if not models:
 -                 continue
 - 
 -             first_model = models[0]
 - 
 -             providers_with_models.append(
 -                 ProviderWithModelsResponse(
 -                     tenant_id=tenant_id,
 -                     provider=provider,
 -                     label=first_model.provider.label,
 -                     icon_small=first_model.provider.icon_small,
 -                     icon_large=first_model.provider.icon_large,
 -                     status=CustomConfigurationStatus.ACTIVE,
 -                     models=[
 -                         ProviderModelWithStatusEntity(
 -                             model=model.model,
 -                             label=model.label,
 -                             model_type=model.model_type,
 -                             features=model.features,
 -                             fetch_from=model.fetch_from,
 -                             model_properties=model.model_properties,
 -                             status=model.status,
 -                             load_balancing_enabled=model.load_balancing_enabled,
 -                         )
 -                         for model in models
 -                     ],
 -                 )
 -             )
 - 
 -         return providers_with_models
 - 
 -     def get_model_parameter_rules(self, tenant_id: str, provider: str, model: str) -> list[ParameterRule]:
 -         """
 -         get model parameter rules.
 -         Only supports LLM.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model: model name
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 - 
 -         # fetch credentials
 -         credentials = provider_configuration.get_current_credentials(model_type=ModelType.LLM, model=model)
 - 
 -         if not credentials:
 -             return []
 - 
 -         model_schema = provider_configuration.get_model_schema(
 -             model_type=ModelType.LLM, model=model, credentials=credentials
 -         )
 - 
 -         return model_schema.parameter_rules if model_schema else []
 - 
 -     def get_default_model_of_model_type(self, tenant_id: str, model_type: str) -> Optional[DefaultModelResponse]:
 -         """
 -         get default model of model type.
 - 
 -         :param tenant_id: workspace id
 -         :param model_type: model type
 -         :return:
 -         """
 -         model_type_enum = ModelType.value_of(model_type)
 - 
 -         try:
 -             result = self.provider_manager.get_default_model(tenant_id=tenant_id, model_type=model_type_enum)
 -             return (
 -                 DefaultModelResponse(
 -                     model=result.model,
 -                     model_type=result.model_type,
 -                     provider=SimpleProviderEntityResponse(
 -                         tenant_id=tenant_id,
 -                         provider=result.provider.provider,
 -                         label=result.provider.label,
 -                         icon_small=result.provider.icon_small,
 -                         icon_large=result.provider.icon_large,
 -                         supported_model_types=result.provider.supported_model_types,
 -                     ),
 -                 )
 -                 if result
 -                 else None
 -             )
 -         except Exception as e:
 -             logger.debug("get_default_model_of_model_type error: %s", e)
 -             return None
 - 
 -     def update_default_model_of_model_type(self, tenant_id: str, model_type: str, provider: str, model: str) -> None:
 -         """
 -         update default model of model type.
 - 
 -         :param tenant_id: workspace id
 -         :param model_type: model type
 -         :param provider: provider name
 -         :param model: model name
 -         :return:
 -         """
 -         model_type_enum = ModelType.value_of(model_type)
 -         self.provider_manager.update_default_model_record(
 -             tenant_id=tenant_id, model_type=model_type_enum, provider=provider, model=model
 -         )
 - 
 -     def get_model_provider_icon(
 -         self, tenant_id: str, provider: str, icon_type: str, lang: str
 -     ) -> tuple[Optional[bytes], Optional[str]]:
 -         """
 -         get model provider icon.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param icon_type: icon type (icon_small or icon_large)
 -         :param lang: language (zh_Hans or en_US)
 -         :return:
 -         """
 -         model_provider_factory = ModelProviderFactory(tenant_id)
 -         byte_data, mime_type = model_provider_factory.get_provider_icon(provider, icon_type, lang)
 - 
 -         return byte_data, mime_type
 - 
 -     def switch_preferred_provider(self, tenant_id: str, provider: str, preferred_provider_type: str) -> None:
 -         """
 -         switch preferred provider.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param preferred_provider_type: preferred provider type
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 - 
 -         # Convert preferred_provider_type to ProviderType
 -         preferred_provider_type_enum = ProviderType.value_of(preferred_provider_type)
 - 
 -         # Switch preferred provider type
 -         provider_configuration.switch_preferred_provider_type(preferred_provider_type_enum)
 - 
 -     def enable_model(self, tenant_id: str, provider: str, model: str, model_type: str) -> None:
 -         """
 -         enable model.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model: model name
 -         :param model_type: model type
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.enable_model(model=model, model_type=ModelType.value_of(model_type))
 - 
 -     def disable_model(self, tenant_id: str, provider: str, model: str, model_type: str) -> None:
 -         """
 -         disable model.
 - 
 -         :param tenant_id: workspace id
 -         :param provider: provider name
 -         :param model: model name
 -         :param model_type: model type
 -         :return:
 -         """
 -         provider_configuration = self._get_provider_configuration(tenant_id, provider)
 -         provider_configuration.disable_model(model=model, model_type=ModelType.value_of(model_type))
 
 
  |