| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 | 'use client'import type { FC } from 'react'import React, { useCallback, useEffect, useMemo, useState } from 'react'import useSWR from 'swr'import { useTranslation } from 'react-i18next'import { useRouter } from 'next/navigation'import { useDebounce, useDebounceFn } from 'ahooks'import { groupBy, omit } from 'lodash-es'import { PlusIcon } from '@heroicons/react/24/solid'import { RiExternalLinkLine } from '@remixicon/react'import AutoDisabledDocument from '../common/document-status-with-action/auto-disabled-document'import List from './list'import s from './style.module.css'import Loading from '@/app/components/base/loading'import Button from '@/app/components/base/button'import Input from '@/app/components/base/input'import { get } from '@/service/base'import { createDocument, fetchDocuments } from '@/service/datasets'import { useDatasetDetailContext } from '@/context/dataset-detail'import { NotionPageSelectorModal } from '@/app/components/base/notion-page-selector'import type { NotionPage } from '@/models/common'import type { CreateDocumentReq } from '@/models/datasets'import { DataSourceType } from '@/models/datasets'import IndexFailed from '@/app/components/datasets/common/document-status-with-action/index-failed'import { useProviderContext } from '@/context/provider-context'import cn from '@/utils/classnames'import { useInvalidDocumentDetailKey } from '@/service/knowledge/use-document'import { useInvalid } from '@/service/use-base'import { useChildSegmentListKey, useSegmentListKey } from '@/service/knowledge/use-segment'const FolderPlusIcon = ({ className }: React.SVGProps<SVGElement>) => {  return <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg" className={className ?? ''}>    <path d="M10.8332 5.83333L9.90355 3.9741C9.63601 3.439 9.50222 3.17144 9.30265 2.97597C9.12615 2.80311 8.91344 2.67164 8.6799 2.59109C8.41581 2.5 8.11668 2.5 7.51841 2.5H4.33317C3.39975 2.5 2.93304 2.5 2.57652 2.68166C2.26292 2.84144 2.00795 3.09641 1.84816 3.41002C1.6665 3.76654 1.6665 4.23325 1.6665 5.16667V5.83333M1.6665 5.83333H14.3332C15.7333 5.83333 16.4334 5.83333 16.9681 6.10582C17.4386 6.3455 17.821 6.72795 18.0607 7.19836C18.3332 7.73314 18.3332 8.4332 18.3332 9.83333V13.5C18.3332 14.9001 18.3332 15.6002 18.0607 16.135C17.821 16.6054 17.4386 16.9878 16.9681 17.2275C16.4334 17.5 15.7333 17.5 14.3332 17.5H5.6665C4.26637 17.5 3.56631 17.5 3.03153 17.2275C2.56112 16.9878 2.17867 16.6054 1.93899 16.135C1.6665 15.6002 1.6665 14.9001 1.6665 13.5V5.83333ZM9.99984 14.1667V9.16667M7.49984 11.6667H12.4998" stroke="#667085" strokeWidth="1.5" strokeLinecap="round" strokeLinejoin="round" />  </svg>}const ThreeDotsIcon = ({ className }: React.SVGProps<SVGElement>) => {  return <svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg" className={className ?? ''}>    <path d="M5 6.5V5M8.93934 7.56066L10 6.5M10.0103 11.5H11.5103" stroke="#374151" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round" />  </svg>}const NotionIcon = ({ className }: React.SVGProps<SVGElement>) => {  return <svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg" className={className ?? ''}>    <g clipPath="url(#clip0_2164_11263)">      <path fillRule="evenodd" clipRule="evenodd" d="M3.5725 18.2611L1.4229 15.5832C0.905706 14.9389 0.625 14.1466 0.625 13.3312V3.63437C0.625 2.4129 1.60224 1.39936 2.86295 1.31328L12.8326 0.632614C13.5569 0.583164 14.2768 0.775682 14.8717 1.17794L18.3745 3.5462C19.0015 3.97012 19.375 4.66312 19.375 5.40266V16.427C19.375 17.6223 18.4141 18.6121 17.1798 18.688L6.11458 19.3692C5.12958 19.4298 4.17749 19.0148 3.5725 18.2611Z" fill="white" />      <path d="M7.03006 8.48669V8.35974C7.03006 8.03794 7.28779 7.77104 7.61997 7.74886L10.0396 7.58733L13.3857 12.5147V8.19009L12.5244 8.07528V8.01498C12.5244 7.68939 12.788 7.42074 13.1244 7.4035L15.326 7.29073V7.60755C15.326 7.75628 15.2154 7.88349 15.0638 7.90913L14.534 7.99874V15.0023L13.8691 15.231C13.3136 15.422 12.6952 15.2175 12.3772 14.7377L9.12879 9.83574V14.5144L10.1287 14.7057L10.1147 14.7985C10.0711 15.089 9.82028 15.3087 9.51687 15.3222L7.03006 15.4329C6.99718 15.1205 7.23132 14.841 7.55431 14.807L7.88143 14.7727V8.53453L7.03006 8.48669Z" fill="black" />      <path fillRule="evenodd" clipRule="evenodd" d="M12.9218 1.85424L2.95217 2.53491C2.35499 2.57568 1.89209 3.05578 1.89209 3.63437V13.3312C1.89209 13.8748 2.07923 14.403 2.42402 14.8325L4.57362 17.5104C4.92117 17.9434 5.46812 18.1818 6.03397 18.147L17.0991 17.4658C17.6663 17.4309 18.1078 16.9762 18.1078 16.427V5.40266C18.1078 5.06287 17.9362 4.74447 17.6481 4.54969L14.1453 2.18143C13.7883 1.94008 13.3564 1.82457 12.9218 1.85424ZM3.44654 3.78562C3.30788 3.68296 3.37387 3.46909 3.54806 3.4566L12.9889 2.77944C13.2897 2.75787 13.5886 2.8407 13.8318 3.01305L15.7261 4.35508C15.798 4.40603 15.7642 4.51602 15.6752 4.52086L5.67742 5.0646C5.37485 5.08106 5.0762 4.99217 4.83563 4.81406L3.44654 3.78562ZM5.20848 6.76919C5.20848 6.4444 5.47088 6.1761 5.80642 6.15783L16.3769 5.58216C16.7039 5.56435 16.9792 5.81583 16.9792 6.13239V15.6783C16.9792 16.0025 16.7177 16.2705 16.3829 16.2896L5.8793 16.8872C5.51537 16.9079 5.20848 16.6283 5.20848 16.2759V6.76919Z" fill="black" />    </g>    <defs>      <clipPath id="clip0_2164_11263">        <rect width="20" height="20" fill="white" />      </clipPath>    </defs>  </svg>}const EmptyElement: FC<{ canAdd: boolean; onClick: () => void; type?: 'upload' | 'sync' }> = ({ canAdd = true, onClick, type = 'upload' }) => {  const { t } = useTranslation()  return <div className={s.emptyWrapper}>    <div className={s.emptyElement}>      <div className={s.emptySymbolIconWrapper}>        {type === 'upload' ? <FolderPlusIcon /> : <NotionIcon />}      </div>      <span className={s.emptyTitle}>{t('datasetDocuments.list.empty.title')}<ThreeDotsIcon className='inline relative -top-3 -left-1.5' /></span>      <div className={s.emptyTip}>        {t(`datasetDocuments.list.empty.${type}.tip`)}      </div>      {type === 'upload' && canAdd && <Button onClick={onClick} className={s.addFileBtn}>        <PlusIcon className={s.plusIcon} />{t('datasetDocuments.list.addFile')}      </Button>}    </div>  </div>}type IDocumentsProps = {  datasetId: string}export const fetcher = (url: string) => get(url, {}, {})const DEFAULT_LIMIT = 15const Documents: FC<IDocumentsProps> = ({ datasetId }) => {  const { t } = useTranslation()  const { plan } = useProviderContext()  const isFreePlan = plan.type === 'sandbox'  const [inputValue, setInputValue] = useState<string>('') // the input value  const [searchValue, setSearchValue] = useState<string>('')  const [currPage, setCurrPage] = React.useState<number>(0)  const [limit, setLimit] = useState<number>(DEFAULT_LIMIT)  const router = useRouter()  const { dataset } = useDatasetDetailContext()  const [notionPageSelectorModalVisible, setNotionPageSelectorModalVisible] = useState(false)  const [timerCanRun, setTimerCanRun] = useState(true)  const isDataSourceNotion = dataset?.data_source_type === DataSourceType.NOTION  const isDataSourceWeb = dataset?.data_source_type === DataSourceType.WEB  const isDataSourceFile = dataset?.data_source_type === DataSourceType.FILE  const embeddingAvailable = !!dataset?.embedding_available  const debouncedSearchValue = useDebounce(searchValue, { wait: 500 })  const query = useMemo(() => {    return { page: currPage + 1, limit, keyword: debouncedSearchValue, fetch: isDataSourceNotion ? true : '' }  }, [currPage, debouncedSearchValue, isDataSourceNotion, limit])  const { data: documentsRes, mutate, isLoading: isListLoading } = useSWR(    {      action: 'fetchDocuments',      datasetId,      params: query,    },    apiParams => fetchDocuments(omit(apiParams, 'action')),    { refreshInterval: (isDataSourceNotion && timerCanRun) ? 2500 : 0 },  )  const [isMuting, setIsMuting] = useState(false)  useEffect(() => {    if (!isListLoading && isMuting)      setIsMuting(false)  }, [isListLoading, isMuting])  const invalidDocumentDetail = useInvalidDocumentDetailKey()  const invalidChunkList = useInvalid(useSegmentListKey)  const invalidChildChunkList = useInvalid(useChildSegmentListKey)  const handleUpdate = useCallback(() => {    setIsMuting(true)    mutate()    invalidDocumentDetail()    setTimeout(() => {      invalidChunkList()      invalidChildChunkList()    }, 5000)    // eslint-disable-next-line react-hooks/exhaustive-deps  }, [])  const documentsWithProgress = useMemo(() => {    let completedNum = 0    let percent = 0    const documentsData = documentsRes?.data?.map((documentItem) => {      const { indexing_status, completed_segments, total_segments } = documentItem      const isEmbedded = indexing_status === 'completed' || indexing_status === 'paused' || indexing_status === 'error'      if (isEmbedded)        completedNum++      const completedCount = completed_segments || 0      const totalCount = total_segments || 0      if (totalCount === 0 && completedCount === 0) {        percent = isEmbedded ? 100 : 0      }      else {        const per = Math.round(completedCount * 100 / totalCount)        percent = per > 100 ? 100 : per      }      return {        ...documentItem,        percent,      }    })    if (completedNum === documentsRes?.data?.length)      setTimerCanRun(false)    return {      ...documentsRes,      data: documentsData,    }  }, [documentsRes])  const total = documentsRes?.total || 0  const routeToDocCreate = () => {    if (isDataSourceNotion) {      setNotionPageSelectorModalVisible(true)      return    }    router.push(`/datasets/${datasetId}/documents/create`)  }  const isLoading = isListLoading // !documentsRes && !error  const handleSaveNotionPageSelected = async (selectedPages: NotionPage[]) => {    const workspacesMap = groupBy(selectedPages, 'workspace_id')    const workspaces = Object.keys(workspacesMap).map((workspaceId) => {      return {        workspaceId,        pages: workspacesMap[workspaceId],      }    })    const params = {      data_source: {        type: dataset?.data_source_type,        info_list: {          data_source_type: dataset?.data_source_type,          notion_info_list: workspaces.map((workspace) => {            return {              workspace_id: workspace.workspaceId,              pages: workspace.pages.map((page) => {                const { page_id, page_name, page_icon, type } = page                return {                  page_id,                  page_name,                  page_icon,                  type,                }              }),            }          }),        },      },      indexing_technique: dataset?.indexing_technique,      process_rule: {        rules: {},        mode: 'automatic',      },    } as CreateDocumentReq    await createDocument({      datasetId,      body: params,    })    mutate()    setTimerCanRun(true)    // mutateDatasetIndexingStatus(undefined, { revalidate: true })    setNotionPageSelectorModalVisible(false)  }  const documentsList = isDataSourceNotion ? documentsWithProgress?.data : documentsRes?.data  const [selectedIds, setSelectedIds] = useState<string[]>([])  const { run: handleSearch } = useDebounceFn(() => {    setSearchValue(inputValue)  }, { wait: 500 })  const handleInputChange = (value: string) => {    setInputValue(value)    handleSearch()  }  return (    <div className='flex flex-col h-full overflow-y-auto'>      <div className='flex flex-col justify-center gap-1 px-6 pt-4'>        <h1 className='text-base font-semibold text-text-primary'>{t('datasetDocuments.list.title')}</h1>        <div className='flex items-center text-sm font-normal text-text-tertiary space-x-0.5'>          <span>{t('datasetDocuments.list.desc')}</span>          <a            className='flex items-center text-text-accent'            target='_blank'            href='https://docs.dify.ai/guides/knowledge-base/integrate-knowledge-within-application'>            <span>{t('datasetDocuments.list.learnMore')}</span>            <RiExternalLinkLine className='w-3 h-3' />          </a>        </div>      </div>      <div className='flex flex-col px-6 py-4 flex-1'>        <div className='flex items-center justify-between flex-wrap'>          <Input            showLeftIcon            showClearIcon            wrapperClassName='!w-[200px]'            value={inputValue}            onChange={e => handleInputChange(e.target.value)}            onClear={() => handleInputChange('')}          />          <div className='flex gap-2 justify-center items-center !h-8'>            {!isFreePlan && <AutoDisabledDocument datasetId={datasetId} />}            <IndexFailed datasetId={datasetId} />            {embeddingAvailable && (              <Button variant='primary' onClick={routeToDocCreate} className='shrink-0'>                <PlusIcon className={cn('h-4 w-4 mr-2 stroke-current')} />                {isDataSourceNotion && t('datasetDocuments.list.addPages')}                {isDataSourceWeb && t('datasetDocuments.list.addUrl')}                {(!dataset?.data_source_type || isDataSourceFile) && t('datasetDocuments.list.addFile')}              </Button>            )}          </div>        </div>        {(isLoading && !isMuting)          ? <Loading type='app' />          : total > 0            ? <List              embeddingAvailable={embeddingAvailable}              documents={documentsList || []}              datasetId={datasetId}              onUpdate={handleUpdate}              selectedIds={selectedIds}              onSelectedIdChange={setSelectedIds}              pagination={{                total,                limit,                onLimitChange: setLimit,                current: currPage,                onChange: setCurrPage,              }}            />            : <EmptyElement canAdd={embeddingAvailable} onClick={routeToDocCreate} type={isDataSourceNotion ? 'sync' : 'upload'} />        }        <NotionPageSelectorModal          isShow={notionPageSelectorModalVisible}          onClose={() => setNotionPageSelectorModalVisible(false)}          onSave={handleSaveNotionPageSelected}          datasetId={dataset?.id || ''}        />      </div>    </div>  )}export default Documents
 |