import { type FC, type ReactNode, useEffect, useRef, useState, forwardRef, useImperativeHandle } from 'react';
import clsx from 'clsx'
import { useTranslation } from 'react-i18next'
import { useParams } from 'react-router-dom';
import { Row, Col, Space, Form, Input, Switch, Button, App, Spin } from 'antd'
import Chat from './components/Chat'
import RbCard from '@/components/RbCard/Card'
import Card from './components/Card'
import ModelConfigModal from './components/ModelConfigModal'
import type {
ModelConfigModalRef,
ChatData,
Config,
ModelConfig,
AgentRef,
MemoryConfig,
AiPromptModalRef,
Source,
ChatVariableConfigModalRef
} from './types'
import type { Variable } from './components/VariableList/types'
import type { KnowledgeConfig } from './components/Knowledge/types'
import type { ModelListItem } from '@/views/ModelManagement/types'
import { getModelList } from '@/api/models';
import { saveAgentConfig } from '@/api/application'
import Knowledge from './components/Knowledge/Knowledge'
import VariableList from './components/VariableList/VariableList'
import { getApplicationConfig } from '@/api/application'
import { memoryConfigListUrl } from '@/api/memory'
import CustomSelect from '@/components/CustomSelect'
import aiPrompt from '@/assets/images/application/aiPrompt.png'
import AiPromptModal from './components/AiPromptModal'
import ToolList from './components/ToolList/ToolList'
import ChatVariableConfigModal from './components/ChatVariableConfigModal';
const DescWrapper: FC<{desc: string, className?: string}> = ({desc, className}) => {
return (
{desc}
)
}
const LabelWrapper: FC<{title: string, className?: string; children?: ReactNode}> = ({title, className, children}) => {
return (
{title}
{children}
)
}
const SwitchWrapper: FC<{ title: string, desc?: string, name: string | string[]; needTransition?: boolean; }> = ({ title, desc, name, needTransition = true }) => {
const { t } = useTranslation();
return (
{desc && }
)
}
const SelectWrapper: FC<{ title: string, desc: string, name: string | string[], url: string }> = ({ title, desc, name, url }) => {
const { t } = useTranslation();
return (
<>
>
)
}
const Agent = forwardRef((_props, ref) => {
const { t } = useTranslation()
const { id } = useParams();
const { message } = App.useApp()
const [form] = Form.useForm()
const [loading, setLoading] = useState(false)
const [data, setData] = useState(null);
const modelConfigModalRef = useRef(null)
const [modelList, setModelList] = useState([])
const [defaultModel, setDefaultModel] = useState(null)
const [chatList, setChatList] = useState([])
const values = Form.useWatch([], form)
const [isSave, setIsSave] = useState(false)
const initialized = useRef(false)
// 初始化完成标记
useEffect(() => {
if (data) {
initialized.current = true
}
}, [data])
useEffect(() => {
if (!initialized.current) return
if (isSave) return
setIsSave(true)
}, [values])
useEffect(() => {
getModels()
getData()
}, [])
const getData = () => {
setLoading(true)
getApplicationConfig(id as string).then(res => {
const response = res as Config
let allTools = Array.isArray(response.tools) ? response.tools : []
form.setFieldsValue({
...response,
tools: allTools,
memory: {
...response.memory,
memory_content: response.memory?.memory_content ? Number(response.memory?.memory_content) : undefined
}
})
setData({
...response,
tools: allTools
})
}).finally(() => {
setLoading(false)
})
}
const refresh = (vo: ModelConfig, type: Source) => {
if (type === 'model') {
const { default_model_config_id, ...rest } = vo
form.setFieldsValue({
default_model_config_id,
model_parameters: {...rest}
})
if (default_model_config_id === values?.default_model_config_id) {
setChatList([{
label: vo.label || '',
model_config_id: default_model_config_id || '',
model_parameters: {...rest},
list: []
}])
}
} else if (type === 'chat') {
if (chatList.length >= 4) {
message.warning(t('application.maxChatCount'))
return
}
const { label, default_model_config_id, ...reset } = vo
setChatList((prev: ChatData[]) => {
const newChatItem: ChatData = {
label,
model_config_id: default_model_config_id || '',
model_parameters: {...reset},
list: []
};
return [
...(prev || []).map(item => ({
...item,
conversation_id: undefined,
list: []
})),
newChatItem
];
})
}
}
const handleModelConfig = () => {
modelConfigModalRef.current?.handleOpen('model')
}
const handleClearDebugging = () => {
setChatList([])
}
// 保存Agent配置
const handleSave = (flag = true) => {
if (!isSave || !data) return Promise.resolve()
const { memory, knowledge_retrieval, tools, ...rest } = values
const { knowledge_bases = [], ...knowledgeRest } = knowledge_retrieval || {}
const { memory_content } = memory || {}
// 从原数据中获取memory的其他必要属性
const originalMemory = data.memory || ({} as MemoryConfig)
const params: Config = {
...data,
...rest,
memory: {
...originalMemory,
...memory,
memory_content: memory_content ? String(memory_content) : '',
},
knowledge_retrieval: knowledge_bases.length > 0 ? {
...data.knowledge_retrieval,
...knowledgeRest,
knowledge_bases: knowledge_bases.map(item => ({
kb_id: item.kb_id || item.id,
...(item.config || {})
}))
} as KnowledgeConfig : null,
tools: tools.map(vo => ({
tool_id: vo.tool_id,
operation: vo.operation,
enabled: vo.enabled
}))
}
return new Promise((resolve, reject) => {
saveAgentConfig(data.app_id, params)
.then((res) => {
if (flag) {
message.success(t('common.saveSuccess'))
}
setIsSave(false)
resolve(res)
}).catch(error => {
reject(error)
})
})
}
const getModels = () => {
getModelList({ type: 'llm,chat', pagesize: 100, page: 1, is_active: true })
.then(res => {
const response = res as { items: ModelListItem[] }
setModelList(response.items)
})
}
const handleAddModel = () => {
modelConfigModalRef.current?.handleOpen('chat')
}
useEffect(() => {
if (values?.default_model_config_id && modelList.length > 0) {
const filterValue = modelList.find(item => item.id === values.default_model_config_id)
setDefaultModel(filterValue as ModelListItem | null)
setChatList([{
label: filterValue?.name || '',
model_config_id: filterValue?.id || '',
model_parameters: {...(filterValue?.config || {})} as unknown as ModelConfig,
list: []
}])
}
}, [modelList, values?.default_model_config_id])
useImperativeHandle(ref, () => ({
handleSave
}))
const aiPromptModalRef = useRef(null)
const handlePrompt = () => {
aiPromptModalRef.current?.handleOpen()
}
const updatePrompt = (value: string) => {
form.setFieldValue('system_prompt', value)
const variables = value.match(/\{\{([^}]+)\}\}/g)?.map(match => match.slice(2, -2)) || []
const uniqueVariables = [...new Set(variables)]
const newVariableList: Variable[] = uniqueVariables.map((name, index) => ({
index,
type: 'text',
name,
display_name: name,
required: false
}))
updateVariableList(newVariableList)
}
const updateVariableList = (list: Variable[]) => {
form.setFieldValue('variables', [...list])
setChatVariables([...list])
}
const chatVariableConfigModalRef = useRef(null)
const [chatVariables, setChatVariables] = useState([])
const handleOpenVariableConfig = () => {
chatVariableConfigModalRef.current?.handleOpen(chatVariables)
}
const handleSaveChatVariable = (values: Variable[]) => {
setChatVariables(values)
}
console.log('values', values)
return (
<>
{loading && }
{t('application.configuration')}
({t('application.configurationDesc')})
{/* 记忆配置 */}
{/* 工具配置 */}
{t('application.debuggingAndPreview')}
>
);
});
export default Agent;