mirror of
https://github.com/labring/FastGPT.git
synced 2025-08-06 15:36:21 +00:00
297 lines
10 KiB
TypeScript
297 lines
10 KiB
TypeScript
import React, { useEffect, useMemo, useState } from 'react';
|
||
import { Box, Textarea, Button, Flex, useTheme, Grid, Progress } from '@chakra-ui/react';
|
||
import { useDatasetStore } from '@/web/core/dataset/store/dataset';
|
||
import { useSearchTestStore, SearchTestStoreItemType } from '@/web/core/dataset/store/searchTest';
|
||
import { getDatasetDataItemById, postSearchText } from '@/web/core/dataset/api';
|
||
import MyIcon from '@/components/Icon';
|
||
import { useRequest } from '@/web/common/hooks/useRequest';
|
||
import { formatTimeToChatTime } from '@/utils/tools';
|
||
import InputDataModal, { type InputDataType } from './InputDataModal';
|
||
import { useSystemStore } from '@/web/common/system/useSystemStore';
|
||
import { getErrText } from '@fastgpt/global/common/error/utils';
|
||
import { useToast } from '@/web/common/hooks/useToast';
|
||
import { customAlphabet } from 'nanoid';
|
||
import MyTooltip from '@/components/MyTooltip';
|
||
import { QuestionOutlineIcon } from '@chakra-ui/icons';
|
||
const nanoid = customAlphabet('abcdefghijklmnopqrstuvwxyz1234567890', 12);
|
||
|
||
const Test = ({ datasetId }: { datasetId: string }) => {
|
||
const theme = useTheme();
|
||
const { toast } = useToast();
|
||
const { setLoading } = useSystemStore();
|
||
const { datasetDetail } = useDatasetStore();
|
||
const { datasetTestList, pushDatasetTestItem, delDatasetTestItemById, updateDatasetItemById } =
|
||
useSearchTestStore();
|
||
const [inputText, setInputText] = useState('');
|
||
const [datasetTestItem, setDatasetTestItem] = useState<SearchTestStoreItemType>();
|
||
const [editInputData, setEditInputData] = useState<InputDataType>();
|
||
|
||
const kbTestHistory = useMemo(
|
||
() => datasetTestList.filter((item) => item.datasetId === datasetId),
|
||
[datasetId, datasetTestList]
|
||
);
|
||
|
||
const { mutate, isLoading } = useRequest({
|
||
mutationFn: () => postSearchText({ datasetId, text: inputText.trim() }),
|
||
onSuccess(res) {
|
||
const testItem = {
|
||
id: nanoid(),
|
||
datasetId,
|
||
text: inputText.trim(),
|
||
time: new Date(),
|
||
results: res
|
||
};
|
||
pushDatasetTestItem(testItem);
|
||
setDatasetTestItem(testItem);
|
||
},
|
||
onError(err) {
|
||
toast({
|
||
title: getErrText(err),
|
||
status: 'error'
|
||
});
|
||
}
|
||
});
|
||
|
||
useEffect(() => {
|
||
setDatasetTestItem(undefined);
|
||
}, [datasetId]);
|
||
|
||
return (
|
||
<Box h={'100%'} display={['block', 'flex']}>
|
||
<Box
|
||
h={['auto', '100%']}
|
||
display={['block', 'flex']}
|
||
flexDirection={'column'}
|
||
flex={1}
|
||
maxW={'500px'}
|
||
py={4}
|
||
borderRight={['none', theme.borders.base]}
|
||
>
|
||
<Box border={'2px solid'} borderColor={'myBlue.600'} p={3} mx={4} borderRadius={'md'}>
|
||
<Box fontSize={'sm'} fontWeight={'bold'}>
|
||
<MyIcon mr={2} name={'text'} w={'18px'} h={'18px'} color={'myBlue.700'} />
|
||
测试文本
|
||
</Box>
|
||
<Textarea
|
||
rows={6}
|
||
resize={'none'}
|
||
variant={'unstyled'}
|
||
maxLength={datasetDetail.vectorModel.maxToken}
|
||
placeholder="输入需要测试的文本"
|
||
value={inputText}
|
||
onChange={(e) => setInputText(e.target.value)}
|
||
/>
|
||
<Flex alignItems={'center'} justifyContent={'flex-end'}>
|
||
<Box mr={3} color={'myGray.500'}>
|
||
{inputText.length}
|
||
</Box>
|
||
<Button isDisabled={inputText === ''} isLoading={isLoading} onClick={mutate}>
|
||
测试
|
||
</Button>
|
||
</Flex>
|
||
</Box>
|
||
<Box mt={5} flex={'1 0 0'} px={4} overflow={'overlay'} display={['none', 'block']}>
|
||
<Flex alignItems={'center'} color={'myGray.600'}>
|
||
<MyIcon mr={2} name={'history'} w={'16px'} h={'16px'} />
|
||
<Box fontSize={'2xl'}>测试历史</Box>
|
||
</Flex>
|
||
<Box mt={2}>
|
||
<Flex py={2} fontWeight={'bold'} borderBottom={theme.borders.sm}>
|
||
<Box flex={1}>测试文本</Box>
|
||
<Box w={'80px'}>时间</Box>
|
||
<Box w={'14px'}></Box>
|
||
</Flex>
|
||
{kbTestHistory.map((item) => (
|
||
<Flex
|
||
key={item.id}
|
||
p={1}
|
||
alignItems={'center'}
|
||
borderBottom={theme.borders.base}
|
||
_hover={{
|
||
bg: '#f4f4f4',
|
||
'& .delete': {
|
||
display: 'block'
|
||
}
|
||
}}
|
||
cursor={'pointer'}
|
||
onClick={() => setDatasetTestItem(item)}
|
||
>
|
||
<Box flex={1} mr={2}>
|
||
{item.text}
|
||
</Box>
|
||
<Box w={'80px'}>{formatTimeToChatTime(item.time)}</Box>
|
||
<MyTooltip label={'删除该测试记录'}>
|
||
<Box w={'14px'} h={'14px'}>
|
||
<MyIcon
|
||
className="delete"
|
||
name={'delete'}
|
||
w={'14px'}
|
||
display={'none'}
|
||
_hover={{ color: 'red.600' }}
|
||
onClick={(e) => {
|
||
e.stopPropagation();
|
||
delDatasetTestItemById(item.id);
|
||
datasetTestItem?.id === item.id && setDatasetTestItem(undefined);
|
||
}}
|
||
/>
|
||
</Box>
|
||
</MyTooltip>
|
||
</Flex>
|
||
))}
|
||
</Box>
|
||
</Box>
|
||
</Box>
|
||
<Box p={4} h={['auto', '100%']} overflow={'overlay'} flex={1}>
|
||
{!datasetTestItem?.results || datasetTestItem.results.length === 0 ? (
|
||
<Flex
|
||
mt={[10, 0]}
|
||
h={'100%'}
|
||
flexDirection={'column'}
|
||
alignItems={'center'}
|
||
justifyContent={'center'}
|
||
>
|
||
<MyIcon name={'empty'} color={'transparent'} w={'54px'} />
|
||
<Box mt={3} color={'myGray.600'}>
|
||
测试结果将在这里展示
|
||
</Box>
|
||
</Flex>
|
||
) : (
|
||
<>
|
||
<Flex alignItems={'center'}>
|
||
<Box fontSize={'3xl'} color={'myGray.600'}>
|
||
测试结果
|
||
</Box>
|
||
<MyTooltip
|
||
label={
|
||
'根据知识库内容与测试文本的相似度进行排序,你可以根据测试结果调整对应的文本。\n注意:测试记录中的数据可能已经被修改过,点击某条测试数据后将展示最新的数据。'
|
||
}
|
||
forceShow
|
||
>
|
||
<QuestionOutlineIcon
|
||
ml={2}
|
||
color={'myGray.600'}
|
||
cursor={'pointer'}
|
||
fontSize={'lg'}
|
||
/>
|
||
</MyTooltip>
|
||
</Flex>
|
||
<Grid
|
||
mt={1}
|
||
gridTemplateColumns={[
|
||
'repeat(1,1fr)',
|
||
'repeat(1,1fr)',
|
||
'repeat(1,1fr)',
|
||
'repeat(1,1fr)',
|
||
'repeat(2,1fr)'
|
||
]}
|
||
gridGap={4}
|
||
>
|
||
{datasetTestItem?.results.map((item) => (
|
||
<Box
|
||
key={item.id}
|
||
pb={2}
|
||
borderRadius={'sm'}
|
||
border={theme.borders.base}
|
||
_notLast={{ mb: 2 }}
|
||
cursor={'pointer'}
|
||
title={'编辑'}
|
||
onClick={async () => {
|
||
try {
|
||
setLoading(true);
|
||
const data = await getDatasetDataItemById(item.id);
|
||
|
||
if (!data) {
|
||
throw new Error('该数据已被删除');
|
||
}
|
||
|
||
setEditInputData({
|
||
id: data.id,
|
||
datasetId: data.datasetId,
|
||
collectionId: data.collectionId,
|
||
q: data.q,
|
||
a: data.a,
|
||
sourceName: data.sourceName,
|
||
sourceId: data.sourceId
|
||
});
|
||
} catch (err) {
|
||
toast({
|
||
status: 'warning',
|
||
title: getErrText(err)
|
||
});
|
||
}
|
||
setLoading(false);
|
||
}}
|
||
>
|
||
<Flex p={3} alignItems={'center'} color={'myGray.500'}>
|
||
<MyIcon name={'kbTest'} w={'14px'} />
|
||
<Progress
|
||
mx={2}
|
||
flex={1}
|
||
value={item.score * 100}
|
||
size="sm"
|
||
borderRadius={'20px'}
|
||
colorScheme="gray"
|
||
/>
|
||
<Box>{item.score.toFixed(4)}</Box>
|
||
</Flex>
|
||
<Box
|
||
px={2}
|
||
fontSize={'xs'}
|
||
color={'myGray.600'}
|
||
maxH={'200px'}
|
||
overflow={'overlay'}
|
||
>
|
||
<Box>{item.q}</Box>
|
||
<Box>{item.a}</Box>
|
||
</Box>
|
||
</Box>
|
||
))}
|
||
</Grid>
|
||
</>
|
||
)}
|
||
</Box>
|
||
|
||
{!!editInputData && (
|
||
<InputDataModal
|
||
datasetId={editInputData.datasetId}
|
||
defaultValues={editInputData}
|
||
onClose={() => setEditInputData(undefined)}
|
||
onSuccess={(data) => {
|
||
if (datasetTestItem && editInputData.id) {
|
||
const newTestItem: SearchTestStoreItemType = {
|
||
...datasetTestItem,
|
||
results: datasetTestItem.results.map((item) =>
|
||
item.id === editInputData.id
|
||
? {
|
||
...item,
|
||
q: data.q || '',
|
||
a: data.a || ''
|
||
}
|
||
: item
|
||
)
|
||
};
|
||
updateDatasetItemById(newTestItem);
|
||
setDatasetTestItem(newTestItem);
|
||
}
|
||
|
||
setEditInputData(undefined);
|
||
}}
|
||
onDelete={() => {
|
||
if (datasetTestItem && editInputData.id) {
|
||
const newTestItem = {
|
||
...datasetTestItem,
|
||
results: datasetTestItem.results.filter((item) => item.id !== editInputData.id)
|
||
};
|
||
updateDatasetItemById(newTestItem);
|
||
setDatasetTestItem(newTestItem);
|
||
}
|
||
setEditInputData(undefined);
|
||
}}
|
||
/>
|
||
)}
|
||
</Box>
|
||
);
|
||
};
|
||
|
||
export default Test;
|