mirror of
https://github.com/csunny/DB-GPT.git
synced 2025-07-27 05:47:47 +00:00
Co-authored-by: Fangyin Cheng <staneyffer@gmail.com> Co-authored-by: 谨欣 <echo.cmy@antgroup.com> Co-authored-by: 严志勇 <yanzhiyong@tiansuixiansheng.com> Co-authored-by: yanzhiyong <932374019@qq.com>
204 lines
6.3 KiB
TypeScript
204 lines
6.3 KiB
TypeScript
import { apiInterceptors, recallMethodOptions, recallTest, recallTestRecommendQuestion } from '@/client/api';
|
|
import MarkDownContext from '@/new-components/common/MarkdownContext';
|
|
import { ISpace, RecallTestProps } from '@/types/knowledge';
|
|
import { SettingOutlined } from '@ant-design/icons';
|
|
import { useRequest } from 'ahooks';
|
|
import { Button, Card, Empty, Form, Input, InputNumber, Modal, Popover, Select, Spin, Tag } from 'antd';
|
|
import React, { useEffect } from 'react';
|
|
|
|
type RecallTestModalProps = {
|
|
open: boolean;
|
|
setOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
|
space: ISpace;
|
|
};
|
|
|
|
// const tagColors = ['magenta', 'orange', 'geekblue', 'purple', 'cyan', 'green'];
|
|
|
|
const RecallTestModal: React.FC<RecallTestModalProps> = ({ open, setOpen, space }) => {
|
|
const [form] = Form.useForm();
|
|
const [extraForm] = Form.useForm();
|
|
|
|
// 获取推荐问题
|
|
const { run: questionsRun } = useRequest(
|
|
// const { data: questions = [], run: questionsRun } = useRequest(
|
|
async () => {
|
|
const [, res] = await apiInterceptors(recallTestRecommendQuestion(space.name + ''));
|
|
return res ?? [];
|
|
},
|
|
{
|
|
manual: true,
|
|
},
|
|
);
|
|
|
|
// 召回方法选项
|
|
const { data: options = [], run: optionsRun } = useRequest(
|
|
async () => {
|
|
const [, res] = await apiInterceptors(recallMethodOptions(space.name + ''));
|
|
return res ?? [];
|
|
},
|
|
{
|
|
manual: true,
|
|
onSuccess: data => {
|
|
extraForm.setFieldValue('recall_retrievers', data);
|
|
},
|
|
},
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (open) {
|
|
// questionsRun();
|
|
optionsRun();
|
|
}
|
|
}, [open, optionsRun, questionsRun]);
|
|
|
|
// 召回测试
|
|
const {
|
|
run: recallTestRun,
|
|
data: resultList = [],
|
|
loading,
|
|
} = useRequest(
|
|
async (props: RecallTestProps) => {
|
|
const [, res] = await apiInterceptors(recallTest({ ...props }, space.name + ''));
|
|
return res ?? [];
|
|
},
|
|
{
|
|
manual: true,
|
|
},
|
|
);
|
|
const onTest = async () => {
|
|
form.validateFields().then(async values => {
|
|
const extraVal = extraForm.getFieldsValue();
|
|
await recallTestRun({ recall_top_k: 1, recall_retrievers: options, ...values, ...extraVal });
|
|
});
|
|
};
|
|
|
|
return (
|
|
<Modal
|
|
title='召回测试'
|
|
width={'60%'}
|
|
open={open}
|
|
footer={false}
|
|
onCancel={() => setOpen(false)}
|
|
centered
|
|
destroyOnClose={true}
|
|
>
|
|
<Card
|
|
title='召回配置'
|
|
size='small'
|
|
className='my-4'
|
|
extra={
|
|
<Popover
|
|
placement='bottomRight'
|
|
trigger='hover'
|
|
title='向量检索设置'
|
|
content={
|
|
<Form
|
|
form={extraForm}
|
|
initialValues={{
|
|
recall_top_k: 1,
|
|
}}
|
|
>
|
|
<Form.Item label='Topk' tooltip='基于相似度得分的前 k 个向量' name='recall_top_k'>
|
|
<InputNumber placeholder='请输入' className='w-full' />
|
|
</Form.Item>
|
|
<Form.Item label='召回方法' name='recall_retrievers'>
|
|
<Select
|
|
mode='multiple'
|
|
options={options.map(item => {
|
|
return { label: item, value: item };
|
|
})}
|
|
className='w-full'
|
|
allowClear
|
|
disabled
|
|
/>
|
|
</Form.Item>
|
|
<Form.Item label='score阈值' name='recall_score_threshold'>
|
|
<InputNumber placeholder='请输入' className='w-full' step={0.1} />
|
|
</Form.Item>
|
|
</Form>
|
|
}
|
|
>
|
|
<SettingOutlined className='text-lg' />
|
|
</Popover>
|
|
}
|
|
>
|
|
<Form form={form} layout='vertical' onFinish={onTest}>
|
|
<Form.Item
|
|
label='测试问题'
|
|
required={true}
|
|
name='question'
|
|
rules={[{ required: true, message: '请输入测试问题' }]}
|
|
className='m-0 p-0'
|
|
>
|
|
<div className='flex w-full items-center gap-8'>
|
|
<Input placeholder='请输入测试问题' autoComplete='off' allowClear className='w-1/2' />
|
|
<Button type='primary' htmlType='submit'>
|
|
测试
|
|
</Button>
|
|
</div>
|
|
</Form.Item>
|
|
|
|
{/* {questions?.length > 0 && (
|
|
<Col span={16}>
|
|
<Form.Item label="推荐问题" tooltip="点击选择,自动填入">
|
|
<div className="flex flex-wrap gap-2">
|
|
{questions.map((item, index) => (
|
|
<Tag
|
|
color={tagColors[index]}
|
|
key={item}
|
|
className="cursor-pointer"
|
|
onClick={() => {
|
|
form.setFieldValue('question', item);
|
|
}}
|
|
>
|
|
{item}
|
|
</Tag>
|
|
))}
|
|
</div>
|
|
</Form.Item>
|
|
</Col>
|
|
)} */}
|
|
</Form>
|
|
</Card>
|
|
<Card title='召回结果' size='small'>
|
|
<Spin spinning={loading}>
|
|
{resultList.length > 0 ? (
|
|
<div
|
|
className='flex flex-col overflow-y-auto'
|
|
style={{
|
|
height: '45vh',
|
|
}}
|
|
>
|
|
{resultList.map(item => (
|
|
<Card
|
|
title={
|
|
<div className='flex items-center'>
|
|
<Tag color='blue'># {item.chunk_id}</Tag>
|
|
{item.metadata.prop_field.title}
|
|
</div>
|
|
}
|
|
extra={
|
|
<div className='flex items-center gap-2'>
|
|
<span className='font-semibold'>score:</span>
|
|
<span className='text-blue-500'>{item.score}</span>
|
|
</div>
|
|
}
|
|
key={item.chunk_id}
|
|
size='small'
|
|
className='mb-4 border-gray-500 shadow-md'
|
|
>
|
|
<MarkDownContext>{item.content}</MarkDownContext>
|
|
</Card>
|
|
))}
|
|
</div>
|
|
) : (
|
|
<Empty />
|
|
)}
|
|
</Spin>
|
|
</Card>
|
|
</Modal>
|
|
);
|
|
};
|
|
|
|
export default RecallTestModal;
|