| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377 | 'use client'import React, { useCallback, useEffect, useState } from 'react'import { useTranslation } from 'react-i18next'import { useContext } from 'use-context-selector'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 AppIcon from '@/app/components/base/app-icon'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 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 {  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'type Props = {  collection: Collection  onRefreshData: () => void}const ProviderDetail = ({  collection,  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 [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(null)  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)    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 (    <div className='px-6 py-3'>      <div className='flex items-center py-1 gap-2'>        <div className='relative shrink-0'>          {typeof collection.icon === 'string' && (            <div className='w-8 h-8 bg-center bg-cover bg-no-repeat rounded-md' style={{ backgroundImage: `url(${collection.icon})` }} />          )}          {typeof collection.icon !== 'string' && (            <AppIcon              size='small'              icon={collection.icon.content}              background={collection.icon.background}            />          )}        </div>        <div className='grow w-0 py-[1px]'>          <div className='flex items-center text-md leading-6 font-semibold text-gray-900'>            <div className='truncate' title={collection.label[language]}>{collection.label[language]}</div>          </div>        </div>      </div>      <div className='mt-2 min-h-[36px] text-gray-500 text-sm leading-[18px]'>{collection.description[language]}</div>      <div className='flex gap-1 border-b-[0.5px] border-black/5'>        {(collection.type === CollectionType.builtIn) && needAuth && (          <Button            variant={isAuthed ? 'secondary' : 'primary'}            className={cn('shrink-0 my-3 w-full', isAuthed && 'bg-white')}            onClick={() => {              if (collection.type === CollectionType.builtIn || collection.type === CollectionType.model)                showSettingAuthModal()            }}            disabled={!isCurrentWorkspaceManager}          >            {isAuthed && <Indicator className='mr-2' color={'green'} />}            <div className={cn('text-white leading-[18px] text-[13px] font-medium', isAuthed && '!text-gray-700')}>              {isAuthed ? t('tools.auth.authorized') : t('tools.auth.unauthorized')}            </div>          </Button>        )}        {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-gray-500' />            <div className='leading-5 text-sm font-medium text-gray-700'>{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-white' href={`/app/${(customCollection as WorkflowToolProviderResponse).workflow_app_id}/workflow`} rel='noreferrer' target='_blank'>                <div className='leading-5 text-sm font-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='leading-5 text-sm font-medium text-gray-700'>{t('tools.createTool.editAction')}</div>            </Button>          </>        )}      </div>      {/* Tools */}      <div className='pt-3'>        {isDetailLoading && <div className='flex h-[200px]'><Loading type='app' /></div>}        {!isDetailLoading && (          <div className='text-xs font-medium leading-6 text-gray-500'>            {collection.type === CollectionType.workflow && <span className=''>{t('tools.createTool.toolInput.title').toLocaleUpperCase()}</span>}            {collection.type !== CollectionType.workflow && <span className=''>{t('tools.includeToolNum', { num: toolList.length }).toLocaleUpperCase()}</span>}            {needAuth && (isBuiltIn || isModel) && !isAuthed && (              <>                <span className='px-1'>·</span>                <span className='text-[#DC6803]'>{t('tools.auth.setup').toLocaleUpperCase()}</span>              </>            )}          </div>        )}        {!isDetailLoading && (          <div className='mt-1'>            {collection.type !== CollectionType.workflow && toolList.map(tool => (              <ToolItem                key={tool.name}                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-2 px-4 py-3 rounded-xl bg-gray-25 border-[0.5px] border-gray-200'>                <div className='flex items-center gap-2'>                  <span className='font-medium text-sm text-gray-900'>{item.name}</span>                  <span className='text-xs leading-[18px] text-gray-500'>{item.type}</span>                  <span className='font-medium text-xs leading-[18px] text-[#ec4a0a]'>{item.required ? t('tools.createTool.toolInput.required') : ''}</span>                </div>                <div className='h-[18px] leading-[18px] text-gray-500 text-xs'>{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}          onClose={() => setShowConfirmDelete(false)}          onConfirm={handleConfirmDelete}          onCancel={() => setShowConfirmDelete(false)}        />      )}    </div>  )}export default ProviderDetail
 |