| 'use client' | |
| import { | |
| useEffect, | |
| useState, | |
| } from 'react' | |
| import { useTranslation } from 'react-i18next' | |
| import { PlusIcon, XMarkIcon } from '@heroicons/react/20/solid' | |
| import useSWR, { useSWRConfig } from 'swr' | |
| import copy from 'copy-to-clipboard' | |
| import SecretKeyGenerateModal from './secret-key-generate' | |
| import s from './style.module.css' | |
| import Modal from '@/app/components/base/modal' | |
| import Button from '@/app/components/base/button' | |
| import { | |
| createApikey as createAppApikey, | |
| delApikey as delAppApikey, | |
| fetchApiKeysList as fetchAppApiKeysList, | |
| } from '@/service/apps' | |
| import { | |
| createApikey as createDatasetApikey, | |
| delApikey as delDatasetApikey, | |
| fetchApiKeysList as fetchDatasetApiKeysList, | |
| } from '@/service/datasets' | |
| import type { CreateApiKeyResponse } from '@/models/app' | |
| import Tooltip from '@/app/components/base/tooltip' | |
| import Loading from '@/app/components/base/loading' | |
| import Confirm from '@/app/components/base/confirm' | |
| import useTimestamp from '@/hooks/use-timestamp' | |
| import { useAppContext } from '@/context/app-context' | |
| type ISecretKeyModalProps = { | |
| isShow: boolean | |
| appId?: string | |
| onClose: () => void | |
| } | |
| const SecretKeyModal = ({ | |
| isShow = false, | |
| appId, | |
| onClose, | |
| }: ISecretKeyModalProps) => { | |
| const { t } = useTranslation() | |
| const { formatTime } = useTimestamp() | |
| const { currentWorkspace, isCurrentWorkspaceManager, isCurrentWorkspaceEditor } = useAppContext() | |
| const [showConfirmDelete, setShowConfirmDelete] = useState(false) | |
| const [isVisible, setVisible] = useState(false) | |
| const [newKey, setNewKey] = useState<CreateApiKeyResponse | undefined>(undefined) | |
| const { mutate } = useSWRConfig() | |
| const commonParams = appId | |
| ? { url: `/apps/${appId}/api-keys`, params: {} } | |
| : { url: '/datasets/api-keys', params: {} } | |
| const fetchApiKeysList = appId ? fetchAppApiKeysList : fetchDatasetApiKeysList | |
| const { data: apiKeysList } = useSWR(commonParams, fetchApiKeysList) | |
| const [delKeyID, setDelKeyId] = useState('') | |
| const [copyValue, setCopyValue] = useState('') | |
| useEffect(() => { | |
| if (copyValue) { | |
| const timeout = setTimeout(() => { | |
| setCopyValue('') | |
| }, 1000) | |
| return () => { | |
| clearTimeout(timeout) | |
| } | |
| } | |
| }, [copyValue]) | |
| const onDel = async () => { | |
| setShowConfirmDelete(false) | |
| if (!delKeyID) | |
| return | |
| const delApikey = appId ? delAppApikey : delDatasetApikey | |
| const params = appId | |
| ? { url: `/apps/${appId}/api-keys/${delKeyID}`, params: {} } | |
| : { url: `/datasets/api-keys/${delKeyID}`, params: {} } | |
| await delApikey(params) | |
| mutate(commonParams) | |
| } | |
| const onCreate = async () => { | |
| const params = appId | |
| ? { url: `/apps/${appId}/api-keys`, body: {} } | |
| : { url: '/datasets/api-keys', body: {} } | |
| const createApikey = appId ? createAppApikey : createDatasetApikey | |
| const res = await createApikey(params) | |
| setVisible(true) | |
| setNewKey(res) | |
| mutate(commonParams) | |
| } | |
| const generateToken = (token: string) => { | |
| return `${token.slice(0, 3)}...${token.slice(-20)}` | |
| } | |
| return ( | |
| <Modal isShow={isShow} onClose={onClose} title={`${t('appApi.apiKeyModal.apiSecretKey')}`} className={`${s.customModal} px-8 flex flex-col`}> | |
| <XMarkIcon className={`w-6 h-6 absolute cursor-pointer text-gray-500 ${s.close}`} onClick={onClose} /> | |
| <p className='mt-1 text-[13px] text-gray-500 font-normal leading-5 flex-shrink-0'>{t('appApi.apiKeyModal.apiSecretKeyTips')}</p> | |
| {!apiKeysList && <div className='mt-4'><Loading /></div>} | |
| { | |
| !!apiKeysList?.data?.length && ( | |
| <div className='flex flex-col flex-grow mt-4 overflow-hidden'> | |
| <div className='flex items-center flex-shrink-0 text-xs font-semibold text-gray-500 border-b border-solid h-9'> | |
| <div className='flex-shrink-0 w-64 px-3'>{t('appApi.apiKeyModal.secretKey')}</div> | |
| <div className='flex-shrink-0 px-3 w-[200px]'>{t('appApi.apiKeyModal.created')}</div> | |
| <div className='flex-shrink-0 px-3 w-[200px]'>{t('appApi.apiKeyModal.lastUsed')}</div> | |
| <div className='flex-grow px-3'></div> | |
| </div> | |
| <div className='flex-grow overflow-auto'> | |
| {apiKeysList.data.map(api => ( | |
| <div className='flex items-center text-sm font-normal text-gray-700 border-b border-solid h-9' key={api.id}> | |
| <div className='flex-shrink-0 w-64 px-3 font-mono truncate'>{generateToken(api.token)}</div> | |
| <div className='flex-shrink-0 px-3 truncate w-[200px]'>{formatTime(Number(api.created_at), t('appLog.dateTimeFormat') as string)}</div> | |
| <div className='flex-shrink-0 px-3 truncate w-[200px]'>{api.last_used_at ? formatTime(Number(api.last_used_at), t('appLog.dateTimeFormat') as string) : t('appApi.never')}</div> | |
| <div className='flex flex-grow px-3'> | |
| <Tooltip | |
| popupContent={copyValue === api.token ? `${t('appApi.copied')}` : `${t('appApi.copy')}`} | |
| popupClassName='mr-1' | |
| > | |
| <div className={`flex items-center justify-center flex-shrink-0 w-6 h-6 mr-1 rounded-lg cursor-pointer hover:bg-gray-100 ${s.copyIcon} ${copyValue === api.token ? s.copied : ''}`} onClick={() => { | |
| // setIsCopied(true) | |
| copy(api.token) | |
| setCopyValue(api.token) | |
| }}></div> | |
| </Tooltip> | |
| {isCurrentWorkspaceManager | |
| && <div className={`flex items-center justify-center flex-shrink-0 w-6 h-6 rounded-lg cursor-pointer ${s.trashIcon}`} onClick={() => { | |
| setDelKeyId(api.id) | |
| setShowConfirmDelete(true) | |
| }}> | |
| </div> | |
| } | |
| </div> | |
| </div> | |
| ))} | |
| </div> | |
| </div> | |
| ) | |
| } | |
| <div className='flex'> | |
| <Button className={`flex flex-shrink-0 mt-4 ${s.autoWidth}`} onClick={onCreate} disabled={!currentWorkspace || !isCurrentWorkspaceEditor}> | |
| <PlusIcon className='flex flex-shrink-0 w-4 h-4' /> | |
| <div className='text-xs font-medium text-gray-800'>{t('appApi.apiKeyModal.createNewSecretKey')}</div> | |
| </Button> | |
| </div> | |
| <SecretKeyGenerateModal className='flex-shrink-0' isShow={isVisible} onClose={() => setVisible(false)} newKey={newKey} /> | |
| {showConfirmDelete && ( | |
| <Confirm | |
| title={`${t('appApi.actionMsg.deleteConfirmTitle')}`} | |
| content={`${t('appApi.actionMsg.deleteConfirmTips')}`} | |
| isShow={showConfirmDelete} | |
| onConfirm={onDel} | |
| onCancel={() => { | |
| setDelKeyId('') | |
| setShowConfirmDelete(false) | |
| }} | |
| /> | |
| )} | |
| </Modal > | |
| ) | |
| } | |
| export default SecretKeyModal | |