| '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 |
| |