hooks.ts 7.1 KB

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