azure_provider.py 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. import json
  2. import logging
  3. from typing import Optional, Union
  4. import requests
  5. from core.llm.provider.base import BaseProvider
  6. from core.llm.provider.errors import ValidateFailedError
  7. from models.provider import ProviderName
  8. class AzureProvider(BaseProvider):
  9. def get_models(self, model_id: Optional[str] = None, credentials: Optional[dict] = None) -> list[dict]:
  10. credentials = self.get_credentials(model_id) if not credentials else credentials
  11. url = "{}/openai/deployments?api-version={}".format(
  12. str(credentials.get('openai_api_base')),
  13. str(credentials.get('openai_api_version'))
  14. )
  15. headers = {
  16. "api-key": str(credentials.get('openai_api_key')),
  17. "content-type": "application/json; charset=utf-8"
  18. }
  19. response = requests.get(url, headers=headers)
  20. if response.status_code == 200:
  21. result = response.json()
  22. return [{
  23. 'id': deployment['id'],
  24. 'name': '{} ({})'.format(deployment['id'], deployment['model'])
  25. } for deployment in result['data'] if deployment['status'] == 'succeeded']
  26. else:
  27. if response.status_code == 401:
  28. raise AzureAuthenticationError()
  29. else:
  30. raise AzureRequestFailedError('Failed to request Azure OpenAI. Status code: {}'.format(response.status_code))
  31. def get_credentials(self, model_id: Optional[str] = None) -> dict:
  32. """
  33. Returns the API credentials for Azure OpenAI as a dictionary.
  34. """
  35. config = self.get_provider_api_key(model_id=model_id)
  36. config['openai_api_type'] = 'azure'
  37. if model_id == 'text-embedding-ada-002':
  38. config['deployment'] = model_id.replace('.', '') if model_id else None
  39. else:
  40. config['deployment_name'] = model_id.replace('.', '') if model_id else None
  41. return config
  42. def get_provider_name(self):
  43. return ProviderName.AZURE_OPENAI
  44. def get_provider_configs(self, obfuscated: bool = False) -> Union[str | dict]:
  45. """
  46. Returns the provider configs.
  47. """
  48. try:
  49. config = self.get_provider_api_key()
  50. except:
  51. config = {
  52. 'openai_api_type': 'azure',
  53. 'openai_api_version': '2023-03-15-preview',
  54. 'openai_api_base': '',
  55. 'openai_api_key': ''
  56. }
  57. if obfuscated:
  58. if not config.get('openai_api_key'):
  59. config = {
  60. 'openai_api_type': 'azure',
  61. 'openai_api_version': '2023-03-15-preview',
  62. 'openai_api_base': '',
  63. 'openai_api_key': ''
  64. }
  65. config['openai_api_key'] = self.obfuscated_token(config.get('openai_api_key'))
  66. return config
  67. return config
  68. def get_token_type(self):
  69. # TODO: change to dict when implemented
  70. return dict
  71. def config_validate(self, config: Union[dict | str]):
  72. """
  73. Validates the given config.
  74. """
  75. try:
  76. if not isinstance(config, dict):
  77. raise ValueError('Config must be a object.')
  78. if 'openai_api_version' not in config:
  79. config['openai_api_version'] = '2023-03-15-preview'
  80. models = self.get_models(credentials=config)
  81. if not models:
  82. raise ValidateFailedError("Please add deployments for 'text-davinci-003', "
  83. "'gpt-3.5-turbo', 'text-embedding-ada-002' (required) "
  84. "and 'gpt-4', 'gpt-35-turbo-16k' (optional).")
  85. fixed_model_ids = [
  86. 'text-davinci-003',
  87. 'gpt-35-turbo',
  88. 'text-embedding-ada-002'
  89. ]
  90. current_model_ids = [model['id'] for model in models]
  91. missing_model_ids = [fixed_model_id for fixed_model_id in fixed_model_ids if
  92. fixed_model_id not in current_model_ids]
  93. if missing_model_ids:
  94. raise ValidateFailedError("Please add deployments for '{}'.".format(", ".join(missing_model_ids)))
  95. except ValidateFailedError as e:
  96. raise e
  97. except AzureAuthenticationError:
  98. raise ValidateFailedError('Validation failed, please check your API Key.')
  99. except (requests.ConnectionError, requests.RequestException):
  100. raise ValidateFailedError('Validation failed, please check your API Base Endpoint.')
  101. except AzureRequestFailedError as ex:
  102. raise ValidateFailedError('Validation failed, error: {}.'.format(str(ex)))
  103. except Exception as ex:
  104. logging.exception('Azure OpenAI Credentials validation failed')
  105. raise ValidateFailedError('Validation failed, error: {}.'.format(str(ex)))
  106. def get_encrypted_token(self, config: Union[dict | str]):
  107. """
  108. Returns the encrypted token.
  109. """
  110. return json.dumps({
  111. 'openai_api_type': 'azure',
  112. 'openai_api_version': '2023-03-15-preview',
  113. 'openai_api_base': config['openai_api_base'],
  114. 'openai_api_key': self.encrypt_token(config['openai_api_key'])
  115. })
  116. def get_decrypted_token(self, token: str):
  117. """
  118. Returns the decrypted token.
  119. """
  120. config = json.loads(token)
  121. config['openai_api_key'] = self.decrypt_token(config['openai_api_key'])
  122. return config
  123. class AzureAuthenticationError(Exception):
  124. pass
  125. class AzureRequestFailedError(Exception):
  126. pass