|
'use client'
|
|
import type { FC } from 'react'
|
|
import React, { useState } from 'react'
|
|
import useSWR from 'swr'
|
|
import { ArrowLeftIcon } from '@heroicons/react/24/solid'
|
|
import { createContext, useContext } from 'use-context-selector'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { useRouter } from 'next/navigation'
|
|
import { omit } from 'lodash-es'
|
|
import cn from 'classnames'
|
|
import { OperationAction, StatusItem } from '../list'
|
|
import s from '../style.module.css'
|
|
import Completed from './completed'
|
|
import Embedding from './embedding'
|
|
import Metadata from './metadata'
|
|
import SegmentAdd, { ProcessStatus } from './segment-add'
|
|
import BatchModal from './batch-modal'
|
|
import style from './style.module.css'
|
|
import Divider from '@/app/components/base/divider'
|
|
import Loading from '@/app/components/base/loading'
|
|
import type { MetadataType } from '@/service/datasets'
|
|
import { checkSegmentBatchImportProgress, fetchDocumentDetail, segmentBatchImport } from '@/service/datasets'
|
|
import { ToastContext } from '@/app/components/base/toast'
|
|
import type { DocForm } from '@/models/datasets'
|
|
import { useDatasetDetailContext } from '@/context/dataset-detail'
|
|
import FloatRightContainer from '@/app/components/base/float-right-container'
|
|
import useBreakpoints, { MediaType } from '@/hooks/use-breakpoints'
|
|
|
|
export const DocumentContext = createContext<{ datasetId?: string; documentId?: string; docForm: string }>({ docForm: '' })
|
|
|
|
type DocumentTitleProps = {
|
|
extension?: string
|
|
name?: string
|
|
iconCls?: string
|
|
textCls?: string
|
|
wrapperCls?: string
|
|
}
|
|
|
|
export const DocumentTitle: FC<DocumentTitleProps> = ({ extension, name, iconCls, textCls, wrapperCls }) => {
|
|
const localExtension = extension?.toLowerCase() || name?.split('.')?.pop()?.toLowerCase()
|
|
return <div className={cn('flex items-center justify-start flex-1', wrapperCls)}>
|
|
<div className={cn(s[`${localExtension || 'txt'}Icon`], style.titleIcon, iconCls)}></div>
|
|
<span className={cn('font-semibold text-lg text-gray-900 ml-1', textCls)}> {name || '--'}</span>
|
|
</div>
|
|
}
|
|
|
|
type Props = {
|
|
datasetId: string
|
|
documentId: string
|
|
}
|
|
|
|
const DocumentDetail: FC<Props> = ({ datasetId, documentId }) => {
|
|
const router = useRouter()
|
|
const { t } = useTranslation()
|
|
|
|
const media = useBreakpoints()
|
|
const isMobile = media === MediaType.mobile
|
|
|
|
const { notify } = useContext(ToastContext)
|
|
const { dataset } = useDatasetDetailContext()
|
|
const embeddingAvailable = !!dataset?.embedding_available
|
|
const [showMetadata, setShowMetadata] = useState(!isMobile)
|
|
const [newSegmentModalVisible, setNewSegmentModalVisible] = useState(false)
|
|
const [batchModalVisible, setBatchModalVisible] = useState(false)
|
|
const [importStatus, setImportStatus] = useState<ProcessStatus | string>()
|
|
const showNewSegmentModal = () => setNewSegmentModalVisible(true)
|
|
const showBatchModal = () => setBatchModalVisible(true)
|
|
const hideBatchModal = () => setBatchModalVisible(false)
|
|
const resetProcessStatus = () => setImportStatus('')
|
|
const checkProcess = async (jobID: string) => {
|
|
try {
|
|
const res = await checkSegmentBatchImportProgress({ jobID })
|
|
setImportStatus(res.job_status)
|
|
if (res.job_status === ProcessStatus.WAITING || res.job_status === ProcessStatus.PROCESSING)
|
|
setTimeout(() => checkProcess(res.job_id), 2500)
|
|
if (res.job_status === ProcessStatus.ERROR)
|
|
notify({ type: 'error', message: `${t('datasetDocuments.list.batchModal.runError')}` })
|
|
}
|
|
catch (e: any) {
|
|
notify({ type: 'error', message: `${t('datasetDocuments.list.batchModal.runError')}${'message' in e ? `: ${e.message}` : ''}` })
|
|
}
|
|
}
|
|
const runBatch = async (csv: File) => {
|
|
const formData = new FormData()
|
|
formData.append('file', csv)
|
|
try {
|
|
const res = await segmentBatchImport({
|
|
url: `/datasets/${datasetId}/documents/${documentId}/segments/batch_import`,
|
|
body: formData,
|
|
})
|
|
setImportStatus(res.job_status)
|
|
checkProcess(res.job_id)
|
|
}
|
|
catch (e: any) {
|
|
notify({ type: 'error', message: `${t('datasetDocuments.list.batchModal.runError')}${'message' in e ? `: ${e.message}` : ''}` })
|
|
}
|
|
}
|
|
|
|
const { data: documentDetail, error, mutate: detailMutate } = useSWR({
|
|
action: 'fetchDocumentDetail',
|
|
datasetId,
|
|
documentId,
|
|
params: { metadata: 'without' as MetadataType },
|
|
}, apiParams => fetchDocumentDetail(omit(apiParams, 'action')))
|
|
|
|
const { data: documentMetadata, error: metadataErr, mutate: metadataMutate } = useSWR({
|
|
action: 'fetchDocumentDetail',
|
|
datasetId,
|
|
documentId,
|
|
params: { metadata: 'only' as MetadataType },
|
|
}, apiParams => fetchDocumentDetail(omit(apiParams, 'action')),
|
|
)
|
|
|
|
const backToPrev = () => {
|
|
router.push(`/datasets/${datasetId}/documents`)
|
|
}
|
|
|
|
const isDetailLoading = !documentDetail && !error
|
|
const isMetadataLoading = !documentMetadata && !metadataErr
|
|
|
|
const embedding = ['queuing', 'indexing', 'paused'].includes((documentDetail?.display_status || '').toLowerCase())
|
|
|
|
const handleOperate = (operateName?: string) => {
|
|
if (operateName === 'delete')
|
|
backToPrev()
|
|
else
|
|
detailMutate()
|
|
}
|
|
|
|
return (
|
|
<DocumentContext.Provider value={{ datasetId, documentId, docForm: documentDetail?.doc_form || '' }}>
|
|
<div className='flex flex-col h-full'>
|
|
<div className='flex min-h-16 border-b-gray-100 border-b items-center p-4 justify-between flex-wrap gap-y-2'>
|
|
<div onClick={backToPrev} className={'shrink-0 rounded-full w-8 h-8 flex justify-center items-center border-gray-100 cursor-pointer border hover:border-gray-300 shadow-[0px_12px_16px_-4px_rgba(16,24,40,0.08),0px_4px_6px_-2px_rgba(16,24,40,0.03)]'}>
|
|
<ArrowLeftIcon className='text-primary-600 fill-current stroke-current h-4 w-4' />
|
|
</div>
|
|
<Divider className='!h-4' type='vertical' />
|
|
<DocumentTitle extension={documentDetail?.data_source_info?.upload_file?.extension} name={documentDetail?.name} />
|
|
<div className='flex items-center flex-wrap gap-y-2'>
|
|
<StatusItem status={documentDetail?.display_status || 'available'} scene='detail' errorMessage={documentDetail?.error || ''} />
|
|
{embeddingAvailable && documentDetail && !documentDetail.archived && (
|
|
<SegmentAdd
|
|
importStatus={importStatus}
|
|
clearProcessStatus={resetProcessStatus}
|
|
showNewSegmentModal={showNewSegmentModal}
|
|
showBatchModal={showBatchModal}
|
|
/>
|
|
)}
|
|
<OperationAction
|
|
scene='detail'
|
|
embeddingAvailable={embeddingAvailable}
|
|
detail={{
|
|
enabled: documentDetail?.enabled || false,
|
|
archived: documentDetail?.archived || false,
|
|
id: documentId,
|
|
data_source_type: documentDetail?.data_source_type || '',
|
|
doc_form: documentDetail?.doc_form || '',
|
|
}}
|
|
datasetId={datasetId}
|
|
onUpdate={handleOperate}
|
|
className='!w-[216px]'
|
|
/>
|
|
<button
|
|
className={cn(style.layoutRightIcon, showMetadata ? style.iconShow : style.iconClose)}
|
|
onClick={() => setShowMetadata(!showMetadata)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div className='flex flex-row flex-1' style={{ height: 'calc(100% - 4rem)' }}>
|
|
{isDetailLoading
|
|
? <Loading type='app' />
|
|
: <div className={`h-full w-full flex flex-col ${embedding ? 'px-6 py-3 sm:py-12 sm:px-16' : 'pb-[30px] pt-3 px-6'}`}>
|
|
{embedding
|
|
? <Embedding detail={documentDetail} detailUpdate={detailMutate} />
|
|
: <Completed
|
|
embeddingAvailable={embeddingAvailable}
|
|
showNewSegmentModal={newSegmentModalVisible}
|
|
onNewSegmentModalChange={setNewSegmentModalVisible}
|
|
importStatus={importStatus}
|
|
archived={documentDetail?.archived}
|
|
/>
|
|
}
|
|
</div>
|
|
}
|
|
<FloatRightContainer showClose isOpen={showMetadata} onClose={() => setShowMetadata(false)} isMobile={isMobile} panelClassname='!justify-start' footer={null}>
|
|
<Metadata
|
|
docDetail={{ ...documentDetail, ...documentMetadata, doc_type: documentMetadata?.doc_type === 'others' ? '' : documentMetadata?.doc_type } as any}
|
|
loading={isMetadataLoading}
|
|
onUpdate={metadataMutate}
|
|
/>
|
|
</FloatRightContainer>
|
|
</div>
|
|
<BatchModal
|
|
isShow={batchModalVisible}
|
|
onCancel={hideBatchModal}
|
|
onConfirm={runBatch}
|
|
docForm={documentDetail?.doc_form as DocForm}
|
|
/>
|
|
</div>
|
|
</DocumentContext.Provider>
|
|
)
|
|
}
|
|
|
|
export default DocumentDetail
|
|
|