DB-GPT/web/components/knowledge/RecallTestModal.tsx
Dreammy23 471689ba20
feat(web): Unified frontend code style (#1923)
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>
2024-08-30 14:03:06 +08:00

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;