Files
urbanLifeline/dify/web/app/components/datasets/extra-info/service-api/card.tsx

144 lines
5.4 KiB
TypeScript
Raw Normal View History

2025-12-01 17:21:38 +08:00
import React, { useCallback, useState } from 'react'
import { useTranslation } from 'react-i18next'
import { ApiAggregate } from '@/app/components/base/icons/src/vender/knowledge'
import Indicator from '@/app/components/header/indicator'
import Switch from '@/app/components/base/switch'
import { useSelector as useAppContextSelector } from '@/context/app-context'
import cn from '@/utils/classnames'
import CopyFeedback from '@/app/components/base/copy-feedback'
import Button from '@/app/components/base/button'
import { RiBookOpenLine, RiKey2Line } from '@remixicon/react'
import { useDisableDatasetServiceApi, useEnableDatasetServiceApi } from '@/service/knowledge/use-dataset'
import { useDatasetDetailContextWithSelector } from '@/context/dataset-detail'
import Link from 'next/link'
import SecretKeyModal from '@/app/components/develop/secret-key/secret-key-modal'
import { useDatasetApiAccessUrl } from '@/hooks/use-api-access-url'
type CardProps = {
apiEnabled: boolean
apiBaseUrl: string
}
const Card = ({
apiEnabled,
apiBaseUrl,
}: CardProps) => {
const { t } = useTranslation()
const datasetId = useDatasetDetailContextWithSelector(state => state.dataset?.id)
const mutateDatasetRes = useDatasetDetailContextWithSelector(state => state.mutateDatasetRes)
const { mutateAsync: enableDatasetServiceApi } = useEnableDatasetServiceApi()
const { mutateAsync: disableDatasetServiceApi } = useDisableDatasetServiceApi()
const [isSecretKeyModalVisible, setIsSecretKeyModalVisible] = useState(false)
const isCurrentWorkspaceManager = useAppContextSelector(state => state.isCurrentWorkspaceManager)
const apiReferenceUrl = useDatasetApiAccessUrl()
const onToggle = useCallback(async (state: boolean) => {
let result: 'success' | 'fail'
if (state)
result = (await enableDatasetServiceApi(datasetId ?? '')).result
else
result = (await disableDatasetServiceApi(datasetId ?? '')).result
if (result === 'success')
mutateDatasetRes?.()
}, [datasetId, enableDatasetServiceApi, disableDatasetServiceApi])
const handleOpenSecretKeyModal = useCallback(() => {
setIsSecretKeyModalVisible(true)
}, [])
const handleCloseSecretKeyModal = useCallback(() => {
setIsSecretKeyModalVisible(false)
}, [])
return (
<div className='flex w-[360px] flex-col rounded-xl border border-components-panel-border bg-components-panel-bg shadow-lg shadow-shadow-shadow-1'>
<div className='flex flex-col gap-y-3 p-4'>
<div className='flex items-center gap-x-3'>
<div className='flex grow items-center gap-x-2'>
<div className='flex size-6 shrink-0 items-center justify-center rounded-lg border-[0.5px] border-divider-subtle bg-util-colors-blue-brand-blue-brand-500 shadow-md shadow-shadow-shadow-5'>
<ApiAggregate className='size-4 text-text-primary-on-surface' />
</div>
<div className='system-sm-semibold grow truncate text-text-secondary'>
{t('dataset.serviceApi.card.title')}
</div>
</div>
<div className='flex items-center gap-x-1'>
<Indicator
className='shrink-0'
color={apiEnabled ? 'green' : 'yellow'}
/>
<div
className={cn(
'system-xs-semibold-uppercase',
apiEnabled ? 'text-text-success' : 'text-text-warning',
)}
>
{apiEnabled
? t('dataset.serviceApi.enabled')
: t('dataset.serviceApi.disabled')}
</div>
</div>
<Switch
defaultValue={apiEnabled}
onChange={onToggle}
disabled={!isCurrentWorkspaceManager}
/>
</div>
<div className='flex flex-col'>
<div className='system-xs-regular leading-6 text-text-tertiary'>
{t('dataset.serviceApi.card.endpoint')}
</div>
<div className='flex h-8 items-center gap-0.5 rounded-lg bg-components-input-bg-normal p-1 pl-2'>
<div className='flex h-4 min-w-0 flex-1 items-start justify-start gap-2 px-1'>
<div className='system-xs-medium truncate text-text-secondary'>
{apiBaseUrl}
</div>
</div>
<CopyFeedback
content={apiBaseUrl}
/>
</div>
</div>
</div>
{/* Actions */}
<div className='flex gap-x-1 border-t-[0.5px] border-divider-subtle p-4'>
<Button
variant='ghost'
size='small'
className='gap-x-px text-text-tertiary'
onClick={handleOpenSecretKeyModal}
>
<RiKey2Line className='size-3.5 shrink-0' />
<span className='system-xs-medium px-[3px]'>
{t('dataset.serviceApi.card.apiKey')}
</span>
</Button>
<Link
href={apiReferenceUrl}
target='_blank'
rel='noopener noreferrer'
>
<Button
variant='ghost'
size='small'
className='gap-x-px text-text-tertiary'
>
<RiBookOpenLine className='size-3.5 shrink-0' />
<span className='system-xs-medium px-[3px]'>
{t('dataset.serviceApi.card.apiReference')}
</span>
</Button>
</Link>
</div>
<SecretKeyModal
isShow={isSecretKeyModalVisible}
onClose={handleCloseSecretKeyModal}
/>
</div>
)
}
export default React.memo(Card)