| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useEffect, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   useMemo, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   useRef, | 
					
						
							|  |  |  |   useState, | 
					
						
							|  |  |  | } from 'react' | 
					
						
							|  |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							|  |  |  | import { produce, setAutoFreeze } from 'immer' | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | import { uniqBy } from 'lodash-es' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { useWorkflowRun } from '../../hooks' | 
					
						
							| 
									
										
										
										
											2024-05-11 16:23:31 +08:00
										 |  |  | import { NodeRunningStatus, WorkflowRunningStatus } from '../../types' | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  | import { useWorkflowStore } from '../../store' | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  | import { DEFAULT_ITER_TIMES, DEFAULT_LOOP_TIMES } from '../../constants' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import type { | 
					
						
							|  |  |  |   ChatItem, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   ChatItemInTree, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   Inputs, | 
					
						
							|  |  |  | } from '@/app/components/base/chat/types' | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | import type { InputForm } from '@/app/components/base/chat/chat/type' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   getProcessedInputs, | 
					
						
							|  |  |  |   processOpeningStatement, | 
					
						
							|  |  |  | } from '@/app/components/base/chat/chat/utils' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { useToastContext } from '@/app/components/base/toast' | 
					
						
							|  |  |  | import { TransferMethod } from '@/types/app' | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   getProcessedFiles, | 
					
						
							|  |  |  |   getProcessedFilesFromResponse, | 
					
						
							|  |  |  | } from '@/app/components/base/file-uploader/utils' | 
					
						
							|  |  |  | import type { FileEntity } from '@/app/components/base/file-uploader/types' | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  | import { getThreadMessages } from '@/app/components/base/chat/utils' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | type GetAbortController = (abortController: AbortController) => void | 
					
						
							|  |  |  | type SendCallback = { | 
					
						
							|  |  |  |   onGetSuggestedQuestions?: (responseItemId: string, getAbortController: GetAbortController) => Promise<any> | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | export const useChat = ( | 
					
						
							|  |  |  |   config: any, | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |   formSettings?: { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     inputs: Inputs | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     inputsForm: InputForm[] | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   prevChatTree?: ChatItemInTree[], | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   stopChat?: (taskId: string) => void, | 
					
						
							|  |  |  | ) => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							|  |  |  |   const { notify } = useToastContext() | 
					
						
							|  |  |  |   const { handleRun } = useWorkflowRun() | 
					
						
							|  |  |  |   const hasStopResponded = useRef(false) | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							| 
									
										
										
										
											2024-08-06 13:33:21 +08:00
										 |  |  |   const conversationId = useRef('') | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const taskIdRef = useRef('') | 
					
						
							|  |  |  |   const [isResponding, setIsResponding] = useState(false) | 
					
						
							|  |  |  |   const isRespondingRef = useRef(false) | 
					
						
							|  |  |  |   const [suggestedQuestions, setSuggestQuestions] = useState<string[]>([]) | 
					
						
							|  |  |  |   const suggestedQuestionsAbortControllerRef = useRef<AbortController | null>(null) | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |   const { | 
					
						
							|  |  |  |     setIterTimes, | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |     setLoopTimes, | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |   } = workflowStore.getState() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleResponding = useCallback((isResponding: boolean) => { | 
					
						
							|  |  |  |     setIsResponding(isResponding) | 
					
						
							|  |  |  |     isRespondingRef.current = isResponding | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   const [chatTree, setChatTree] = useState<ChatItemInTree[]>(prevChatTree || []) | 
					
						
							|  |  |  |   const chatTreeRef = useRef<ChatItemInTree[]>(chatTree) | 
					
						
							|  |  |  |   const [targetMessageId, setTargetMessageId] = useState<string>() | 
					
						
							|  |  |  |   const threadMessages = useMemo(() => getThreadMessages(chatTree, targetMessageId), [chatTree, targetMessageId]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const getIntroduction = useCallback((str: string) => { | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     return processOpeningStatement(str, formSettings?.inputs || {}, formSettings?.inputsForm || []) | 
					
						
							|  |  |  |   }, [formSettings?.inputs, formSettings?.inputsForm]) | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /** Final chat list that will be rendered */ | 
					
						
							|  |  |  |   const chatList = useMemo(() => { | 
					
						
							|  |  |  |     const ret = [...threadMessages] | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     if (config?.opening_statement) { | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       const index = threadMessages.findIndex(item => item.isOpeningStatement) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (index > -1) { | 
					
						
							|  |  |  |         ret[index] = { | 
					
						
							|  |  |  |           ...ret[index], | 
					
						
							|  |  |  |           content: getIntroduction(config.opening_statement), | 
					
						
							|  |  |  |           suggestedQuestions: config.suggested_questions, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         ret.unshift({ | 
					
						
							|  |  |  |           id: `${Date.now()}`, | 
					
						
							|  |  |  |           content: getIntroduction(config.opening_statement), | 
					
						
							|  |  |  |           isAnswer: true, | 
					
						
							|  |  |  |           isOpeningStatement: true, | 
					
						
							|  |  |  |           suggestedQuestions: config.suggested_questions, | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ret | 
					
						
							|  |  |  |   }, [threadMessages, config?.opening_statement, getIntroduction, config?.suggested_questions]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     setAutoFreeze(false) | 
					
						
							|  |  |  |     return () => { | 
					
						
							|  |  |  |       setAutoFreeze(true) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /** Find the target node by bfs and then operate on it */ | 
					
						
							|  |  |  |   const produceChatTreeNode = useCallback((targetId: string, operation: (node: ChatItemInTree) => void) => { | 
					
						
							|  |  |  |     return produce(chatTreeRef.current, (draft) => { | 
					
						
							|  |  |  |       const queue: ChatItemInTree[] = [...draft] | 
					
						
							|  |  |  |       while (queue.length > 0) { | 
					
						
							|  |  |  |         const current = queue.shift()! | 
					
						
							|  |  |  |         if (current.id === targetId) { | 
					
						
							|  |  |  |           operation(current) | 
					
						
							|  |  |  |           break | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (current.children) | 
					
						
							|  |  |  |           queue.push(...current.children) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |   }, []) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleStop = useCallback(() => { | 
					
						
							|  |  |  |     hasStopResponded.current = true | 
					
						
							|  |  |  |     handleResponding(false) | 
					
						
							|  |  |  |     if (stopChat && taskIdRef.current) | 
					
						
							|  |  |  |       stopChat(taskIdRef.current) | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |     setIterTimes(DEFAULT_ITER_TIMES) | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |     setLoopTimes(DEFAULT_LOOP_TIMES) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     if (suggestedQuestionsAbortControllerRef.current) | 
					
						
							|  |  |  |       suggestedQuestionsAbortControllerRef.current.abort() | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |   }, [handleResponding, setIterTimes, setLoopTimes, stopChat]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleRestart = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2024-08-06 13:33:21 +08:00
										 |  |  |     conversationId.current = '' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     taskIdRef.current = '' | 
					
						
							|  |  |  |     handleStop() | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |     setIterTimes(DEFAULT_ITER_TIMES) | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |     setLoopTimes(DEFAULT_LOOP_TIMES) | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     setChatTree([]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     setSuggestQuestions([]) | 
					
						
							|  |  |  |   }, [ | 
					
						
							|  |  |  |     handleStop, | 
					
						
							| 
									
										
										
										
											2024-11-05 10:32:49 +08:00
										 |  |  |     setIterTimes, | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |     setLoopTimes, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   ]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   const updateCurrentQAOnTree = useCallback(({ | 
					
						
							|  |  |  |     parentId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     responseItem, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     placeholderQuestionId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     questionItem, | 
					
						
							|  |  |  |   }: { | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     parentId?: string | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     responseItem: ChatItem | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     placeholderQuestionId: string | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     questionItem: ChatItem | 
					
						
							|  |  |  |   }) => { | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     let nextState: ChatItemInTree[] | 
					
						
							|  |  |  |     const currentQA = { ...questionItem, children: [{ ...responseItem, children: [] }] } | 
					
						
							|  |  |  |     if (!parentId && !chatTree.some(item => [placeholderQuestionId, questionItem.id].includes(item.id))) { | 
					
						
							|  |  |  |       // QA whose parent is not provided is considered as a first message of the conversation,
 | 
					
						
							|  |  |  |       // and it should be a root node of the chat tree
 | 
					
						
							|  |  |  |       nextState = produce(chatTree, (draft) => { | 
					
						
							|  |  |  |         draft.push(currentQA) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       // find the target QA in the tree and update it; if not found, insert it to its parent node
 | 
					
						
							|  |  |  |       nextState = produceChatTreeNode(parentId!, (parentNode) => { | 
					
						
							|  |  |  |         const questionNodeIndex = parentNode.children!.findIndex(item => [placeholderQuestionId, questionItem.id].includes(item.id)) | 
					
						
							|  |  |  |         if (questionNodeIndex === -1) | 
					
						
							|  |  |  |           parentNode.children!.push(currentQA) | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           parentNode.children![questionNodeIndex] = currentQA | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       }) | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     setChatTree(nextState) | 
					
						
							|  |  |  |     chatTreeRef.current = nextState | 
					
						
							|  |  |  |   }, [chatTree, produceChatTreeNode]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleSend = useCallback(( | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     params: { | 
					
						
							|  |  |  |       query: string | 
					
						
							|  |  |  |       files?: FileEntity[] | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       parent_message_id?: string | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       [key: string]: any | 
					
						
							|  |  |  |     }, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     { | 
					
						
							|  |  |  |       onGetSuggestedQuestions, | 
					
						
							|  |  |  |     }: SendCallback, | 
					
						
							|  |  |  |   ) => { | 
					
						
							|  |  |  |     if (isRespondingRef.current) { | 
					
						
							|  |  |  |       notify({ type: 'info', message: t('appDebug.errorMessage.waitForResponse') }) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     const parentMessage = threadMessages.find(item => item.id === params.parent_message_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const placeholderQuestionId = `question-${Date.now()}` | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const questionItem = { | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       id: placeholderQuestionId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       content: params.query, | 
					
						
							|  |  |  |       isAnswer: false, | 
					
						
							|  |  |  |       message_files: params.files, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       parentMessageId: params.parent_message_id, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const placeholderAnswerId = `answer-placeholder-${Date.now()}` | 
					
						
							|  |  |  |     const placeholderAnswerItem = { | 
					
						
							|  |  |  |       id: placeholderAnswerId, | 
					
						
							|  |  |  |       content: '', | 
					
						
							|  |  |  |       isAnswer: true, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       parentMessageId: questionItem.id, | 
					
						
							|  |  |  |       siblingIndex: parentMessage?.children?.length ?? chatTree.length, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     setTargetMessageId(parentMessage?.id) | 
					
						
							|  |  |  |     updateCurrentQAOnTree({ | 
					
						
							|  |  |  |       parentId: params.parent_message_id, | 
					
						
							|  |  |  |       responseItem: placeholderAnswerItem, | 
					
						
							|  |  |  |       placeholderQuestionId, | 
					
						
							|  |  |  |       questionItem, | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // answer
 | 
					
						
							|  |  |  |     const responseItem: ChatItem = { | 
					
						
							| 
									
										
										
										
											2024-05-11 16:23:31 +08:00
										 |  |  |       id: placeholderAnswerId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       content: '', | 
					
						
							|  |  |  |       agent_thoughts: [], | 
					
						
							|  |  |  |       message_files: [], | 
					
						
							|  |  |  |       isAnswer: true, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |       parentMessageId: questionItem.id, | 
					
						
							|  |  |  |       siblingIndex: parentMessage?.children?.length ?? chatTree.length, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handleResponding(true) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     const { files, inputs, ...restParams } = params | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const bodyParams = { | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       files: getProcessedFiles(files || []), | 
					
						
							|  |  |  |       inputs: getProcessedInputs(inputs || {}, formSettings?.inputsForm || []), | 
					
						
							|  |  |  |       ...restParams, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (bodyParams?.files?.length) { | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       bodyParams.files = bodyParams.files.map((item) => { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         if (item.transfer_method === TransferMethod.local_file) { | 
					
						
							|  |  |  |           return { | 
					
						
							|  |  |  |             ...item, | 
					
						
							|  |  |  |             url: '', | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return item | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     let hasSetResponseId = false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     handleRun( | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       bodyParams, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       { | 
					
						
							|  |  |  |         onData: (message: string, isFirstMessage: boolean, { conversationId: newConversationId, messageId, taskId }: any) => { | 
					
						
							|  |  |  |           responseItem.content = responseItem.content + message | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (messageId && !hasSetResponseId) { | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |             questionItem.id = `question-${messageId}` | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             responseItem.id = messageId | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |             responseItem.parentMessageId = questionItem.id | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             hasSetResponseId = true | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (isFirstMessage && newConversationId) | 
					
						
							| 
									
										
										
										
											2024-08-06 13:33:21 +08:00
										 |  |  |             conversationId.current = newConversationId | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |           taskIdRef.current = taskId | 
					
						
							|  |  |  |           if (messageId) | 
					
						
							|  |  |  |             responseItem.id = messageId | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             questionItem, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |           }) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         async onCompleted(hasError?: boolean, errorMessage?: string) { | 
					
						
							|  |  |  |           handleResponding(false) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (hasError) { | 
					
						
							|  |  |  |             if (errorMessage) { | 
					
						
							|  |  |  |               responseItem.content = errorMessage | 
					
						
							|  |  |  |               responseItem.isError = true | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |               updateCurrentQAOnTree({ | 
					
						
							|  |  |  |                 placeholderQuestionId, | 
					
						
							|  |  |  |                 questionItem, | 
					
						
							|  |  |  |                 responseItem, | 
					
						
							|  |  |  |                 parentId: params.parent_message_id, | 
					
						
							|  |  |  |               }) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             } | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (config?.suggested_questions_after_answer?.enabled && !hasStopResponded.current && onGetSuggestedQuestions) { | 
					
						
							| 
									
										
										
										
											2024-09-03 14:09:46 +08:00
										 |  |  |             try { | 
					
						
							|  |  |  |               const { data }: any = await onGetSuggestedQuestions( | 
					
						
							|  |  |  |                 responseItem.id, | 
					
						
							|  |  |  |                 newAbortController => suggestedQuestionsAbortControllerRef.current = newAbortController, | 
					
						
							|  |  |  |               ) | 
					
						
							|  |  |  |               setSuggestQuestions(data) | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |             // eslint-disable-next-line unused-imports/no-unused-vars
 | 
					
						
							| 
									
										
										
										
											2024-09-03 14:09:46 +08:00
										 |  |  |             catch (error) { | 
					
						
							|  |  |  |               setSuggestQuestions([]) | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onMessageEnd: (messageEnd) => { | 
					
						
							|  |  |  |           responseItem.citation = messageEnd.metadata?.retriever_resources || [] | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |           const processedFilesFromResponse = getProcessedFilesFromResponse(messageEnd.files || []) | 
					
						
							|  |  |  |           responseItem.allFiles = uniqBy([...(responseItem.allFiles || []), ...(processedFilesFromResponse || [])], 'id') | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         onMessageReplace: (messageReplace) => { | 
					
						
							|  |  |  |           responseItem.content = messageReplace.answer | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onError() { | 
					
						
							|  |  |  |           handleResponding(false) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onWorkflowStarted: ({ workflow_run_id, task_id }) => { | 
					
						
							|  |  |  |           taskIdRef.current = task_id | 
					
						
							|  |  |  |           responseItem.workflow_run_id = workflow_run_id | 
					
						
							|  |  |  |           responseItem.workflowProcess = { | 
					
						
							|  |  |  |             status: WorkflowRunningStatus.Running, | 
					
						
							|  |  |  |             tracing: [], | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         onWorkflowFinished: ({ data }) => { | 
					
						
							|  |  |  |           responseItem.workflowProcess!.status = data.status as WorkflowRunningStatus | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         onIterationStart: ({ data }) => { | 
					
						
							| 
									
										
										
										
											2024-05-11 16:23:31 +08:00
										 |  |  |           responseItem.workflowProcess!.tracing!.push({ | 
					
						
							|  |  |  |             ...data, | 
					
						
							|  |  |  |             status: NodeRunningStatus.Running, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           }) | 
					
						
							|  |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |         onIterationFinish: ({ data }) => { | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |           const currentTracingIndex = responseItem.workflowProcess!.tracing!.findIndex(item => item.id === data.id) | 
					
						
							|  |  |  |           if (currentTracingIndex > -1) { | 
					
						
							|  |  |  |             responseItem.workflowProcess!.tracing[currentTracingIndex] = { | 
					
						
							|  |  |  |               ...responseItem.workflowProcess!.tracing[currentTracingIndex], | 
					
						
							|  |  |  |               ...data, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             updateCurrentQAOnTree({ | 
					
						
							|  |  |  |               placeholderQuestionId, | 
					
						
							|  |  |  |               questionItem, | 
					
						
							|  |  |  |               responseItem, | 
					
						
							|  |  |  |               parentId: params.parent_message_id, | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2025-03-05 17:41:15 +08:00
										 |  |  |         onLoopStart: ({ data }) => { | 
					
						
							|  |  |  |           responseItem.workflowProcess!.tracing!.push({ | 
					
						
							|  |  |  |             ...data, | 
					
						
							|  |  |  |             status: NodeRunningStatus.Running, | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onLoopFinish: ({ data }) => { | 
					
						
							|  |  |  |           const currentTracingIndex = responseItem.workflowProcess!.tracing!.findIndex(item => item.id === data.id) | 
					
						
							|  |  |  |           if (currentTracingIndex > -1) { | 
					
						
							|  |  |  |             responseItem.workflowProcess!.tracing[currentTracingIndex] = { | 
					
						
							|  |  |  |               ...responseItem.workflowProcess!.tracing[currentTracingIndex], | 
					
						
							|  |  |  |               ...data, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             updateCurrentQAOnTree({ | 
					
						
							|  |  |  |               placeholderQuestionId, | 
					
						
							|  |  |  |               questionItem, | 
					
						
							|  |  |  |               responseItem, | 
					
						
							|  |  |  |               parentId: params.parent_message_id, | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         onNodeStarted: ({ data }) => { | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |           responseItem.workflowProcess!.tracing!.push({ | 
					
						
							|  |  |  |             ...data, | 
					
						
							|  |  |  |             status: NodeRunningStatus.Running, | 
					
						
							|  |  |  |           } as any) | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2024-12-20 15:44:37 +08:00
										 |  |  |         onNodeRetry: ({ data }) => { | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |           responseItem.workflowProcess!.tracing!.push(data) | 
					
						
							| 
									
										
										
										
											2024-12-20 15:44:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-08 14:38:10 +08:00
										 |  |  |           updateCurrentQAOnTree({ | 
					
						
							|  |  |  |             placeholderQuestionId, | 
					
						
							|  |  |  |             questionItem, | 
					
						
							|  |  |  |             responseItem, | 
					
						
							|  |  |  |             parentId: params.parent_message_id, | 
					
						
							|  |  |  |           }) | 
					
						
							| 
									
										
										
										
											2024-12-20 15:44:37 +08:00
										 |  |  |         }, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         onNodeFinished: ({ data }) => { | 
					
						
							| 
									
										
										
										
											2025-02-17 17:05:13 +08:00
										 |  |  |           const currentTracingIndex = responseItem.workflowProcess!.tracing!.findIndex(item => item.id === data.id) | 
					
						
							|  |  |  |           if (currentTracingIndex > -1) { | 
					
						
							|  |  |  |             responseItem.workflowProcess!.tracing[currentTracingIndex] = { | 
					
						
							|  |  |  |               ...responseItem.workflowProcess!.tracing[currentTracingIndex], | 
					
						
							|  |  |  |               ...data, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             updateCurrentQAOnTree({ | 
					
						
							|  |  |  |               placeholderQuestionId, | 
					
						
							|  |  |  |               questionItem, | 
					
						
							|  |  |  |               responseItem, | 
					
						
							|  |  |  |               parentId: params.parent_message_id, | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onAgentLog: ({ data }) => { | 
					
						
							|  |  |  |           const currentNodeIndex = responseItem.workflowProcess!.tracing!.findIndex(item => item.node_id === data.node_id) | 
					
						
							|  |  |  |           if (currentNodeIndex > -1) { | 
					
						
							|  |  |  |             const current = responseItem.workflowProcess!.tracing![currentNodeIndex] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (current.execution_metadata) { | 
					
						
							|  |  |  |               if (current.execution_metadata.agent_log) { | 
					
						
							|  |  |  |                 const currentLogIndex = current.execution_metadata.agent_log.findIndex(log => log.id === data.id) | 
					
						
							|  |  |  |                 if (currentLogIndex > -1) { | 
					
						
							|  |  |  |                   current.execution_metadata.agent_log[currentLogIndex] = { | 
					
						
							|  |  |  |                     ...current.execution_metadata.agent_log[currentLogIndex], | 
					
						
							|  |  |  |                     ...data, | 
					
						
							|  |  |  |                   } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else { | 
					
						
							|  |  |  |                   current.execution_metadata.agent_log.push(data) | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               else { | 
					
						
							|  |  |  |                 current.execution_metadata.agent_log = [data] | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |               current.execution_metadata = { | 
					
						
							|  |  |  |                 agent_log: [data], | 
					
						
							|  |  |  |               } as any | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             responseItem.workflowProcess!.tracing[currentNodeIndex] = { | 
					
						
							|  |  |  |               ...current, | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             updateCurrentQAOnTree({ | 
					
						
							|  |  |  |               placeholderQuestionId, | 
					
						
							|  |  |  |               questionItem, | 
					
						
							|  |  |  |               responseItem, | 
					
						
							|  |  |  |               parentId: params.parent_message_id, | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |   }, [threadMessages, chatTree.length, updateCurrentQAOnTree, handleResponding, formSettings?.inputsForm, handleRun, notify, t, config?.suggested_questions_after_answer?.enabled]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							| 
									
										
										
										
											2024-08-06 13:33:21 +08:00
										 |  |  |     conversationId: conversationId.current, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     chatList, | 
					
						
							| 
									
										
										
										
											2025-01-31 13:05:10 +08:00
										 |  |  |     setTargetMessageId, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     handleSend, | 
					
						
							|  |  |  |     handleStop, | 
					
						
							|  |  |  |     handleRestart, | 
					
						
							|  |  |  |     isResponding, | 
					
						
							|  |  |  |     suggestedQuestions, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |