hooks.ts 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. import {
  2. useCallback,
  3. useEffect,
  4. useMemo,
  5. useState,
  6. } from 'react'
  7. import useSWR, { useSWRConfig } from 'swr'
  8. import { useContext } from 'use-context-selector'
  9. import type {
  10. CustomConfigrationModelFixedFields,
  11. DefaultModel,
  12. DefaultModelResponse,
  13. Model,
  14. } from './declarations'
  15. import {
  16. ConfigurateMethodEnum,
  17. ModelTypeEnum,
  18. } from './declarations'
  19. import { getModelRuntimeSupported } from '@/utils/language'
  20. import I18n from '@/context/i18n'
  21. import {
  22. fetchDefaultModal,
  23. fetchModelList,
  24. fetchModelProviderCredentials,
  25. fetchModelProviders,
  26. getPayUrl,
  27. submitFreeQuota,
  28. } from '@/service/common'
  29. import { useProviderContext } from '@/context/provider-context'
  30. type UseDefaultModelAndModelList = (
  31. defaultModel: DefaultModelResponse | undefined,
  32. modelList: Model[],
  33. ) => [DefaultModel | undefined, (model: DefaultModel) => void]
  34. export const useSystemDefaultModelAndModelList: UseDefaultModelAndModelList = (
  35. defaultModel,
  36. modelList,
  37. ) => {
  38. const currentDefaultModel = useMemo(() => {
  39. const currentProvider = modelList.find(provider => provider.provider === defaultModel?.provider.provider)
  40. const currentModel = currentProvider?.models.find(model => model.model === defaultModel?.model)
  41. const currentDefaultModel = currentProvider && currentModel && {
  42. model: currentModel.model,
  43. provider: currentProvider.provider,
  44. }
  45. return currentDefaultModel
  46. }, [defaultModel, modelList])
  47. const [defaultModelState, setDefaultModelState] = useState<DefaultModel | undefined>(currentDefaultModel)
  48. const handleDefaultModelChange = useCallback((model: DefaultModel) => {
  49. setDefaultModelState(model)
  50. }, [])
  51. useEffect(() => {
  52. setDefaultModelState(currentDefaultModel)
  53. }, [currentDefaultModel])
  54. return [defaultModelState, handleDefaultModelChange]
  55. }
  56. export const useLanguage = () => {
  57. const { locale } = useContext(I18n)
  58. return getModelRuntimeSupported(locale)
  59. }
  60. export const useProviderCrenditialsFormSchemasValue = (
  61. provider: string,
  62. configurateMethod: ConfigurateMethodEnum,
  63. configured?: boolean,
  64. currentCustomConfigrationModelFixedFields?: CustomConfigrationModelFixedFields,
  65. ) => {
  66. const { data: predefinedFormSchemasValue } = useSWR(
  67. (configurateMethod === ConfigurateMethodEnum.predefinedModel && configured)
  68. ? `/workspaces/current/model-providers/${provider}/credentials`
  69. : null,
  70. fetchModelProviderCredentials,
  71. )
  72. const { data: customFormSchemasValue } = useSWR(
  73. (configurateMethod === ConfigurateMethodEnum.customizableModel && currentCustomConfigrationModelFixedFields)
  74. ? `/workspaces/current/model-providers/${provider}/models/credentials?model=${currentCustomConfigrationModelFixedFields?.__model_name}&model_type=${currentCustomConfigrationModelFixedFields?.__model_type}`
  75. : null,
  76. fetchModelProviderCredentials,
  77. )
  78. const value = useMemo(() => {
  79. return configurateMethod === ConfigurateMethodEnum.predefinedModel
  80. ? predefinedFormSchemasValue?.credentials
  81. : customFormSchemasValue?.credentials
  82. ? {
  83. ...customFormSchemasValue?.credentials,
  84. ...currentCustomConfigrationModelFixedFields,
  85. }
  86. : undefined
  87. }, [
  88. configurateMethod,
  89. currentCustomConfigrationModelFixedFields,
  90. customFormSchemasValue?.credentials,
  91. predefinedFormSchemasValue?.credentials,
  92. ])
  93. return value
  94. }
  95. export type ModelTypeIndex = 1 | 2 | 3 | 4 | 5
  96. export const MODEL_TYPE_MAPS = {
  97. 1: ModelTypeEnum.textGeneration,
  98. 2: ModelTypeEnum.textEmbedding,
  99. 3: ModelTypeEnum.rerank,
  100. 4: ModelTypeEnum.speech2text,
  101. 5: ModelTypeEnum.tts,
  102. }
  103. export const useModelList = (type: ModelTypeIndex) => {
  104. const { data, mutate, isLoading } = useSWR(`/workspaces/current/models/model-types/${MODEL_TYPE_MAPS[type]}`, fetchModelList)
  105. return {
  106. data: data?.data || [],
  107. mutate,
  108. isLoading,
  109. }
  110. }
  111. export const useDefaultModel = (type: ModelTypeIndex) => {
  112. const { data, mutate, isLoading } = useSWR(`/workspaces/current/default-model?model_type=${MODEL_TYPE_MAPS[type]}`, fetchDefaultModal)
  113. return {
  114. data: data?.data,
  115. mutate,
  116. isLoading,
  117. }
  118. }
  119. export const useCurrentProviderAndModel = (modelList: Model[], defaultModel?: DefaultModel) => {
  120. const currentProvider = modelList.find(provider => provider.provider === defaultModel?.provider)
  121. const currentModel = currentProvider?.models.find(model => model.model === defaultModel?.model)
  122. return {
  123. currentProvider,
  124. currentModel,
  125. }
  126. }
  127. export const useTextGenerationCurrentProviderAndModelAndModelList = (defaultModel?: DefaultModel) => {
  128. const { textGenerationModelList } = useProviderContext()
  129. const {
  130. currentProvider,
  131. currentModel,
  132. } = useCurrentProviderAndModel(textGenerationModelList, defaultModel)
  133. return {
  134. currentProvider,
  135. currentModel,
  136. textGenerationModelList,
  137. }
  138. }
  139. export const useAgentThoughtCurrentProviderAndModelAndModelList = (defaultModel?: DefaultModel) => {
  140. const { agentThoughtModelList } = useProviderContext()
  141. const {
  142. currentProvider,
  143. currentModel,
  144. } = useCurrentProviderAndModel(agentThoughtModelList, defaultModel)
  145. return {
  146. currentProvider,
  147. currentModel,
  148. agentThoughtModelList,
  149. }
  150. }
  151. export const useModelListAndDefaultModel = (type: ModelTypeIndex) => {
  152. const { data: modelList } = useModelList(type)
  153. const { data: defaultModel } = useDefaultModel(type)
  154. return {
  155. modelList,
  156. defaultModel,
  157. }
  158. }
  159. export const useModelListAndDefaultModelAndCurrentProviderAndModel = (type: ModelTypeIndex) => {
  160. const { modelList, defaultModel } = useModelListAndDefaultModel(type)
  161. const { currentProvider, currentModel } = useCurrentProviderAndModel(
  162. modelList,
  163. { provider: defaultModel?.provider.provider || '', model: defaultModel?.model || '' },
  164. )
  165. return {
  166. modelList,
  167. defaultModel,
  168. currentProvider,
  169. currentModel,
  170. }
  171. }
  172. export const useUpdateModelList = () => {
  173. const { mutate } = useSWRConfig()
  174. const updateModelList = useCallback((type: ModelTypeIndex | ModelTypeEnum) => {
  175. const modelType = typeof type === 'number' ? MODEL_TYPE_MAPS[type] : type
  176. mutate(`/workspaces/current/models/model-types/${modelType}`)
  177. }, [mutate])
  178. return updateModelList
  179. }
  180. export const useAnthropicBuyQuota = () => {
  181. const [loading, setLoading] = useState(false)
  182. const handleGetPayUrl = async () => {
  183. if (loading)
  184. return
  185. setLoading(true)
  186. try {
  187. const res = await getPayUrl('/workspaces/current/model-providers/anthropic/checkout-url')
  188. window.location.href = res.url
  189. }
  190. finally {
  191. setLoading(false)
  192. }
  193. }
  194. return handleGetPayUrl
  195. }
  196. export const useFreeQuota = (onSuccess: () => void) => {
  197. const [loading, setLoading] = useState(false)
  198. const handleClick = async (type: string) => {
  199. if (loading)
  200. return
  201. try {
  202. setLoading(true)
  203. const res = await submitFreeQuota(`/workspaces/current/model-providers/${type}/free-quota-submit`)
  204. if (res.type === 'redirect' && res.redirect_url)
  205. window.location.href = res.redirect_url
  206. else if (res.type === 'submit' && res.result === 'success')
  207. onSuccess()
  208. }
  209. finally {
  210. setLoading(false)
  211. }
  212. }
  213. return handleClick
  214. }
  215. export const useModelProviders = () => {
  216. const { data: providersData, mutate, isLoading } = useSWR('/workspaces/current/model-providers', fetchModelProviders)
  217. return {
  218. data: providersData?.data || [],
  219. mutate,
  220. isLoading,
  221. }
  222. }
  223. export const useUpdateModelProviders = () => {
  224. const { mutate } = useSWRConfig()
  225. const updateModelProviders = useCallback(() => {
  226. mutate('/workspaces/current/model-providers')
  227. }, [mutate])
  228. return updateModelProviders
  229. }