Files
dify/web/app/components/datasets/hit-testing/components/query-input/index.tsx
Stephen Zhou 6d0e36479b refactor(i18n): use JSON with flattened key and namespace (#30114)
Co-authored-by: yyh <yuanyouhuilyz@gmail.com>
Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com>
2025-12-29 14:52:32 +08:00

262 lines
9.4 KiB
TypeScript

import type { UseMutateAsyncFunction } from '@tanstack/react-query'
import type { ChangeEvent } from 'react'
import type { FileEntity } from '@/app/components/datasets/common/image-uploader/types'
import type {
Attachment,
ExternalKnowledgeBaseHitTestingRequest,
ExternalKnowledgeBaseHitTestingResponse,
HitTestingRequest,
HitTestingResponse,
Query,
} from '@/models/datasets'
import type { RetrievalConfig } from '@/types/app'
import {
RiEqualizer2Line,
RiPlayCircleLine,
} from '@remixicon/react'
import Image from 'next/image'
import * as React from 'react'
import { useCallback, useMemo, useState } from 'react'
import { useTranslation } from 'react-i18next'
import { v4 as uuid4 } from 'uuid'
import Button from '@/app/components/base/button'
import ImageUploaderInRetrievalTesting from '@/app/components/datasets/common/image-uploader/image-uploader-in-retrieval-testing'
import { getIcon } from '@/app/components/datasets/common/retrieval-method-info'
import ModifyExternalRetrievalModal from '@/app/components/datasets/hit-testing/modify-external-retrieval-modal'
import { useDatasetDetailContextWithSelector } from '@/context/dataset-detail'
import { RETRIEVE_METHOD } from '@/types/app'
import { cn } from '@/utils/classnames'
import Textarea from './textarea'
type QueryInputProps = {
onUpdateList: () => void
setHitResult: (res: HitTestingResponse) => void
setExternalHitResult: (res: ExternalKnowledgeBaseHitTestingResponse) => void
loading: boolean
queries: Query[]
setQueries: (v: Query[]) => void
isExternal?: boolean
onClickRetrievalMethod: () => void
retrievalConfig: RetrievalConfig
isEconomy: boolean
onSubmit?: () => void
hitTestingMutation: UseMutateAsyncFunction<HitTestingResponse, Error, HitTestingRequest, unknown>
externalKnowledgeBaseHitTestingMutation: UseMutateAsyncFunction<
ExternalKnowledgeBaseHitTestingResponse,
Error,
ExternalKnowledgeBaseHitTestingRequest,
unknown
>
}
const QueryInput = ({
onUpdateList,
setHitResult,
setExternalHitResult,
loading,
queries,
setQueries,
isExternal = false,
onClickRetrievalMethod,
retrievalConfig,
isEconomy,
onSubmit: _onSubmit,
hitTestingMutation,
externalKnowledgeBaseHitTestingMutation,
}: QueryInputProps) => {
const { t } = useTranslation()
const isMultimodal = useDatasetDetailContextWithSelector(s => !!s.dataset?.is_multimodal)
const [isSettingsOpen, setIsSettingsOpen] = useState(false)
const [externalRetrievalSettings, setExternalRetrievalSettings] = useState({
top_k: 4,
score_threshold: 0.5,
score_threshold_enabled: false,
})
const text = useMemo(() => {
return queries.find(query => query.content_type === 'text_query')?.content ?? ''
}, [queries])
const images = useMemo(() => {
const imageQueries = queries
.filter(query => query.content_type === 'image_query')
.map(query => query.file_info)
.filter(Boolean) as Attachment[]
return imageQueries.map(item => ({
id: uuid4(),
name: item.name,
size: item.size,
mimeType: item.mime_type,
extension: item.extension,
sourceUrl: item.source_url,
uploadedId: item.id,
progress: 100,
})) || []
}, [queries])
const isAllUploaded = useMemo(() => {
return images.every(image => !!image.uploadedId)
}, [images])
const handleSaveExternalRetrievalSettings = useCallback((data: {
top_k: number
score_threshold: number
score_threshold_enabled: boolean
}) => {
setExternalRetrievalSettings(data)
setIsSettingsOpen(false)
}, [])
const handleTextChange = useCallback((event: ChangeEvent<HTMLTextAreaElement>) => {
const newQueries = [...queries]
const textQuery = newQueries.find(query => query.content_type === 'text_query')
if (!textQuery) {
newQueries.push({
content: event.target.value,
content_type: 'text_query',
file_info: null,
})
}
else {
textQuery.content = event.target.value
}
setQueries(newQueries)
}, [queries, setQueries])
const handleImageChange = useCallback((files: FileEntity[]) => {
let newQueries = [...queries]
newQueries = newQueries.filter(query => query.content_type !== 'image_query')
files.forEach((file) => {
newQueries.push({
content: file.sourceUrl || '',
content_type: 'image_query',
file_info: {
id: file.uploadedId || '',
mime_type: file.mimeType,
source_url: file.sourceUrl || '',
name: file.name,
size: file.size,
extension: file.extension,
},
})
})
setQueries(newQueries)
}, [queries, setQueries])
const onSubmit = useCallback(async () => {
await hitTestingMutation({
query: text,
attachment_ids: images.map(image => image.uploadedId),
retrieval_model: {
...retrievalConfig,
search_method: isEconomy ? RETRIEVE_METHOD.keywordSearch : retrievalConfig.search_method,
},
}, {
onSuccess: (data) => {
setHitResult(data)
onUpdateList?.()
if (_onSubmit)
_onSubmit()
},
})
}, [text, retrievalConfig, isEconomy, hitTestingMutation, onUpdateList, _onSubmit, images, setHitResult])
const externalRetrievalTestingOnSubmit = useCallback(async () => {
await externalKnowledgeBaseHitTestingMutation({
query: text,
external_retrieval_model: {
top_k: externalRetrievalSettings.top_k,
score_threshold: externalRetrievalSettings.score_threshold,
score_threshold_enabled: externalRetrievalSettings.score_threshold_enabled,
},
}, {
onSuccess: (data) => {
setExternalHitResult(data)
onUpdateList?.()
},
})
}, [text, externalRetrievalSettings, externalKnowledgeBaseHitTestingMutation, onUpdateList, setExternalHitResult])
const retrievalMethod = isEconomy ? RETRIEVE_METHOD.keywordSearch : retrievalConfig.search_method
const icon = <Image className="size-3.5 text-util-colors-purple-purple-600" src={getIcon(retrievalMethod)} alt="" />
const TextAreaComp = useMemo(() => {
return (
<Textarea
text={text}
handleTextChange={handleTextChange}
/>
)
}, [text, handleTextChange])
const ActionButtonComp = useMemo(() => {
return (
<Button
onClick={isExternal ? externalRetrievalTestingOnSubmit : onSubmit}
variant="primary"
loading={loading}
disabled={(text.length === 0 && images.length === 0) || text.length > 200 || (images.length > 0 && !isAllUploaded)}
className="w-[88px]"
>
<RiPlayCircleLine className="mr-1 size-4" />
{t('input.testing', { ns: 'datasetHitTesting' })}
</Button>
)
}, [isExternal, externalRetrievalTestingOnSubmit, onSubmit, text, loading, t, images, isAllUploaded])
return (
<div className={cn('relative flex h-80 shrink-0 flex-col overflow-hidden rounded-xl bg-gradient-to-r from-components-input-border-active-prompt-1 to-components-input-border-active-prompt-2 p-0.5 shadow-xs')}>
<div className="flex h-full flex-col overflow-hidden rounded-[10px] bg-background-section-burn">
<div className="relative flex shrink-0 items-center justify-between p-1.5 pb-1 pl-3">
<span className="system-sm-semibold-uppercase text-text-secondary">
{t('input.title', { ns: 'datasetHitTesting' })}
</span>
{isExternal
? (
<Button
variant="secondary"
size="small"
onClick={() => setIsSettingsOpen(!isSettingsOpen)}
>
<RiEqualizer2Line className="h-3.5 w-3.5 text-components-button-secondary-text" />
<div className="flex items-center justify-center gap-1 px-[3px]">
<span className="system-xs-medium text-components-button-secondary-text">{t('settingTitle', { ns: 'datasetHitTesting' })}</span>
</div>
</Button>
)
: (
<div
onClick={onClickRetrievalMethod}
className="flex h-7 cursor-pointer items-center space-x-0.5 rounded-lg border-[0.5px] border-components-button-secondary-bg bg-components-button-secondary-bg px-1.5 shadow-xs backdrop-blur-[5px] hover:bg-components-button-secondary-bg-hover"
>
{icon}
<div className="text-xs font-medium uppercase text-text-secondary">{t(`retrieval.${retrievalMethod}.title`, { ns: 'dataset' })}</div>
<RiEqualizer2Line className="size-4 text-components-menu-item-text"></RiEqualizer2Line>
</div>
)}
{
isSettingsOpen && (
<ModifyExternalRetrievalModal
onClose={() => setIsSettingsOpen(false)}
onSave={handleSaveExternalRetrievalSettings}
initialTopK={externalRetrievalSettings.top_k}
initialScoreThreshold={externalRetrievalSettings.score_threshold}
initialScoreThresholdEnabled={externalRetrievalSettings.score_threshold_enabled}
/>
)
}
</div>
<ImageUploaderInRetrievalTesting
textArea={TextAreaComp}
actionButton={ActionButtonComp}
onChange={handleImageChange}
value={images}
showUploader={isMultimodal}
className="grow"
actionAreaClassName="px-4 py-2 shrink-0 bg-background-default"
/>
</div>
</div>
)
}
export default QueryInput