index.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. import { useState } from 'react'
  2. import useSWR from 'swr'
  3. import { useTranslation } from 'react-i18next'
  4. import { useContext } from 'use-context-selector'
  5. import type {
  6. BackendModel,
  7. FormValue,
  8. ProviderConfigModal,
  9. ProviderEnum,
  10. } from './declarations'
  11. import ModelSelector from './model-selector'
  12. import ModelCard from './model-card'
  13. import ModelItem from './model-item'
  14. import ModelModal from './model-modal'
  15. import config from './configs'
  16. import { ConfigurableProviders } from './utils'
  17. import { ChevronDownDouble } from '@/app/components/base/icons/src/vender/line/arrows'
  18. import { HelpCircle } from '@/app/components/base/icons/src/vender/line/general'
  19. import {
  20. changeModelProviderPriority,
  21. deleteModelProvider,
  22. deleteModelProviderModel,
  23. fetchDefaultModal,
  24. fetchModelProviders,
  25. setModelProvider,
  26. updateDefaultModel,
  27. } from '@/service/common'
  28. import { useToastContext } from '@/app/components/base/toast'
  29. import Confirm from '@/app/components/base/confirm/common'
  30. import { ModelType } from '@/app/components/header/account-setting/model-page/declarations'
  31. import { useEventEmitterContextContext } from '@/context/event-emitter'
  32. import { useProviderContext } from '@/context/provider-context'
  33. import Tooltip from '@/app/components/base/tooltip'
  34. import I18n from '@/context/i18n'
  35. const MODEL_CARD_LIST = [
  36. config.openai,
  37. config.anthropic,
  38. ]
  39. const titleClassName = `
  40. flex items-center h-9 text-sm font-medium text-gray-900
  41. `
  42. const tipClassName = `
  43. ml-0.5 w-[14px] h-[14px] text-gray-400
  44. `
  45. type DeleteModel = {
  46. model_name: string
  47. model_type: string
  48. }
  49. const ModelPage = () => {
  50. const { t } = useTranslation()
  51. const { locale } = useContext(I18n)
  52. const {
  53. updateModelList,
  54. embeddingsDefaultModel,
  55. mutateEmbeddingsDefaultModel,
  56. speech2textDefaultModel,
  57. mutateSpeech2textDefaultModel,
  58. } = useProviderContext()
  59. const { data: providers, mutate: mutateProviders } = useSWR('/workspaces/current/model-providers', fetchModelProviders)
  60. const { data: textGenerationDefaultModel, mutate: mutateTextGenerationDefaultModel } = useSWR('/workspaces/current/default-model?model_type=text-generation', fetchDefaultModal)
  61. const [showMoreModel, setShowMoreModel] = useState(false)
  62. const [showModal, setShowModal] = useState(false)
  63. const { notify } = useToastContext()
  64. const { eventEmitter } = useEventEmitterContextContext()
  65. const [modelModalConfig, setModelModalConfig] = useState<ProviderConfigModal | undefined>(undefined)
  66. const [confirmShow, setConfirmShow] = useState(false)
  67. const [deleteModel, setDeleteModel] = useState<DeleteModel & { providerKey: ProviderEnum }>()
  68. const [modalMode, setModalMode] = useState('add')
  69. let modelList = []
  70. if (locale === 'en') {
  71. modelList = [
  72. config.azure_openai,
  73. config.replicate,
  74. config.huggingface_hub,
  75. config.minimax,
  76. config.spark,
  77. config.tongyi,
  78. config.wenxin,
  79. config.chatglm,
  80. config.xinference,
  81. config.openllm,
  82. config.localai,
  83. ]
  84. }
  85. else {
  86. modelList = [
  87. config.huggingface_hub,
  88. config.minimax,
  89. config.spark,
  90. config.azure_openai,
  91. config.replicate,
  92. config.tongyi,
  93. config.wenxin,
  94. config.chatglm,
  95. config.xinference,
  96. config.openllm,
  97. config.localai,
  98. ]
  99. }
  100. const handleOpenModal = (newModelModalConfig: ProviderConfigModal | undefined, editValue?: FormValue) => {
  101. if (newModelModalConfig) {
  102. setShowModal(true)
  103. const defaultValue = editValue ? { ...newModelModalConfig.defaultValue, ...editValue } : newModelModalConfig.defaultValue
  104. setModelModalConfig({
  105. ...newModelModalConfig,
  106. defaultValue,
  107. })
  108. if (editValue)
  109. setModalMode('edit')
  110. else
  111. setModalMode('add')
  112. }
  113. }
  114. const handleCancelModal = () => {
  115. setShowModal(false)
  116. }
  117. const handleUpdateProvidersAndModelList = () => {
  118. updateModelList(ModelType.textGeneration)
  119. updateModelList(ModelType.embeddings)
  120. updateModelList(ModelType.speech2text)
  121. mutateProviders()
  122. }
  123. const handleSave = async (originValue?: FormValue) => {
  124. if (originValue && modelModalConfig) {
  125. const v = modelModalConfig.filterValue ? modelModalConfig.filterValue(originValue) : originValue
  126. let body, url
  127. if (ConfigurableProviders.includes(modelModalConfig.key)) {
  128. const { model_name, model_type, ...config } = v
  129. body = {
  130. model_name,
  131. model_type,
  132. config,
  133. }
  134. url = `/workspaces/current/model-providers/${modelModalConfig.key}/models`
  135. }
  136. else {
  137. body = {
  138. config: v,
  139. }
  140. url = `/workspaces/current/model-providers/${modelModalConfig.key}`
  141. }
  142. try {
  143. eventEmitter?.emit('provider-save')
  144. const res = await setModelProvider({ url, body })
  145. if (res.result === 'success') {
  146. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  147. handleUpdateProvidersAndModelList()
  148. handleCancelModal()
  149. }
  150. eventEmitter?.emit('')
  151. }
  152. catch (e) {
  153. eventEmitter?.emit('')
  154. }
  155. }
  156. }
  157. const handleConfirm = (deleteModel: DeleteModel, providerKey: ProviderEnum) => {
  158. setDeleteModel({ ...deleteModel, providerKey })
  159. setConfirmShow(true)
  160. }
  161. const handleOperate = async ({ type, value }: Record<string, any>, provierKey: ProviderEnum) => {
  162. if (type === 'delete') {
  163. if (!value) {
  164. const res = await deleteModelProvider({ url: `/workspaces/current/model-providers/${provierKey}` })
  165. if (res.result === 'success') {
  166. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  167. handleUpdateProvidersAndModelList()
  168. }
  169. }
  170. else {
  171. handleConfirm(value, provierKey)
  172. }
  173. }
  174. if (type === 'priority') {
  175. const res = await changeModelProviderPriority({
  176. url: `/workspaces/current/model-providers/${provierKey}/preferred-provider-type`,
  177. body: {
  178. preferred_provider_type: value,
  179. },
  180. })
  181. if (res.result === 'success') {
  182. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  183. mutateProviders()
  184. }
  185. }
  186. }
  187. const handleDeleteModel = async () => {
  188. const { model_name, model_type, providerKey } = deleteModel || {}
  189. const res = await deleteModelProviderModel({
  190. url: `/workspaces/current/model-providers/${providerKey}/models?model_name=${model_name}&model_type=${model_type}`,
  191. })
  192. if (res.result === 'success') {
  193. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  194. setConfirmShow(false)
  195. handleUpdateProvidersAndModelList()
  196. }
  197. }
  198. const mutateDefaultModel = (type: ModelType) => {
  199. if (type === ModelType.textGeneration)
  200. mutateTextGenerationDefaultModel()
  201. if (type === ModelType.embeddings)
  202. mutateEmbeddingsDefaultModel()
  203. if (type === ModelType.speech2text)
  204. mutateSpeech2textDefaultModel()
  205. }
  206. const handleChangeDefaultModel = async (type: ModelType, v: BackendModel) => {
  207. const res = await updateDefaultModel({
  208. url: '/workspaces/current/default-model',
  209. body: {
  210. model_type: type,
  211. provider_name: v.model_provider.provider_name,
  212. model_name: v.model_name,
  213. },
  214. })
  215. if (res.result === 'success') {
  216. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  217. mutateDefaultModel(type)
  218. }
  219. }
  220. return (
  221. <div className='relative pt-1 -mt-2'>
  222. <div className='grid grid-cols-3 gap-4 mb-5'>
  223. <div className='w-full'>
  224. <div className={titleClassName}>
  225. {t('common.modelProvider.systemReasoningModel.key')}
  226. <Tooltip
  227. selector='model-page-system-reasoning-model-tip'
  228. htmlContent={
  229. <div className='w-[261px] text-gray-500'>{t('common.modelProvider.systemReasoningModel.tip')}</div>
  230. }
  231. >
  232. <HelpCircle className={tipClassName} />
  233. </Tooltip>
  234. </div>
  235. <div>
  236. <ModelSelector
  237. value={textGenerationDefaultModel && { providerName: textGenerationDefaultModel.model_provider.provider_name, modelName: textGenerationDefaultModel.model_name }}
  238. modelType={ModelType.textGeneration}
  239. onChange={v => handleChangeDefaultModel(ModelType.textGeneration, v)}
  240. />
  241. </div>
  242. </div>
  243. <div className='w-full'>
  244. <div className={titleClassName}>
  245. {t('common.modelProvider.embeddingModel.key')}
  246. <Tooltip
  247. selector='model-page-system-embedding-model-tip'
  248. htmlContent={
  249. <div className='w-[261px] text-gray-500'>{t('common.modelProvider.embeddingModel.tip')}</div>
  250. }
  251. >
  252. <HelpCircle className={tipClassName} />
  253. </Tooltip>
  254. </div>
  255. <div>
  256. <ModelSelector
  257. value={embeddingsDefaultModel && { providerName: embeddingsDefaultModel.model_provider.provider_name, modelName: embeddingsDefaultModel.model_name }}
  258. modelType={ModelType.embeddings}
  259. onChange={v => handleChangeDefaultModel(ModelType.embeddings, v)}
  260. />
  261. </div>
  262. </div>
  263. <div className='w-full'>
  264. <div className={titleClassName}>
  265. {t('common.modelProvider.speechToTextModel.key')}
  266. <Tooltip
  267. selector='model-page-system-speechToText-model-tip'
  268. htmlContent={
  269. <div className='w-[261px] text-gray-500'>{t('common.modelProvider.speechToTextModel.tip')}</div>
  270. }
  271. >
  272. <HelpCircle className={tipClassName} />
  273. </Tooltip>
  274. </div>
  275. <div>
  276. <ModelSelector
  277. value={speech2textDefaultModel && { providerName: speech2textDefaultModel.model_provider.provider_name, modelName: speech2textDefaultModel.model_name }}
  278. modelType={ModelType.speech2text}
  279. onChange={v => handleChangeDefaultModel(ModelType.speech2text, v)}
  280. />
  281. </div>
  282. </div>
  283. </div>
  284. <div className='mb-5 h-[0.5px] bg-gray-100' />
  285. <div className='mb-3 text-sm font-medium text-gray-800'>{t('common.modelProvider.models')}</div>
  286. <div className='grid grid-cols-2 gap-4 mb-6'>
  287. {
  288. MODEL_CARD_LIST.map((model, index) => (
  289. <ModelCard
  290. key={index}
  291. modelItem={model.item}
  292. currentProvider={providers?.[model.item.key]}
  293. onOpenModal={editValue => handleOpenModal(model.modal, editValue)}
  294. onOperate={v => handleOperate(v, model.item.key)}
  295. />
  296. ))
  297. }
  298. </div>
  299. {
  300. modelList.slice(0, showMoreModel ? modelList.length : 3).map((model, index) => (
  301. <ModelItem
  302. key={index}
  303. modelItem={model.item}
  304. currentProvider={providers?.[model.item.key]}
  305. onOpenModal={editValue => handleOpenModal(model.modal, editValue)}
  306. onOperate={v => handleOperate(v, model.item.key)}
  307. onUpdate={mutateProviders}
  308. />
  309. ))
  310. }
  311. {
  312. !showMoreModel && (
  313. <div className='inline-flex items-center px-1 h-[26px] cursor-pointer' onClick={() => setShowMoreModel(true)}>
  314. <ChevronDownDouble className='mr-1 w-3 h-3 text-gray-500' />
  315. <div className='text-xs font-medium text-gray-500'>{t('common.modelProvider.showMoreModelProvider')}</div>
  316. </div>
  317. )
  318. }
  319. <ModelModal
  320. isShow={showModal}
  321. modelModal={modelModalConfig}
  322. onCancel={handleCancelModal}
  323. onSave={handleSave}
  324. mode={modalMode}
  325. />
  326. <Confirm
  327. isShow={confirmShow}
  328. onCancel={() => setConfirmShow(false)}
  329. title={deleteModel?.model_name || ''}
  330. desc={t('common.modelProvider.item.deleteDesc', { modelName: deleteModel?.model_name }) || ''}
  331. onConfirm={handleDeleteModel}
  332. />
  333. </div>
  334. )
  335. }
  336. export default ModelPage