index.tsx 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. import type { FC, SVGProps } from 'react'
  2. import React, { useCallback, useEffect, useMemo, useState } from 'react'
  3. import useSWR from 'swr'
  4. import { useRouter } from 'next/navigation'
  5. import { useContext } from 'use-context-selector'
  6. import { useTranslation } from 'react-i18next'
  7. import { omit } from 'lodash-es'
  8. import { ArrowRightIcon } from '@heroicons/react/24/solid'
  9. import { useGetState } from 'ahooks'
  10. import cn from 'classnames'
  11. import SegmentCard from '../completed/SegmentCard'
  12. import { FieldInfo } from '../metadata'
  13. import style from '../completed/style.module.css'
  14. import { DocumentContext } from '../index'
  15. import s from './style.module.css'
  16. import Button from '@/app/components/base/button'
  17. import Divider from '@/app/components/base/divider'
  18. import { ToastContext } from '@/app/components/base/toast'
  19. import type { FullDocumentDetail, ProcessRuleResponse } from '@/models/datasets'
  20. import type { CommonResponse } from '@/models/common'
  21. import { asyncRunSafe } from '@/utils'
  22. import { formatNumber } from '@/utils/format'
  23. import { fetchIndexingStatus as doFetchIndexingStatus, fetchIndexingEstimate, fetchProcessRule, pauseDocIndexing, resumeDocIndexing } from '@/service/datasets'
  24. import DatasetDetailContext from '@/context/dataset-detail'
  25. import StopEmbeddingModal from '@/app/components/datasets/create/stop-embedding-modal'
  26. type Props = {
  27. detail?: FullDocumentDetail
  28. stopPosition?: 'top' | 'bottom'
  29. datasetId?: string
  30. documentId?: string
  31. indexingType?: string
  32. detailUpdate: VoidFunction
  33. }
  34. const StopIcon = ({ className }: SVGProps<SVGElement>) => {
  35. return <svg width="12" height="12" viewBox="0 0 12 12" fill="none" xmlns="http://www.w3.org/2000/svg" className={className ?? ''}>
  36. <g clip-path="url(#clip0_2328_2798)">
  37. <path d="M1.5 3.9C1.5 3.05992 1.5 2.63988 1.66349 2.31901C1.8073 2.03677 2.03677 1.8073 2.31901 1.66349C2.63988 1.5 3.05992 1.5 3.9 1.5H8.1C8.94008 1.5 9.36012 1.5 9.68099 1.66349C9.96323 1.8073 10.1927 2.03677 10.3365 2.31901C10.5 2.63988 10.5 3.05992 10.5 3.9V8.1C10.5 8.94008 10.5 9.36012 10.3365 9.68099C10.1927 9.96323 9.96323 10.1927 9.68099 10.3365C9.36012 10.5 8.94008 10.5 8.1 10.5H3.9C3.05992 10.5 2.63988 10.5 2.31901 10.3365C2.03677 10.1927 1.8073 9.96323 1.66349 9.68099C1.5 9.36012 1.5 8.94008 1.5 8.1V3.9Z" stroke="#344054" strokeWidth="1.5" strokeLinecap="round" strokeLinejoin="round" />
  38. </g>
  39. <defs>
  40. <clipPath id="clip0_2328_2798">
  41. <rect width="12" height="12" fill="white" />
  42. </clipPath>
  43. </defs>
  44. </svg>
  45. }
  46. const ResumeIcon = ({ className }: SVGProps<SVGElement>) => {
  47. return <svg width="12" height="12" viewBox="0 0 12 12" fill="none" xmlns="http://www.w3.org/2000/svg" className={className ?? ''}>
  48. <path d="M10 3.5H5C3.34315 3.5 2 4.84315 2 6.5C2 8.15685 3.34315 9.5 5 9.5H10M10 3.5L8 1.5M10 3.5L8 5.5" stroke="#344054" strokeWidth="1.5" strokeLinecap="round" strokeLinejoin="round" />
  49. </svg>
  50. }
  51. const RuleDetail: FC<{ sourceData?: ProcessRuleResponse; docName?: string }> = ({ sourceData, docName }) => {
  52. const { t } = useTranslation()
  53. const segmentationRuleMap = {
  54. docName: t('datasetDocuments.embedding.docName'),
  55. mode: t('datasetDocuments.embedding.mode'),
  56. segmentLength: t('datasetDocuments.embedding.segmentLength'),
  57. textCleaning: t('datasetDocuments.embedding.textCleaning'),
  58. }
  59. const getRuleName = (key: string) => {
  60. if (key === 'remove_extra_spaces')
  61. return t('datasetCreation.stepTwo.removeExtraSpaces')
  62. if (key === 'remove_urls_emails')
  63. return t('datasetCreation.stepTwo.removeUrlEmails')
  64. if (key === 'remove_stopwords')
  65. return t('datasetCreation.stepTwo.removeStopwords')
  66. }
  67. const getValue = useCallback((field: string) => {
  68. let value: string | number | undefined = '-'
  69. switch (field) {
  70. case 'docName':
  71. value = docName
  72. break
  73. case 'mode':
  74. value = sourceData?.mode === 'automatic' ? (t('datasetDocuments.embedding.automatic') as string) : (t('datasetDocuments.embedding.custom') as string)
  75. break
  76. case 'segmentLength':
  77. value = sourceData?.rules?.segmentation?.max_tokens
  78. break
  79. default:
  80. value = sourceData?.mode === 'automatic'
  81. ? (t('datasetDocuments.embedding.automatic') as string)
  82. // eslint-disable-next-line array-callback-return
  83. : sourceData?.rules?.pre_processing_rules?.map((rule) => {
  84. if (rule.enabled)
  85. return getRuleName(rule.id)
  86. }).filter(Boolean).join(';')
  87. break
  88. }
  89. return value
  90. }, [sourceData, docName])
  91. return <div className='flex flex-col pt-8 pb-10 first:mt-0'>
  92. {Object.keys(segmentationRuleMap).map((field) => {
  93. return <FieldInfo
  94. key={field}
  95. label={segmentationRuleMap[field as keyof typeof segmentationRuleMap]}
  96. displayedValue={String(getValue(field))}
  97. />
  98. })}
  99. </div>
  100. }
  101. const EmbeddingDetail: FC<Props> = ({ detail, stopPosition = 'top', datasetId: dstId, documentId: docId, indexingType, detailUpdate }) => {
  102. const onTop = stopPosition === 'top'
  103. const { t } = useTranslation()
  104. const { notify } = useContext(ToastContext)
  105. const { datasetId = '', documentId = '' } = useContext(DocumentContext)
  106. const { indexingTechnique } = useContext(DatasetDetailContext)
  107. const localDatasetId = dstId ?? datasetId
  108. const localDocumentId = docId ?? documentId
  109. const localIndexingTechnique = indexingType ?? indexingTechnique
  110. // const { data: indexingStatusDetailFromApi, error: indexingStatusErr, mutate: statusMutate } = useSWR({
  111. // action: 'fetchIndexingStatus',
  112. // datasetId: localDatasetId,
  113. // documentId: localDocumentId,
  114. // }, apiParams => fetchIndexingStatus(omit(apiParams, 'action')), {
  115. // refreshInterval: 2500,
  116. // revalidateOnFocus: false,
  117. // })
  118. const [indexingStatusDetail, setIndexingStatusDetail, getIndexingStatusDetail] = useGetState<any>(null)
  119. const fetchIndexingStatus = async () => {
  120. const status = await doFetchIndexingStatus({ datasetId: localDatasetId, documentId: localDocumentId })
  121. setIndexingStatusDetail(status)
  122. }
  123. const [runId, setRunId, getRunId] = useGetState<any>(null)
  124. const stopQueryStatus = () => {
  125. clearInterval(getRunId())
  126. }
  127. const startQueryStatus = () => {
  128. const runId = setInterval(() => {
  129. const indexingStatusDetail = getIndexingStatusDetail()
  130. if (indexingStatusDetail?.indexing_status === 'completed') {
  131. stopQueryStatus()
  132. detailUpdate()
  133. return
  134. }
  135. fetchIndexingStatus()
  136. }, 2500)
  137. setRunId(runId)
  138. }
  139. useEffect(() => {
  140. fetchIndexingStatus()
  141. startQueryStatus()
  142. return () => {
  143. stopQueryStatus()
  144. }
  145. }, [])
  146. const { data: indexingEstimateDetail, error: indexingEstimateErr } = useSWR({
  147. action: 'fetchIndexingEstimate',
  148. datasetId: localDatasetId,
  149. documentId: localDocumentId,
  150. }, apiParams => fetchIndexingEstimate(omit(apiParams, 'action')), {
  151. revalidateOnFocus: false,
  152. })
  153. const { data: ruleDetail, error: ruleError } = useSWR({
  154. action: 'fetchProcessRule',
  155. params: { documentId: localDocumentId },
  156. }, apiParams => fetchProcessRule(omit(apiParams, 'action')), {
  157. revalidateOnFocus: false,
  158. })
  159. const [showModal, setShowModal] = useState(false)
  160. const modalShowHandle = () => setShowModal(true)
  161. const modalCloseHandle = () => setShowModal(false)
  162. const router = useRouter()
  163. const navToDocument = () => {
  164. router.push(`/datasets/${localDatasetId}/documents/${localDocumentId}`)
  165. }
  166. const isEmbedding = useMemo(() => ['indexing', 'splitting', 'parsing', 'cleaning'].includes(indexingStatusDetail?.indexing_status || ''), [indexingStatusDetail])
  167. const isEmbeddingCompleted = useMemo(() => ['completed'].includes(indexingStatusDetail?.indexing_status || ''), [indexingStatusDetail])
  168. const isEmbeddingPaused = useMemo(() => ['paused'].includes(indexingStatusDetail?.indexing_status || ''), [indexingStatusDetail])
  169. const isEmbeddingError = useMemo(() => ['error'].includes(indexingStatusDetail?.indexing_status || ''), [indexingStatusDetail])
  170. const percent = useMemo(() => {
  171. const completedCount = indexingStatusDetail?.completed_segments || 0
  172. const totalCount = indexingStatusDetail?.total_segments || 0
  173. if (totalCount === 0)
  174. return 0
  175. const percent = Math.round(completedCount * 100 / totalCount)
  176. return percent > 100 ? 100 : percent
  177. }, [indexingStatusDetail])
  178. const handleSwitch = async () => {
  179. const opApi = isEmbedding ? pauseDocIndexing : resumeDocIndexing
  180. const [e] = await asyncRunSafe<CommonResponse>(opApi({ datasetId: localDatasetId, documentId: localDocumentId }) as Promise<CommonResponse>)
  181. if (!e) {
  182. notify({ type: 'success', message: t('common.actionMsg.modifiedSuccessfully') })
  183. setIndexingStatusDetail(null)
  184. }
  185. else {
  186. notify({ type: 'error', message: t('common.actionMsg.modifiedUnsuccessfully') })
  187. }
  188. }
  189. // if (!ruleDetail && !error)
  190. // return <Loading type='app' />
  191. return (
  192. <>
  193. <div className={s.embeddingStatus}>
  194. {isEmbedding && t('datasetDocuments.embedding.processing')}
  195. {isEmbeddingCompleted && t('datasetDocuments.embedding.completed')}
  196. {isEmbeddingPaused && t('datasetDocuments.embedding.paused')}
  197. {isEmbeddingError && t('datasetDocuments.embedding.error')}
  198. {onTop && isEmbedding && (
  199. <Button onClick={handleSwitch} className={s.opBtn}>
  200. <StopIcon className={s.opIcon} />
  201. {t('datasetDocuments.embedding.stop')}
  202. </Button>
  203. )}
  204. {onTop && isEmbeddingPaused && (
  205. <Button onClick={handleSwitch} className={s.opBtn}>
  206. <ResumeIcon className={s.opIcon} />
  207. {t('datasetDocuments.embedding.resume')}
  208. </Button>
  209. )}
  210. </div>
  211. {/* progress bar */}
  212. <div className={s.progressContainer}>
  213. {new Array(10).fill('').map((_, idx) => <div
  214. key={idx}
  215. className={cn(s.progressBgItem, isEmbedding ? 'bg-primary-50' : 'bg-gray-100')}
  216. />)}
  217. <div
  218. className={cn(
  219. 'rounded-l-md',
  220. s.progressBar,
  221. (isEmbedding || isEmbeddingCompleted) && s.barProcessing,
  222. (isEmbeddingPaused || isEmbeddingError) && s.barPaused,
  223. indexingStatusDetail?.indexing_status === 'completed' && 'rounded-r-md',
  224. )}
  225. style={{ width: `${percent}%` }}
  226. />
  227. </div>
  228. <div className={s.progressData}>
  229. <div>{t('datasetDocuments.embedding.segments')} {indexingStatusDetail?.completed_segments}/{indexingStatusDetail?.total_segments} · {percent}%</div>
  230. {localIndexingTechnique === 'high_quaility' && (
  231. <div className='flex items-center'>
  232. <div className={cn(s.commonIcon, s.highIcon)} />
  233. {t('datasetDocuments.embedding.highQuality')} · {t('datasetDocuments.embedding.estimate')}
  234. <span className={s.tokens}>{formatNumber(indexingEstimateDetail?.tokens || 0)}</span>tokens
  235. (<span className={s.price}>${formatNumber(indexingEstimateDetail?.total_price || 0)}</span>)
  236. </div>
  237. )}
  238. {localIndexingTechnique === 'economy' && (
  239. <div className='flex items-center'>
  240. <div className={cn(s.commonIcon, s.economyIcon)} />
  241. {t('datasetDocuments.embedding.economy')} · {t('datasetDocuments.embedding.estimate')}
  242. <span className={s.tokens}>0</span>tokens
  243. </div>
  244. )}
  245. </div>
  246. <RuleDetail sourceData={ruleDetail} docName={detail?.name} />
  247. {!onTop && (
  248. <div className='flex items-center gap-2 mt-10'>
  249. {isEmbedding && (
  250. <Button onClick={modalShowHandle} className='w-fit'>
  251. {t('datasetCreation.stepThree.stop')}
  252. </Button>
  253. )}
  254. {isEmbeddingPaused && (
  255. <Button onClick={handleSwitch} className='w-fit'>
  256. {t('datasetCreation.stepThree.resume')}
  257. </Button>
  258. )}
  259. <Button className='w-fit' type='primary' onClick={navToDocument}>
  260. <span>{t('datasetCreation.stepThree.navTo')}</span>
  261. <ArrowRightIcon className='h-4 w-4 ml-2 stroke-current stroke-1' />
  262. </Button>
  263. </div>
  264. )}
  265. {onTop && <>
  266. <Divider />
  267. <div className={s.previewTip}>{t('datasetDocuments.embedding.previewTip')}</div>
  268. <div className={style.cardWrapper}>
  269. {[1, 2, 3].map((v, index) => (
  270. <SegmentCard key={index} loading={true} detail={{ position: v } as any} />
  271. ))}
  272. </div>
  273. </>}
  274. <StopEmbeddingModal show={showModal} onConfirm={handleSwitch} onHide={modalCloseHandle} />
  275. </>
  276. )
  277. }
  278. export default EmbeddingDetail