| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { useCallback } from 'react' | 
					
						
							|  |  |  | import produce from 'immer' | 
					
						
							|  |  |  | import { useStoreApi } from 'reactflow' | 
					
						
							|  |  |  | import { useParams } from 'next/navigation' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useStore, | 
					
						
							|  |  |  |   useWorkflowStore, | 
					
						
							|  |  |  | } from '../store' | 
					
						
							|  |  |  | import { BlockEnum } from '../types' | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  | import { useWorkflowUpdate } from '../hooks' | 
					
						
							| 
									
										
										
										
											2024-08-19 18:11:11 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   useNodesReadOnly, | 
					
						
							|  |  |  | } from './use-workflow' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { syncWorkflowDraft } from '@/service/workflow' | 
					
						
							|  |  |  | import { useFeaturesStore } from '@/app/components/base/features/hooks' | 
					
						
							|  |  |  | import { API_PREFIX } from '@/config' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useNodesSyncDraft = () => { | 
					
						
							|  |  |  |   const store = useStoreApi() | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  |   const featuresStore = useFeaturesStore() | 
					
						
							|  |  |  |   const { getNodesReadOnly } = useNodesReadOnly() | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   const { handleRefreshWorkflowDraft } = useWorkflowUpdate() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const debouncedSyncWorkflowDraft = useStore(s => s.debouncedSyncWorkflowDraft) | 
					
						
							|  |  |  |   const params = useParams() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   const getPostParams = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |       transform, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const [x, y, zoom] = transform | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |     const { | 
					
						
							|  |  |  |       appId, | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |       conversationVariables, | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |       environmentVariables, | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |       syncWorkflowDraftHash, | 
					
						
							|  |  |  |     } = workflowStore.getState() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |     if (appId) { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       const nodes = getNodes() | 
					
						
							|  |  |  |       const hasStartNode = nodes.find(node => node.data.type === BlockEnum.Start) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!hasStartNode) | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const features = featuresStore!.getState().features | 
					
						
							|  |  |  |       const producedNodes = produce(nodes, (draft) => { | 
					
						
							|  |  |  |         draft.forEach((node) => { | 
					
						
							|  |  |  |           Object.keys(node.data).forEach((key) => { | 
					
						
							|  |  |  |             if (key.startsWith('_')) | 
					
						
							|  |  |  |               delete node.data[key] | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       const producedEdges = produce(edges, (draft) => { | 
					
						
							|  |  |  |         draft.forEach((edge) => { | 
					
						
							|  |  |  |           Object.keys(edge.data).forEach((key) => { | 
					
						
							|  |  |  |             if (key.startsWith('_')) | 
					
						
							|  |  |  |               delete edge.data[key] | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       return { | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |         url: `/apps/${appId}/workflows/draft`, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         params: { | 
					
						
							|  |  |  |           graph: { | 
					
						
							|  |  |  |             nodes: producedNodes, | 
					
						
							|  |  |  |             edges: producedEdges, | 
					
						
							|  |  |  |             viewport: { | 
					
						
							|  |  |  |               x, | 
					
						
							|  |  |  |               y, | 
					
						
							|  |  |  |               zoom, | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           features: { | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |             opening_statement: features.opening?.enabled ? (features.opening?.opening_statement || '') : '', | 
					
						
							|  |  |  |             suggested_questions: features.opening?.enabled ? (features.opening?.suggested_questions || []) : [], | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |             suggested_questions_after_answer: features.suggested, | 
					
						
							|  |  |  |             text_to_speech: features.text2speech, | 
					
						
							|  |  |  |             speech_to_text: features.speech2text, | 
					
						
							|  |  |  |             retriever_resource: features.citation, | 
					
						
							|  |  |  |             sensitive_word_avoidance: features.moderation, | 
					
						
							|  |  |  |             file_upload: features.file, | 
					
						
							|  |  |  |           }, | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |           environment_variables: environmentVariables, | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |           conversation_variables: conversationVariables, | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |           hash: syncWorkflowDraftHash, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |         }, | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [store, featuresStore, workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const syncWorkflowDraftWhenPageClose = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2024-04-28 17:09:56 +08:00
										 |  |  |     if (getNodesReadOnly()) | 
					
						
							|  |  |  |       return | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const postParams = getPostParams() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (postParams) { | 
					
						
							|  |  |  |       navigator.sendBeacon( | 
					
						
							|  |  |  |         `${API_PREFIX}/apps/${params.appId}/workflows/draft?_token=${localStorage.getItem('console_token')}`, | 
					
						
							|  |  |  |         JSON.stringify(postParams.params), | 
					
						
							|  |  |  |       ) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-28 17:09:56 +08:00
										 |  |  |   }, [getPostParams, params.appId, getNodesReadOnly]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   const doSyncWorkflowDraft = useCallback(async (notRefreshWhenSyncError?: boolean) => { | 
					
						
							| 
									
										
										
										
											2024-04-28 17:09:56 +08:00
										 |  |  |     if (getNodesReadOnly()) | 
					
						
							|  |  |  |       return | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |     const postParams = getPostParams() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (postParams) { | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |       const { | 
					
						
							|  |  |  |         setSyncWorkflowDraftHash, | 
					
						
							|  |  |  |         setDraftUpdatedAt, | 
					
						
							|  |  |  |       } = workflowStore.getState() | 
					
						
							|  |  |  |       try { | 
					
						
							|  |  |  |         const res = await syncWorkflowDraft(postParams) | 
					
						
							|  |  |  |         setSyncWorkflowDraftHash(res.hash) | 
					
						
							|  |  |  |         setDraftUpdatedAt(res.updated_at) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       catch (error: any) { | 
					
						
							|  |  |  |         if (error && error.json && !error.bodyUsed) { | 
					
						
							|  |  |  |           error.json().then((err: any) => { | 
					
						
							|  |  |  |             if (err.code === 'draft_workflow_not_sync' && !notRefreshWhenSyncError) | 
					
						
							|  |  |  |               handleRefreshWorkflowDraft() | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   }, [workflowStore, getPostParams, getNodesReadOnly, handleRefreshWorkflowDraft]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   const handleSyncWorkflowDraft = useCallback((sync?: boolean, notRefreshWhenSyncError?: boolean) => { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     if (getNodesReadOnly()) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (sync) | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |       doSyncWorkflowDraft(notRefreshWhenSyncError) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     else | 
					
						
							|  |  |  |       debouncedSyncWorkflowDraft(doSyncWorkflowDraft) | 
					
						
							|  |  |  |   }, [debouncedSyncWorkflowDraft, doSyncWorkflowDraft, getNodesReadOnly]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     doSyncWorkflowDraft, | 
					
						
							|  |  |  |     handleSyncWorkflowDraft, | 
					
						
							|  |  |  |     syncWorkflowDraftWhenPageClose, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |