| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | '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' | 
					
						
							| 
									
										
										
										
											2023-08-04 10:49:13 +08:00
										 |  |  | import copy from 'copy-to-clipboard' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 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' | 
					
						
							| 
									
										
										
										
											2023-09-27 16:06:49 +08:00
										 |  |  | 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' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 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' | 
					
						
							| 
									
										
										
										
											2024-05-04 16:15:32 +08:00
										 |  |  | import useTimestamp from '@/hooks/use-timestamp' | 
					
						
							| 
									
										
										
										
											2023-08-15 13:35:47 +08:00
										 |  |  | import { useAppContext } from '@/context/app-context' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ISecretKeyModalProps = { | 
					
						
							|  |  |  |   isShow: boolean | 
					
						
							| 
									
										
										
										
											2023-09-27 16:06:49 +08:00
										 |  |  |   appId?: string | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   onClose: () => void | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const SecretKeyModal = ({ | 
					
						
							|  |  |  |   isShow = false, | 
					
						
							|  |  |  |   appId, | 
					
						
							|  |  |  |   onClose, | 
					
						
							|  |  |  | }: ISecretKeyModalProps) => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							| 
									
										
										
										
											2024-05-04 16:15:32 +08:00
										 |  |  |   const { formatTime } = useTimestamp() | 
					
						
							| 
									
										
										
										
											2024-09-11 16:07:15 +08:00
										 |  |  |   const { currentWorkspace, isCurrentWorkspaceManager, isCurrentWorkspaceEditor } = useAppContext() | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   const [showConfirmDelete, setShowConfirmDelete] = useState(false) | 
					
						
							|  |  |  |   const [isVisible, setVisible] = useState(false) | 
					
						
							|  |  |  |   const [newKey, setNewKey] = useState<CreateApiKeyResponse | undefined>(undefined) | 
					
						
							|  |  |  |   const { mutate } = useSWRConfig() | 
					
						
							| 
									
										
										
										
											2023-09-27 16:06:49 +08:00
										 |  |  |   const commonParams = appId | 
					
						
							|  |  |  |     ? { url: `/apps/${appId}/api-keys`, params: {} } | 
					
						
							|  |  |  |     : { url: '/datasets/api-keys', params: {} } | 
					
						
							|  |  |  |   const fetchApiKeysList = appId ? fetchAppApiKeysList : fetchDatasetApiKeysList | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |   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) | 
					
						
							| 
									
										
										
										
											2023-07-21 11:15:00 +08:00
										 |  |  |     if (!delKeyID) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       return | 
					
						
							| 
									
										
										
										
											2023-07-21 11:15:00 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-27 16:06:49 +08:00
										 |  |  |     const delApikey = appId ? delAppApikey : delDatasetApikey | 
					
						
							|  |  |  |     const params = appId | 
					
						
							|  |  |  |       ? { url: `/apps/${appId}/api-keys/${delKeyID}`, params: {} } | 
					
						
							|  |  |  |       : { url: `/datasets/api-keys/${delKeyID}`, params: {} } | 
					
						
							|  |  |  |     await delApikey(params) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     mutate(commonParams) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const onCreate = async () => { | 
					
						
							| 
									
										
										
										
											2023-09-27 16:06:49 +08:00
										 |  |  |     const params = appId | 
					
						
							|  |  |  |       ? { url: `/apps/${appId}/api-keys`, body: {} } | 
					
						
							|  |  |  |       : { url: '/datasets/api-keys', body: {} } | 
					
						
							|  |  |  |     const createApikey = appId ? createAppApikey : createDatasetApikey | 
					
						
							|  |  |  |     const res = await createApikey(params) | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |     setVisible(true) | 
					
						
							|  |  |  |     setNewKey(res) | 
					
						
							|  |  |  |     mutate(commonParams) | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const generateToken = (token: string) => { | 
					
						
							|  |  |  |     return `${token.slice(0, 3)}...${token.slice(-20)}` | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |     <Modal isShow={isShow} onClose={onClose} title={`${t('appApi.apiKeyModal.apiSecretKey')}`} className={`${s.customModal} flex flex-col px-8`}> | 
					
						
							|  |  |  |       <XMarkIcon className={`absolute h-6 w-6 cursor-pointer text-text-tertiary ${s.close}`} onClick={onClose} /> | 
					
						
							|  |  |  |       <p className='mt-1 shrink-0 text-[13px] font-normal leading-5 text-text-tertiary'>{t('appApi.apiKeyModal.apiSecretKeyTips')}</p> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       {!apiKeysList && <div className='mt-4'><Loading /></div>} | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         !!apiKeysList?.data?.length && ( | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |           <div className='mt-4 flex grow flex-col overflow-hidden'> | 
					
						
							|  |  |  |             <div className='flex h-9 shrink-0 items-center border-b border-solid text-xs font-semibold text-text-tertiary'> | 
					
						
							|  |  |  |               <div className='w-64 shrink-0 px-3'>{t('appApi.apiKeyModal.secretKey')}</div> | 
					
						
							|  |  |  |               <div className='w-[200px] shrink-0 px-3'>{t('appApi.apiKeyModal.created')}</div> | 
					
						
							|  |  |  |               <div className='w-[200px] shrink-0 px-3'>{t('appApi.apiKeyModal.lastUsed')}</div> | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |               <div className='grow px-3'></div> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |             </div> | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |             <div className='grow overflow-auto'> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |               {apiKeysList.data.map(api => ( | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                 <div className='flex h-9 items-center border-b border-solid text-sm font-normal text-text-secondary' key={api.id}> | 
					
						
							|  |  |  |                   <div className='w-64 shrink-0 truncate px-3 font-mono'>{generateToken(api.token)}</div> | 
					
						
							|  |  |  |                   <div className='w-[200px] shrink-0 truncate px-3'>{formatTime(Number(api.created_at), t('appLog.dateTimeFormat') as string)}</div> | 
					
						
							|  |  |  |                   <div className='w-[200px] shrink-0 truncate px-3'>{api.last_used_at ? formatTime(Number(api.last_used_at), t('appLog.dateTimeFormat') as string) : t('appApi.never')}</div> | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |                   <div className='flex grow px-3'> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                     <Tooltip | 
					
						
							| 
									
										
										
										
											2024-08-26 13:00:02 +08:00
										 |  |  |                       popupContent={copyValue === api.token ? `${t('appApi.copied')}` : `${t('appApi.copy')}`} | 
					
						
							|  |  |  |                       popupClassName='mr-1' | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                     > | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                       <div className={`mr-1 flex h-6 w-6 shrink-0 cursor-pointer items-center justify-center rounded-lg hover:bg-state-base-hover ${s.copyIcon} ${copyValue === api.token ? s.copied : ''}`} onClick={() => { | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                         // setIsCopied(true)
 | 
					
						
							|  |  |  |                         copy(api.token) | 
					
						
							|  |  |  |                         setCopyValue(api.token) | 
					
						
							|  |  |  |                       }}></div> | 
					
						
							|  |  |  |                     </Tooltip> | 
					
						
							| 
									
										
										
										
											2024-06-19 14:13:16 +08:00
										 |  |  |                     {isCurrentWorkspaceManager | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |                       && <div className={`flex h-6 w-6 shrink-0 cursor-pointer items-center justify-center rounded-lg ${s.trashIcon}`} onClick={() => { | 
					
						
							| 
									
										
										
										
											2023-08-15 13:35:47 +08:00
										 |  |  |                         setDelKeyId(api.id) | 
					
						
							|  |  |  |                         setShowConfirmDelete(true) | 
					
						
							|  |  |  |                       }}> | 
					
						
							|  |  |  |                       </div> | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |                   </div> | 
					
						
							|  |  |  |                 </div> | 
					
						
							|  |  |  |               ))} | 
					
						
							|  |  |  |             </div> | 
					
						
							|  |  |  |           </div> | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       <div className='flex'> | 
					
						
							| 
									
										
										
										
											2025-03-21 17:41:03 +08:00
										 |  |  |         <Button className={`mt-4 flex shrink-0 ${s.autoWidth}`} onClick={onCreate} disabled={!currentWorkspace || !isCurrentWorkspaceEditor}> | 
					
						
							|  |  |  |           <PlusIcon className='mr-1 flex h-4 w-4 shrink-0' /> | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |           <div className='text-xs font-medium text-text-secondary'>{t('appApi.apiKeyModal.createNewSecretKey')}</div> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |         </Button> | 
					
						
							|  |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |       <SecretKeyGenerateModal className='shrink-0' isShow={isVisible} onClose={() => setVisible(false)} newKey={newKey} /> | 
					
						
							| 
									
										
										
										
											2023-05-15 08:51:32 +08:00
										 |  |  |       {showConfirmDelete && ( | 
					
						
							|  |  |  |         <Confirm | 
					
						
							|  |  |  |           title={`${t('appApi.actionMsg.deleteConfirmTitle')}`} | 
					
						
							|  |  |  |           content={`${t('appApi.actionMsg.deleteConfirmTips')}`} | 
					
						
							|  |  |  |           isShow={showConfirmDelete} | 
					
						
							|  |  |  |           onConfirm={onDel} | 
					
						
							|  |  |  |           onCancel={() => { | 
					
						
							|  |  |  |             setDelKeyId('') | 
					
						
							|  |  |  |             setShowConfirmDelete(false) | 
					
						
							|  |  |  |           }} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       )} | 
					
						
							|  |  |  |     </Modal > | 
					
						
							|  |  |  |   ) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default SecretKeyModal |