mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-30 18:33:30 +00:00 
			
		
		
		
	 403e2d58b9
			
		
	
	
		403e2d58b9
		
			
		
	
	
	
	
		
			
			Signed-off-by: yihong0618 <zouzou0208@gmail.com> Signed-off-by: -LAN- <laipz8200@outlook.com> Signed-off-by: xhe <xw897002528@gmail.com> Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: takatost <takatost@gmail.com> Co-authored-by: kurokobo <kuro664@gmail.com> Co-authored-by: Novice Lee <novicelee@NoviPro.local> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: AkaraChen <akarachen@outlook.com> Co-authored-by: Yi <yxiaoisme@gmail.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: JzoNg <jzongcode@gmail.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: Hiroshi Fujita <fujita-h@users.noreply.github.com> Co-authored-by: AkaraChen <85140972+AkaraChen@users.noreply.github.com> Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: Wu Tianwei <30284043+WTW0313@users.noreply.github.com> Co-authored-by: 非法操作 <hjlarry@163.com> Co-authored-by: Novice <857526207@qq.com> Co-authored-by: Hiroki Nagai <82458324+nagaihiroki-git@users.noreply.github.com> Co-authored-by: Gen Sato <52241300+halogen22@users.noreply.github.com> Co-authored-by: eux <euxuuu@gmail.com> Co-authored-by: huangzhuo1949 <167434202+huangzhuo1949@users.noreply.github.com> Co-authored-by: huangzhuo <huangzhuo1@xiaomi.com> Co-authored-by: lotsik <lotsik@mail.ru> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com> Co-authored-by: nite-knite <nkCoding@gmail.com> Co-authored-by: Jyong <76649700+JohnJyong@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: gakkiyomi <gakkiyomi@aliyun.com> Co-authored-by: CN-P5 <heibai2006@gmail.com> Co-authored-by: CN-P5 <heibai2006@qq.com> Co-authored-by: Chuehnone <1897025+chuehnone@users.noreply.github.com> Co-authored-by: yihong <zouzou0208@gmail.com> Co-authored-by: Kevin9703 <51311316+Kevin9703@users.noreply.github.com> Co-authored-by: -LAN- <laipz8200@outlook.com> Co-authored-by: Boris Feld <lothiraldan@gmail.com> Co-authored-by: mbo <himabo@gmail.com> Co-authored-by: mabo <mabo@aeyes.ai> Co-authored-by: Warren Chen <warren.chen830@gmail.com> Co-authored-by: JzoNgKVO <27049666+JzoNgKVO@users.noreply.github.com> Co-authored-by: jiandanfeng <chenjh3@wangsu.com> Co-authored-by: zhu-an <70234959+xhdd123321@users.noreply.github.com> Co-authored-by: zhaoqingyu.1075 <zhaoqingyu.1075@bytedance.com> Co-authored-by: 海狸大師 <86974027+yenslife@users.noreply.github.com> Co-authored-by: Xu Song <xusong.vip@gmail.com> Co-authored-by: rayshaw001 <396301947@163.com> Co-authored-by: Ding Jiatong <dingjiatong@gmail.com> Co-authored-by: Bowen Liang <liangbowen@gf.com.cn> Co-authored-by: JasonVV <jasonwangiii@outlook.com> Co-authored-by: le0zh <newlight@qq.com> Co-authored-by: zhuxinliang <zhuxinliang@didiglobal.com> Co-authored-by: k-zaku <zaku99@outlook.jp> Co-authored-by: luckylhb90 <luckylhb90@gmail.com> Co-authored-by: hobo.l <hobo.l@binance.com> Co-authored-by: jiangbo721 <365065261@qq.com> Co-authored-by: 刘江波 <jiangbo721@163.com> Co-authored-by: Shun Miyazawa <34241526+miya@users.noreply.github.com> Co-authored-by: EricPan <30651140+Egfly@users.noreply.github.com> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: sino <sino2322@gmail.com> Co-authored-by: Jhvcc <37662342+Jhvcc@users.noreply.github.com> Co-authored-by: lowell <lowell.hu@zkteco.in> Co-authored-by: Boris Polonsky <BorisPolonsky@users.noreply.github.com> Co-authored-by: Ademílson Tonato <ademilsonft@outlook.com> Co-authored-by: Ademílson Tonato <ademilson.tonato@refurbed.com> Co-authored-by: IWAI, Masaharu <iwaim.sub@gmail.com> Co-authored-by: Yueh-Po Peng (Yabi) <94939112+y10ab1@users.noreply.github.com> Co-authored-by: Jason <ggbbddjm@gmail.com> Co-authored-by: Xin Zhang <sjhpzx@gmail.com> Co-authored-by: yjc980121 <3898524+yjc980121@users.noreply.github.com> Co-authored-by: heyszt <36215648+hieheihei@users.noreply.github.com> Co-authored-by: Abdullah AlOsaimi <osaimiacc@gmail.com> Co-authored-by: Abdullah AlOsaimi <189027247+osaimi@users.noreply.github.com> Co-authored-by: Yingchun Lai <laiyingchun@apache.org> Co-authored-by: Hash Brown <hi@xzd.me> Co-authored-by: zuodongxu <192560071+zuodongxu@users.noreply.github.com> Co-authored-by: Masashi Tomooka <tmokmss@users.noreply.github.com> Co-authored-by: aplio <ryo.091219@gmail.com> Co-authored-by: Obada Khalili <54270856+obadakhalili@users.noreply.github.com> Co-authored-by: Nam Vu <zuzoovn@gmail.com> Co-authored-by: Kei YAMAZAKI <1715090+kei-yamazaki@users.noreply.github.com> Co-authored-by: TechnoHouse <13776377+deephbz@users.noreply.github.com> Co-authored-by: Riddhimaan-Senapati <114703025+Riddhimaan-Senapati@users.noreply.github.com> Co-authored-by: MaFee921 <31881301+2284730142@users.noreply.github.com> Co-authored-by: te-chan <t-nakanome@sakura-is.co.jp> Co-authored-by: HQidea <HQidea@users.noreply.github.com> Co-authored-by: Joshbly <36315710+Joshbly@users.noreply.github.com> Co-authored-by: xhe <xw897002528@gmail.com> Co-authored-by: weiwenyan-dev <154779315+weiwenyan-dev@users.noreply.github.com> Co-authored-by: ex_wenyan.wei <ex_wenyan.wei@tcl.com> Co-authored-by: engchina <12236799+engchina@users.noreply.github.com> Co-authored-by: engchina <atjapan2015@gmail.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: 呆萌闷油瓶 <253605712@qq.com> Co-authored-by: Kemal <kemalmeler@outlook.com> Co-authored-by: Lazy_Frog <4590648+lazyFrogLOL@users.noreply.github.com> Co-authored-by: Yi Xiao <54782454+YIXIAO0@users.noreply.github.com> Co-authored-by: Steven sun <98230804+Tuyohai@users.noreply.github.com> Co-authored-by: steven <sunzwj@digitalchina.com> Co-authored-by: Kalo Chin <91766386+fdb02983rhy@users.noreply.github.com> Co-authored-by: Katy Tao <34019945+KatyTao@users.noreply.github.com> Co-authored-by: depy <42985524+h4ckdepy@users.noreply.github.com> Co-authored-by: 胡春东 <gycm520@gmail.com> Co-authored-by: Junjie.M <118170653@qq.com> Co-authored-by: MuYu <mr.muzea@gmail.com> Co-authored-by: Naoki Takashima <39912547+takatea@users.noreply.github.com> Co-authored-by: Summer-Gu <37869445+gubinjie@users.noreply.github.com> Co-authored-by: Fei He <droxer.he@gmail.com> Co-authored-by: ybalbert001 <120714773+ybalbert001@users.noreply.github.com> Co-authored-by: Yuanbo Li <ybalbert@amazon.com> Co-authored-by: douxc <7553076+douxc@users.noreply.github.com> Co-authored-by: liuzhenghua <1090179900@qq.com> Co-authored-by: Wu Jiayang <62842862+Wu-Jiayang@users.noreply.github.com> Co-authored-by: Your Name <you@example.com> Co-authored-by: kimjion <45935338+kimjion@users.noreply.github.com> Co-authored-by: AugNSo <song.tiankai@icloud.com> Co-authored-by: llinvokerl <38915183+llinvokerl@users.noreply.github.com> Co-authored-by: liusurong.lsr <liusurong.lsr@alibaba-inc.com> Co-authored-by: Vasu Negi <vasu-negi@users.noreply.github.com> Co-authored-by: Hundredwz <1808096180@qq.com> Co-authored-by: Xiyuan Chen <52963600+GareArc@users.noreply.github.com>
		
			
				
	
	
		
			429 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			429 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 'use client'
 | |
| import React, { useCallback, useEffect, useState } from 'react'
 | |
| import { useTranslation } from 'react-i18next'
 | |
| import { useContext } from 'use-context-selector'
 | |
| import {
 | |
|   RiCloseLine,
 | |
| } from '@remixicon/react'
 | |
| import { AuthHeaderPrefix, AuthType, CollectionType } from '../types'
 | |
| import type { Collection, CustomCollectionBackend, Tool, WorkflowToolProviderRequest, WorkflowToolProviderResponse } from '../types'
 | |
| import ToolItem from './tool-item'
 | |
| import cn from '@/utils/classnames'
 | |
| import I18n from '@/context/i18n'
 | |
| import { getLanguage } from '@/i18n/language'
 | |
| import Confirm from '@/app/components/base/confirm'
 | |
| import Button from '@/app/components/base/button'
 | |
| import Indicator from '@/app/components/header/indicator'
 | |
| import { LinkExternal02, Settings01 } from '@/app/components/base/icons/src/vender/line/general'
 | |
| import Icon from '@/app/components/plugins/card/base/card-icon'
 | |
| import Title from '@/app/components/plugins/card/base/title'
 | |
| import OrgInfo from '@/app/components/plugins/card/base/org-info'
 | |
| import Description from '@/app/components/plugins/card/base/description'
 | |
| import ConfigCredential from '@/app/components/tools/setting/build-in/config-credentials'
 | |
| import EditCustomToolModal from '@/app/components/tools/edit-custom-collection-modal'
 | |
| import WorkflowToolModal from '@/app/components/tools/workflow-tool'
 | |
| import Toast from '@/app/components/base/toast'
 | |
| import Drawer from '@/app/components/base/drawer'
 | |
| import ActionButton from '@/app/components/base/action-button'
 | |
| 
 | |
| import {
 | |
|   deleteWorkflowTool,
 | |
|   fetchBuiltInToolList,
 | |
|   fetchCustomCollection,
 | |
|   fetchCustomToolList,
 | |
|   fetchModelToolList,
 | |
|   fetchWorkflowToolDetail,
 | |
|   removeBuiltInToolCredential,
 | |
|   removeCustomCollection,
 | |
|   saveWorkflowToolProvider,
 | |
|   updateBuiltInToolCredential,
 | |
|   updateCustomCollection,
 | |
| } from '@/service/tools'
 | |
| import { useModalContext } from '@/context/modal-context'
 | |
| import { useProviderContext } from '@/context/provider-context'
 | |
| import { ConfigurationMethodEnum } from '@/app/components/header/account-setting/model-provider-page/declarations'
 | |
| import Loading from '@/app/components/base/loading'
 | |
| import { useAppContext } from '@/context/app-context'
 | |
| import { useInvalidateAllWorkflowTools } from '@/service/use-tools'
 | |
| 
 | |
| type Props = {
 | |
|   collection: Collection
 | |
|   onHide: () => void
 | |
|   onRefreshData: () => void
 | |
| }
 | |
| 
 | |
| const ProviderDetail = ({
 | |
|   collection,
 | |
|   onHide,
 | |
|   onRefreshData,
 | |
| }: Props) => {
 | |
|   const { t } = useTranslation()
 | |
|   const { locale } = useContext(I18n)
 | |
|   const language = getLanguage(locale)
 | |
| 
 | |
|   const needAuth = collection.allow_delete || collection.type === CollectionType.model
 | |
|   const isAuthed = collection.is_team_authorization
 | |
|   const isBuiltIn = collection.type === CollectionType.builtIn
 | |
|   const isModel = collection.type === CollectionType.model
 | |
|   const { isCurrentWorkspaceManager } = useAppContext()
 | |
|   const invalidateAllWorkflowTools = useInvalidateAllWorkflowTools()
 | |
|   const [isDetailLoading, setIsDetailLoading] = useState(false)
 | |
| 
 | |
|   // built in provider
 | |
|   const [showSettingAuth, setShowSettingAuth] = useState(false)
 | |
|   const { setShowModelModal } = useModalContext()
 | |
|   const { modelProviders: providers } = useProviderContext()
 | |
|   const showSettingAuthModal = () => {
 | |
|     if (isModel) {
 | |
|       const provider = providers.find(item => item.provider === collection?.id)
 | |
|       if (provider) {
 | |
|         setShowModelModal({
 | |
|           payload: {
 | |
|             currentProvider: provider,
 | |
|             currentConfigurationMethod: ConfigurationMethodEnum.predefinedModel,
 | |
|             currentCustomConfigurationModelFixedFields: undefined,
 | |
|           },
 | |
|           onSaveCallback: () => {
 | |
|             onRefreshData()
 | |
|           },
 | |
|         })
 | |
|       }
 | |
|     }
 | |
|     else {
 | |
|       setShowSettingAuth(true)
 | |
|     }
 | |
|   }
 | |
|   // custom provider
 | |
|   const [customCollection, setCustomCollection] = useState<CustomCollectionBackend | WorkflowToolProviderResponse | null>(null)
 | |
|   const [isShowEditCollectionToolModal, setIsShowEditCustomCollectionModal] = useState(false)
 | |
|   const [showConfirmDelete, setShowConfirmDelete] = useState(false)
 | |
|   const [deleteAction, setDeleteAction] = useState('')
 | |
|   const doUpdateCustomToolCollection = async (data: CustomCollectionBackend) => {
 | |
|     await updateCustomCollection(data)
 | |
|     onRefreshData()
 | |
|     Toast.notify({
 | |
|       type: 'success',
 | |
|       message: t('common.api.actionSuccess'),
 | |
|     })
 | |
|     setIsShowEditCustomCollectionModal(false)
 | |
|   }
 | |
|   const doRemoveCustomToolCollection = async () => {
 | |
|     await removeCustomCollection(collection?.name as string)
 | |
|     onRefreshData()
 | |
|     Toast.notify({
 | |
|       type: 'success',
 | |
|       message: t('common.api.actionSuccess'),
 | |
|     })
 | |
|     setIsShowEditCustomCollectionModal(false)
 | |
|   }
 | |
|   const getCustomProvider = useCallback(async () => {
 | |
|     setIsDetailLoading(true)
 | |
|     const res = await fetchCustomCollection(collection.name)
 | |
|     if (res.credentials.auth_type === AuthType.apiKey && !res.credentials.api_key_header_prefix) {
 | |
|       if (res.credentials.api_key_value)
 | |
|         res.credentials.api_key_header_prefix = AuthHeaderPrefix.custom
 | |
|     }
 | |
|     setCustomCollection({
 | |
|       ...res,
 | |
|       labels: collection.labels,
 | |
|       provider: collection.name,
 | |
|     })
 | |
|     setIsDetailLoading(false)
 | |
|   }, [collection.labels, collection.name])
 | |
|   // workflow provider
 | |
|   const [isShowEditWorkflowToolModal, setIsShowEditWorkflowToolModal] = useState(false)
 | |
|   const getWorkflowToolProvider = useCallback(async () => {
 | |
|     setIsDetailLoading(true)
 | |
|     const res = await fetchWorkflowToolDetail(collection.id)
 | |
|     const payload = {
 | |
|       ...res,
 | |
|       parameters: res.tool?.parameters.map((item) => {
 | |
|         return {
 | |
|           name: item.name,
 | |
|           description: item.llm_description,
 | |
|           form: item.form,
 | |
|           required: item.required,
 | |
|           type: item.type,
 | |
|         }
 | |
|       }) || [],
 | |
|       labels: res.tool?.labels || [],
 | |
|     }
 | |
|     setCustomCollection(payload)
 | |
|     setIsDetailLoading(false)
 | |
|   }, [collection.id])
 | |
|   const removeWorkflowToolProvider = async () => {
 | |
|     await deleteWorkflowTool(collection.id)
 | |
|     onRefreshData()
 | |
|     Toast.notify({
 | |
|       type: 'success',
 | |
|       message: t('common.api.actionSuccess'),
 | |
|     })
 | |
|     setIsShowEditWorkflowToolModal(false)
 | |
|   }
 | |
|   const updateWorkflowToolProvider = async (data: WorkflowToolProviderRequest & Partial<{
 | |
|     workflow_app_id: string
 | |
|     workflow_tool_id: string
 | |
|   }>) => {
 | |
|     await saveWorkflowToolProvider(data)
 | |
|     invalidateAllWorkflowTools()
 | |
|     onRefreshData()
 | |
|     getWorkflowToolProvider()
 | |
|     Toast.notify({
 | |
|       type: 'success',
 | |
|       message: t('common.api.actionSuccess'),
 | |
|     })
 | |
|     setIsShowEditWorkflowToolModal(false)
 | |
|   }
 | |
|   const onClickCustomToolDelete = () => {
 | |
|     setDeleteAction('customTool')
 | |
|     setShowConfirmDelete(true)
 | |
|   }
 | |
|   const onClickWorkflowToolDelete = () => {
 | |
|     setDeleteAction('workflowTool')
 | |
|     setShowConfirmDelete(true)
 | |
|   }
 | |
|   const handleConfirmDelete = () => {
 | |
|     if (deleteAction === 'customTool')
 | |
|       doRemoveCustomToolCollection()
 | |
| 
 | |
|     else if (deleteAction === 'workflowTool')
 | |
|       removeWorkflowToolProvider()
 | |
| 
 | |
|     setShowConfirmDelete(false)
 | |
|   }
 | |
| 
 | |
|   // ToolList
 | |
|   const [toolList, setToolList] = useState<Tool[]>([])
 | |
|   const getProviderToolList = useCallback(async () => {
 | |
|     setIsDetailLoading(true)
 | |
|     try {
 | |
|       if (collection.type === CollectionType.builtIn) {
 | |
|         const list = await fetchBuiltInToolList(collection.name)
 | |
|         setToolList(list)
 | |
|       }
 | |
|       else if (collection.type === CollectionType.model) {
 | |
|         const list = await fetchModelToolList(collection.name)
 | |
|         setToolList(list)
 | |
|       }
 | |
|       else if (collection.type === CollectionType.workflow) {
 | |
|         setToolList([])
 | |
|       }
 | |
|       else {
 | |
|         const list = await fetchCustomToolList(collection.name)
 | |
|         setToolList(list)
 | |
|       }
 | |
|     }
 | |
|     catch (e) { }
 | |
|     setIsDetailLoading(false)
 | |
|   }, [collection.name, collection.type])
 | |
| 
 | |
|   useEffect(() => {
 | |
|     if (collection.type === CollectionType.custom)
 | |
|       getCustomProvider()
 | |
|     if (collection.type === CollectionType.workflow)
 | |
|       getWorkflowToolProvider()
 | |
|     getProviderToolList()
 | |
|   }, [collection.name, collection.type, getCustomProvider, getProviderToolList, getWorkflowToolProvider])
 | |
| 
 | |
|   return (
 | |
|     <Drawer
 | |
|       isOpen={!!collection}
 | |
|       clickOutsideNotOpen={false}
 | |
|       onClose={onHide}
 | |
|       footer={null}
 | |
|       mask={false}
 | |
|       positionCenter={false}
 | |
|       panelClassname={cn('justify-start mt-[64px] mr-2 mb-2 !w-[420px] !max-w-[420px] !p-0 !bg-components-panel-bg rounded-2xl border-[0.5px] border-components-panel-border shadow-xl')}
 | |
|     >
 | |
|       <div className='p-4'>
 | |
|         <div className='mb-3 flex'>
 | |
|           <Icon src={collection.icon} />
 | |
|           <div className="ml-3 w-0 grow">
 | |
|             <div className="flex items-center h-5">
 | |
|               <Title title={collection.label[language]} />
 | |
|             </div>
 | |
|             <div className='mb-1 flex justify-between items-center h-4'>
 | |
|               <OrgInfo
 | |
|                 className="mt-0.5"
 | |
|                 packageNameClassName='w-auto'
 | |
|                 orgName={collection.author}
 | |
|                 packageName={collection.name}
 | |
|               />
 | |
|             </div>
 | |
|           </div>
 | |
|           <div className='flex gap-1'>
 | |
|             <ActionButton onClick={onHide}>
 | |
|               <RiCloseLine className='w-4 h-4' />
 | |
|             </ActionButton>
 | |
|           </div>
 | |
|         </div>
 | |
|         {!!collection.description[language] && (
 | |
|           <Description text={collection.description[language]} descriptionLineRows={2}></Description>
 | |
|         )}
 | |
|         <div className='flex gap-1 border-b-[0.5px] border-divider-subtle'>
 | |
|           {collection.type === CollectionType.custom && !isDetailLoading && (
 | |
|             <Button
 | |
|               className={cn('shrink-0 my-3 w-full')}
 | |
|               onClick={() => setIsShowEditCustomCollectionModal(true)}
 | |
|             >
 | |
|               <Settings01 className='mr-1 w-4 h-4 text-text-tertiary' />
 | |
|               <div className='system-sm-medium text-text-secondary'>{t('tools.createTool.editAction')}</div>
 | |
|             </Button>
 | |
|           )}
 | |
|           {collection.type === CollectionType.workflow && !isDetailLoading && customCollection && (
 | |
|             <>
 | |
|               <Button
 | |
|                 variant='primary'
 | |
|                 className={cn('shrink-0 my-3 w-[183px]')}
 | |
|               >
 | |
|                 <a className='flex items-center text-text-primary' href={`/app/${(customCollection as WorkflowToolProviderResponse).workflow_app_id}/workflow`} rel='noreferrer' target='_blank'>
 | |
|                   <div className='system-sm-medium'>{t('tools.openInStudio')}</div>
 | |
|                   <LinkExternal02 className='ml-1 w-4 h-4' />
 | |
|                 </a>
 | |
|               </Button>
 | |
|               <Button
 | |
|                 className={cn('shrink-0 my-3 w-[183px]')}
 | |
|                 onClick={() => setIsShowEditWorkflowToolModal(true)}
 | |
|                 disabled={!isCurrentWorkspaceManager}
 | |
|               >
 | |
|                 <div className='system-sm-medium text-text-secondary'>{t('tools.createTool.editAction')}</div>
 | |
|               </Button>
 | |
|             </>
 | |
|           )}
 | |
|         </div>
 | |
|         {/* Tools */}
 | |
|         <div className='pt-3'>
 | |
|           {isDetailLoading && <div className='flex h-[200px]'><Loading type='app' /></div>}
 | |
|           {/* Builtin type */}
 | |
|           {!isDetailLoading && (collection.type === CollectionType.builtIn) && isAuthed && (
 | |
|             <div className='mb-1 h-6 flex items-center justify-between text-text-secondary system-sm-semibold-uppercase'>
 | |
|               {t('plugin.detailPanel.actionNum', { num: toolList.length, action: toolList.length > 1 ? 'actions' : 'action' })}
 | |
|               {needAuth && (
 | |
|                 <Button
 | |
|                   variant='secondary'
 | |
|                   size='small'
 | |
|                   onClick={() => {
 | |
|                     if (collection.type === CollectionType.builtIn || collection.type === CollectionType.model)
 | |
|                       showSettingAuthModal()
 | |
|                   }}
 | |
|                   disabled={!isCurrentWorkspaceManager}
 | |
|                 >
 | |
|                   <Indicator className='mr-2' color={'green'} />
 | |
|                   {t('tools.auth.authorized')}
 | |
|                 </Button>
 | |
|               )}
 | |
|             </div>
 | |
|           )}
 | |
|           {!isDetailLoading && (collection.type === CollectionType.builtIn) && needAuth && !isAuthed && (
 | |
|             <>
 | |
|               <div className='text-text-secondary system-sm-semibold-uppercase'>
 | |
|                 <span className=''>{t('tools.includeToolNum', { num: toolList.length, action: toolList.length > 1 ? 'actions' : 'action' }).toLocaleUpperCase()}</span>
 | |
|                 <span className='px-1'>·</span>
 | |
|                 <span className='text-util-colors-orange-orange-600'>{t('tools.auth.setup').toLocaleUpperCase()}</span>
 | |
|               </div>
 | |
|               <Button
 | |
|                 variant='primary'
 | |
|                 className={cn('shrink-0 my-3 w-full')}
 | |
|                 onClick={() => {
 | |
|                   if (collection.type === CollectionType.builtIn || collection.type === CollectionType.model)
 | |
|                     showSettingAuthModal()
 | |
|                 }}
 | |
|                 disabled={!isCurrentWorkspaceManager}
 | |
|               >
 | |
|                 {t('tools.auth.unauthorized')}
 | |
|               </Button>
 | |
|             </>
 | |
|           )}
 | |
|           {/* Custom type */}
 | |
|           {!isDetailLoading && (collection.type === CollectionType.custom) && (
 | |
|             <div className='text-text-secondary system-sm-semibold-uppercase'>
 | |
|               <span className=''>{t('tools.includeToolNum', { num: toolList.length }).toLocaleUpperCase()}</span>
 | |
|             </div>
 | |
|           )}
 | |
|           {/* Workflow type */}
 | |
|           {!isDetailLoading && (collection.type === CollectionType.workflow) && (
 | |
|             <div className='text-text-secondary system-sm-semibold-uppercase'>
 | |
|               <span className=''>{t('tools.createTool.toolInput.title').toLocaleUpperCase()}</span>
 | |
|             </div>
 | |
|           )}
 | |
|           {!isDetailLoading && (
 | |
|             <div className='mt-1 py-2'>
 | |
|               {collection.type !== CollectionType.workflow && toolList.map(tool => (
 | |
|                 <ToolItem
 | |
|                   key={tool.name}
 | |
|                   disabled={false}
 | |
|                   // disabled={needAuth && (isBuiltIn || isModel) && !isAuthed}
 | |
|                   collection={collection}
 | |
|                   tool={tool}
 | |
|                   isBuiltIn={isBuiltIn}
 | |
|                   isModel={isModel}
 | |
|                 />
 | |
|               ))}
 | |
|               {collection.type === CollectionType.workflow && (customCollection as WorkflowToolProviderResponse)?.tool?.parameters.map(item => (
 | |
|                 <div key={item.name} className='mb-1 py-1'>
 | |
|                   <div className='mb-1 flex items-center gap-2'>
 | |
|                     <span className='text-text-secondary code-sm-semibold'>{item.name}</span>
 | |
|                     <span className='text-text-tertiary system-xs-regular'>{item.type}</span>
 | |
|                     <span className='text-text-warning-secondary system-xs-medium'>{item.required ? t('tools.createTool.toolInput.required') : ''}</span>
 | |
|                   </div>
 | |
|                   <div className='text-text-tertiary system-xs-regular'>{item.llm_description}</div>
 | |
|                 </div>
 | |
|               ))}
 | |
|             </div>
 | |
|           )}
 | |
|         </div>
 | |
|         {showSettingAuth && (
 | |
|           <ConfigCredential
 | |
|             collection={collection}
 | |
|             onCancel={() => setShowSettingAuth(false)}
 | |
|             onSaved={async (value) => {
 | |
|               await updateBuiltInToolCredential(collection.name, value)
 | |
|               Toast.notify({
 | |
|                 type: 'success',
 | |
|                 message: t('common.api.actionSuccess'),
 | |
|               })
 | |
|               await onRefreshData()
 | |
|               setShowSettingAuth(false)
 | |
|             }}
 | |
|             onRemove={async () => {
 | |
|               await removeBuiltInToolCredential(collection.name)
 | |
|               Toast.notify({
 | |
|                 type: 'success',
 | |
|                 message: t('common.api.actionSuccess'),
 | |
|               })
 | |
|               await onRefreshData()
 | |
|               setShowSettingAuth(false)
 | |
|             }}
 | |
|           />
 | |
|         )}
 | |
|         {isShowEditCollectionToolModal && (
 | |
|           <EditCustomToolModal
 | |
|             payload={customCollection}
 | |
|             onHide={() => setIsShowEditCustomCollectionModal(false)}
 | |
|             onEdit={doUpdateCustomToolCollection}
 | |
|             onRemove={onClickCustomToolDelete}
 | |
|           />
 | |
|         )}
 | |
|         {isShowEditWorkflowToolModal && (
 | |
|           <WorkflowToolModal
 | |
|             payload={customCollection}
 | |
|             onHide={() => setIsShowEditWorkflowToolModal(false)}
 | |
|             onRemove={onClickWorkflowToolDelete}
 | |
|             onSave={updateWorkflowToolProvider}
 | |
|           />
 | |
|         )}
 | |
|         {showConfirmDelete && (
 | |
|           <Confirm
 | |
|             title={t('tools.createTool.deleteToolConfirmTitle')}
 | |
|             content={t('tools.createTool.deleteToolConfirmContent')}
 | |
|             isShow={showConfirmDelete}
 | |
|             onConfirm={handleConfirmDelete}
 | |
|             onCancel={() => setShowConfirmDelete(false)}
 | |
|           />
 | |
|         )}
 | |
|       </div>
 | |
|     </Drawer>
 | |
|   )
 | |
| }
 | |
| export default ProviderDetail
 |