| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useEffect, | 
					
						
							|  |  |  |   useMemo, | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  |   useState, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | } from 'react' | 
					
						
							|  |  |  | import dayjs from 'dayjs' | 
					
						
							|  |  |  | import { uniqBy } from 'lodash-es' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { useContext } from 'use-context-selector' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   getIncomers, | 
					
						
							|  |  |  |   getOutgoers, | 
					
						
							|  |  |  |   useStoreApi, | 
					
						
							|  |  |  | } from 'reactflow' | 
					
						
							|  |  |  | import type { | 
					
						
							|  |  |  |   Connection, | 
					
						
							|  |  |  | } from 'reactflow' | 
					
						
							|  |  |  | import type { | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   Edge, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   Node, | 
					
						
							|  |  |  |   ValueSelector, | 
					
						
							|  |  |  | } from '../types' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   BlockEnum, | 
					
						
							|  |  |  |   WorkflowRunningStatus, | 
					
						
							|  |  |  | } from '../types' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useStore, | 
					
						
							|  |  |  |   useWorkflowStore, | 
					
						
							|  |  |  | } from '../store' | 
					
						
							|  |  |  | import { | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |   getParallelInfo, | 
					
						
							|  |  |  | } from '../utils' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   PARALLEL_DEPTH_LIMIT, | 
					
						
							|  |  |  |   PARALLEL_LIMIT, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   SUPPORT_OUTPUT_VARS_NODE, | 
					
						
							|  |  |  | } from '../constants' | 
					
						
							| 
									
										
										
										
											2024-06-14 17:08:11 +08:00
										 |  |  | import { CUSTOM_NOTE_NODE } from '../note-node/constants' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { findUsedVarNodes, getNodeOutputVars, updateNodeVars } from '../nodes/_base/components/variable/utils' | 
					
						
							|  |  |  | import { useNodesExtraData } from './use-nodes-data' | 
					
						
							|  |  |  | import { useWorkflowTemplate } from './use-workflow-template' | 
					
						
							|  |  |  | import { useStore as useAppStore } from '@/app/components/app/store' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   fetchNodesDefaultConfigs, | 
					
						
							|  |  |  |   fetchPublishedWorkflow, | 
					
						
							|  |  |  |   fetchWorkflowDraft, | 
					
						
							|  |  |  |   syncWorkflowDraft, | 
					
						
							|  |  |  | } from '@/service/workflow' | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  | import type { FetchWorkflowDraftResponse } from '@/types/workflow' | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   fetchAllBuiltInTools, | 
					
						
							|  |  |  |   fetchAllCustomTools, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   fetchAllWorkflowTools, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | } from '@/service/tools' | 
					
						
							|  |  |  | import I18n from '@/context/i18n' | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | import { CollectionType } from '@/app/components/tools/types' | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  | import { CUSTOM_ITERATION_START_NODE } from '@/app/components/workflow/nodes/iteration-start/constants' | 
					
						
							| 
									
										
										
										
											2024-04-23 17:02:23 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | export const useIsChatMode = () => { | 
					
						
							|  |  |  |   const appDetail = useAppStore(s => s.appDetail) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return appDetail?.mode === 'advanced-chat' | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useWorkflow = () => { | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |   const { t } = useTranslation() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const { locale } = useContext(I18n) | 
					
						
							|  |  |  |   const store = useStoreApi() | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  |   const nodesExtraData = useNodesExtraData() | 
					
						
							| 
									
										
										
										
											2024-04-26 14:23:13 +08:00
										 |  |  |   const setPanelWidth = useCallback((width: number) => { | 
					
						
							|  |  |  |     localStorage.setItem('workflow-node-panel-width', `${width}`) | 
					
						
							|  |  |  |     workflowStore.setState({ panelWidth: width }) | 
					
						
							|  |  |  |   }, [workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const getTreeLeafNodes = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     let startNode = nodes.find(node => node.data.type === BlockEnum.Start) | 
					
						
							|  |  |  |     const currentNode = nodes.find(node => node.id === nodeId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (currentNode?.parentId) | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |       startNode = nodes.find(node => node.parentId === currentNode.parentId && node.type === CUSTOM_ITERATION_START_NODE) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (!startNode) | 
					
						
							|  |  |  |       return [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const list: Node[] = [] | 
					
						
							|  |  |  |     const preOrder = (root: Node, callback: (node: Node) => void) => { | 
					
						
							|  |  |  |       if (root.id === nodeId) | 
					
						
							|  |  |  |         return | 
					
						
							|  |  |  |       const outgoers = getOutgoers(root, nodes, edges) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (outgoers.length) { | 
					
						
							|  |  |  |         outgoers.forEach((outgoer) => { | 
					
						
							|  |  |  |           preOrder(outgoer, callback) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							|  |  |  |         if (root.id !== nodeId) | 
					
						
							|  |  |  |           callback(root) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     preOrder(startNode, (node) => { | 
					
						
							|  |  |  |       list.push(node) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const incomers = getIncomers({ id: nodeId } as Node, nodes, edges) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     list.push(...incomers) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return uniqBy(list, 'id').filter((item) => { | 
					
						
							|  |  |  |       return SUPPORT_OUTPUT_VARS_NODE.includes(item.data.type) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   const getBeforeNodesInSameBranch = useCallback((nodeId: string, newNodes?: Node[], newEdges?: Edge[]) => { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     const nodes = newNodes || getNodes() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const currentNode = nodes.find(node => node.id === nodeId) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const list: Node[] = [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!currentNode) | 
					
						
							|  |  |  |       return list | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     if (currentNode.parentId) { | 
					
						
							|  |  |  |       const parentNode = nodes.find(node => node.id === currentNode.parentId) | 
					
						
							|  |  |  |       if (parentNode) { | 
					
						
							|  |  |  |         const parentList = getBeforeNodesInSameBranch(parentNode.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         list.push(...parentList) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const traverse = (root: Node, callback: (node: Node) => void) => { | 
					
						
							|  |  |  |       if (root) { | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |         const incomers = getIncomers(root, nodes, newEdges || edges) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (incomers.length) { | 
					
						
							|  |  |  |           incomers.forEach((node) => { | 
					
						
							| 
									
										
										
										
											2024-04-09 12:24:41 +08:00
										 |  |  |             if (!list.find(n => node.id === n.id)) { | 
					
						
							|  |  |  |               callback(node) | 
					
						
							|  |  |  |               traverse(node, callback) | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |           }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     traverse(currentNode, (node) => { | 
					
						
							|  |  |  |       list.push(node) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const length = list.length | 
					
						
							|  |  |  |     if (length) { | 
					
						
							|  |  |  |       return uniqBy(list, 'id').reverse().filter((item) => { | 
					
						
							|  |  |  |         return SUPPORT_OUTPUT_VARS_NODE.includes(item.data.type) | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return [] | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   const getBeforeNodesInSameBranchIncludeParent = useCallback((nodeId: string, newNodes?: Node[], newEdges?: Edge[]) => { | 
					
						
							|  |  |  |     const nodes = getBeforeNodesInSameBranch(nodeId, newNodes, newEdges) | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const allNodes = getNodes() | 
					
						
							|  |  |  |     const node = allNodes.find(n => n.id === nodeId) | 
					
						
							|  |  |  |     const parentNodeId = node?.parentId | 
					
						
							|  |  |  |     const parentNode = allNodes.find(n => n.id === parentNodeId) | 
					
						
							|  |  |  |     if (parentNode) | 
					
						
							|  |  |  |       nodes.push(parentNode) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nodes | 
					
						
							|  |  |  |   }, [getBeforeNodesInSameBranch, store]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const getAfterNodesInSameBranch = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const currentNode = nodes.find(node => node.id === nodeId)! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!currentNode) | 
					
						
							|  |  |  |       return [] | 
					
						
							|  |  |  |     const list: Node[] = [currentNode] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const traverse = (root: Node, callback: (node: Node) => void) => { | 
					
						
							|  |  |  |       if (root) { | 
					
						
							|  |  |  |         const outgoers = getOutgoers(root, nodes, edges) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (outgoers.length) { | 
					
						
							|  |  |  |           outgoers.forEach((node) => { | 
					
						
							|  |  |  |             callback(node) | 
					
						
							|  |  |  |             traverse(node, callback) | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     traverse(currentNode, (node) => { | 
					
						
							|  |  |  |       list.push(node) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return uniqBy(list, 'id') | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const getBeforeNodeById = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const node = nodes.find(node => node.id === nodeId)! | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return getIncomers(node, nodes, edges) | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   const getIterationNodeChildren = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nodes.filter(node => node.parentId === nodeId) | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-12 21:24:43 +08:00
										 |  |  |   const isFromStartNode = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const currentNode = nodes.find(node => node.id === nodeId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!currentNode) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (currentNode.data.type === BlockEnum.Start) | 
					
						
							|  |  |  |       return true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const checkPreviousNodes = (node: Node) => { | 
					
						
							|  |  |  |       const previousNodes = getBeforeNodeById(node.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (const prevNode of previousNodes) { | 
					
						
							|  |  |  |         if (prevNode.data.type === BlockEnum.Start) | 
					
						
							|  |  |  |           return true | 
					
						
							|  |  |  |         if (checkPreviousNodes(prevNode)) | 
					
						
							|  |  |  |           return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return checkPreviousNodes(currentNode) | 
					
						
							|  |  |  |   }, [store, getBeforeNodeById]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const handleOutVarRenameChange = useCallback((nodeId: string, oldValeSelector: ValueSelector, newVarSelector: ValueSelector) => { | 
					
						
							|  |  |  |     const { getNodes, setNodes } = store.getState() | 
					
						
							|  |  |  |     const afterNodes = getAfterNodesInSameBranch(nodeId) | 
					
						
							|  |  |  |     const effectNodes = findUsedVarNodes(oldValeSelector, afterNodes) | 
					
						
							|  |  |  |     if (effectNodes.length > 0) { | 
					
						
							|  |  |  |       const newNodes = getNodes().map((node) => { | 
					
						
							|  |  |  |         if (effectNodes.find(n => n.id === node.id)) | 
					
						
							|  |  |  |           return updateNodeVars(node, oldValeSelector, newVarSelector) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return node | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       setNodes(newNodes) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // eslint-disable-next-line react-hooks/exhaustive-deps
 | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const isVarUsedInNodes = useCallback((varSelector: ValueSelector) => { | 
					
						
							|  |  |  |     const nodeId = varSelector[0] | 
					
						
							|  |  |  |     const afterNodes = getAfterNodesInSameBranch(nodeId) | 
					
						
							|  |  |  |     const effectNodes = findUsedVarNodes(varSelector, afterNodes) | 
					
						
							|  |  |  |     return effectNodes.length > 0 | 
					
						
							|  |  |  |   }, [getAfterNodesInSameBranch]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const removeUsedVarInNodes = useCallback((varSelector: ValueSelector) => { | 
					
						
							|  |  |  |     const nodeId = varSelector[0] | 
					
						
							|  |  |  |     const { getNodes, setNodes } = store.getState() | 
					
						
							|  |  |  |     const afterNodes = getAfterNodesInSameBranch(nodeId) | 
					
						
							|  |  |  |     const effectNodes = findUsedVarNodes(varSelector, afterNodes) | 
					
						
							|  |  |  |     if (effectNodes.length > 0) { | 
					
						
							|  |  |  |       const newNodes = getNodes().map((node) => { | 
					
						
							|  |  |  |         if (effectNodes.find(n => n.id === node.id)) | 
					
						
							|  |  |  |           return updateNodeVars(node, varSelector, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return node | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       setNodes(newNodes) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [getAfterNodesInSameBranch, store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const isNodeVarsUsedInNodes = useCallback((node: Node, isChatMode: boolean) => { | 
					
						
							|  |  |  |     const outputVars = getNodeOutputVars(node, isChatMode) | 
					
						
							|  |  |  |     const isUsed = outputVars.some((varSelector) => { | 
					
						
							|  |  |  |       return isVarUsedInNodes(varSelector) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     return isUsed | 
					
						
							|  |  |  |   }, [isVarUsedInNodes]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-11 11:30:33 +08:00
										 |  |  |   const checkParallelLimit = useCallback((nodeId: string, nodeHandle = 'source') => { | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |     const { | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							| 
									
										
										
										
											2024-09-11 11:30:33 +08:00
										 |  |  |     const connectedEdges = edges.filter(edge => edge.source === nodeId && edge.sourceHandle === nodeHandle) | 
					
						
							|  |  |  |     if (connectedEdges.length > PARALLEL_LIMIT - 1) { | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |       const { setShowTips } = workflowStore.getState() | 
					
						
							|  |  |  |       setShowTips(t('workflow.common.parallelTip.limit', { num: PARALLEL_LIMIT })) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true | 
					
						
							|  |  |  |   }, [store, workflowStore, t]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const checkNestedParallelLimit = useCallback((nodes: Node[], edges: Edge[], parentNodeId?: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       parallelList, | 
					
						
							|  |  |  |       hasAbnormalEdges, | 
					
						
							|  |  |  |     } = getParallelInfo(nodes, edges, parentNodeId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (hasAbnormalEdges) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (let i = 0; i < parallelList.length; i++) { | 
					
						
							|  |  |  |       const parallel = parallelList[i] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (parallel.depth > PARALLEL_DEPTH_LIMIT) { | 
					
						
							|  |  |  |         const { setShowTips } = workflowStore.getState() | 
					
						
							|  |  |  |         setShowTips(t('workflow.common.parallelTip.depthLimit', { num: PARALLEL_DEPTH_LIMIT })) | 
					
						
							|  |  |  |         return false | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true | 
					
						
							|  |  |  |   }, [t, workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-11 11:30:33 +08:00
										 |  |  |   const isValidConnection = useCallback(({ source, sourceHandle, target }: Connection) => { | 
					
						
							| 
									
										
										
										
											2024-04-09 12:24:41 +08:00
										 |  |  |     const { | 
					
						
							|  |  |  |       edges, | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const sourceNode: Node = nodes.find(node => node.id === source)! | 
					
						
							|  |  |  |     const targetNode: Node = nodes.find(node => node.id === target)! | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-11 11:30:33 +08:00
										 |  |  |     if (!checkParallelLimit(source!, sourceHandle || 'source')) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-14 17:08:11 +08:00
										 |  |  |     if (sourceNode.type === CUSTOM_NOTE_NODE || targetNode.type === CUSTOM_NOTE_NODE) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |     if (sourceNode.parentId !== targetNode.parentId) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     if (sourceNode && targetNode) { | 
					
						
							|  |  |  |       const sourceNodeAvailableNextNodes = nodesExtraData[sourceNode.data.type].availableNextNodes | 
					
						
							|  |  |  |       const targetNodeAvailablePrevNodes = [...nodesExtraData[targetNode.data.type].availablePrevNodes, BlockEnum.Start] | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       if (!sourceNodeAvailableNextNodes.includes(targetNode.data.type)) | 
					
						
							|  |  |  |         return false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!targetNodeAvailablePrevNodes.includes(sourceNode.data.type)) | 
					
						
							|  |  |  |         return false | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-09 12:24:41 +08:00
										 |  |  |     const hasCycle = (node: Node, visited = new Set()) => { | 
					
						
							|  |  |  |       if (visited.has(node.id)) | 
					
						
							|  |  |  |         return false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       visited.add(node.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (const outgoer of getOutgoers(node, nodes, edges)) { | 
					
						
							|  |  |  |         if (outgoer.id === source) | 
					
						
							|  |  |  |           return true | 
					
						
							|  |  |  |         if (hasCycle(outgoer, visited)) | 
					
						
							|  |  |  |           return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return !hasCycle(targetNode) | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |   }, [store, nodesExtraData, checkParallelLimit]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const formatTimeFromNow = useCallback((time: number) => { | 
					
						
							|  |  |  |     return dayjs(time).locale(locale === 'zh-Hans' ? 'zh-cn' : locale).fromNow() | 
					
						
							|  |  |  |   }, [locale]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const getNode = useCallback((nodeId?: string) => { | 
					
						
							|  |  |  |     const { getNodes } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return nodes.find(node => node.id === nodeId) || nodes.find(node => node.data.type === BlockEnum.Start) | 
					
						
							|  |  |  |   }, [store]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							| 
									
										
										
										
											2024-04-26 14:23:13 +08:00
										 |  |  |     setPanelWidth, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     getTreeLeafNodes, | 
					
						
							|  |  |  |     getBeforeNodesInSameBranch, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     getBeforeNodesInSameBranchIncludeParent, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     getAfterNodesInSameBranch, | 
					
						
							|  |  |  |     handleOutVarRenameChange, | 
					
						
							|  |  |  |     isVarUsedInNodes, | 
					
						
							|  |  |  |     removeUsedVarInNodes, | 
					
						
							|  |  |  |     isNodeVarsUsedInNodes, | 
					
						
							| 
									
										
										
										
											2024-09-10 15:23:16 +08:00
										 |  |  |     checkParallelLimit, | 
					
						
							|  |  |  |     checkNestedParallelLimit, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     isValidConnection, | 
					
						
							| 
									
										
										
										
											2024-10-12 21:24:43 +08:00
										 |  |  |     isFromStartNode, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     formatTimeFromNow, | 
					
						
							|  |  |  |     getNode, | 
					
						
							|  |  |  |     getBeforeNodeById, | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     getIterationNodeChildren, | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useFetchToolsData = () => { | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleFetchAllTools = useCallback(async (type: string) => { | 
					
						
							|  |  |  |     if (type === 'builtin') { | 
					
						
							|  |  |  |       const buildInTools = await fetchAllBuiltInTools() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       workflowStore.setState({ | 
					
						
							|  |  |  |         buildInTools: buildInTools || [], | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (type === 'custom') { | 
					
						
							|  |  |  |       const customTools = await fetchAllCustomTools() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       workflowStore.setState({ | 
					
						
							|  |  |  |         customTools: customTools || [], | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     if (type === 'workflow') { | 
					
						
							|  |  |  |       const workflowTools = await fetchAllWorkflowTools() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       workflowStore.setState({ | 
					
						
							|  |  |  |         workflowTools: workflowTools || [], | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   }, [workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     handleFetchAllTools, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useWorkflowInit = () => { | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  |   const { | 
					
						
							|  |  |  |     nodes: nodesTemplate, | 
					
						
							|  |  |  |     edges: edgesTemplate, | 
					
						
							|  |  |  |   } = useWorkflowTemplate() | 
					
						
							|  |  |  |   const { handleFetchAllTools } = useFetchToolsData() | 
					
						
							|  |  |  |   const appDetail = useAppStore(state => state.appDetail)! | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   const setSyncWorkflowDraftHash = useStore(s => s.setSyncWorkflowDraftHash) | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  |   const [data, setData] = useState<FetchWorkflowDraftResponse>() | 
					
						
							|  |  |  |   const [isLoading, setIsLoading] = useState(true) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   workflowStore.setState({ appId: appDetail.id }) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  |   const handleGetInitialWorkflowData = useCallback(async () => { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const res = await fetchWorkflowDraft(`/apps/${appDetail.id}/workflows/draft`) | 
					
						
							|  |  |  |       setData(res) | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |       workflowStore.setState({ | 
					
						
							|  |  |  |         envSecrets: (res.environment_variables || []).filter(env => env.value_type === 'secret').reduce((acc, env) => { | 
					
						
							|  |  |  |           acc[env.id] = env.value | 
					
						
							|  |  |  |           return acc | 
					
						
							|  |  |  |         }, {} as Record<string, string>), | 
					
						
							|  |  |  |         environmentVariables: res.environment_variables?.map(env => env.value_type === 'secret' ? { ...env, value: '[__HIDDEN__]' } : env) || [], | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |         // #TODO chatVar sync#
 | 
					
						
							|  |  |  |         conversationVariables: res.conversation_variables || [], | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |       }) | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |       setSyncWorkflowDraftHash(res.hash) | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  |       setIsLoading(false) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     catch (error: any) { | 
					
						
							|  |  |  |       if (error && error.json && !error.bodyUsed && appDetail) { | 
					
						
							|  |  |  |         error.json().then((err: any) => { | 
					
						
							|  |  |  |           if (err.code === 'draft_workflow_not_exist') { | 
					
						
							|  |  |  |             workflowStore.setState({ notInitialWorkflow: true }) | 
					
						
							|  |  |  |             syncWorkflowDraft({ | 
					
						
							|  |  |  |               url: `/apps/${appDetail.id}/workflows/draft`, | 
					
						
							|  |  |  |               params: { | 
					
						
							|  |  |  |                 graph: { | 
					
						
							|  |  |  |                   nodes: nodesTemplate, | 
					
						
							|  |  |  |                   edges: edgesTemplate, | 
					
						
							|  |  |  |                 }, | 
					
						
							| 
									
										
										
										
											2024-06-24 13:03:49 +08:00
										 |  |  |                 features: { | 
					
						
							|  |  |  |                   retriever_resource: { enabled: true }, | 
					
						
							|  |  |  |                 }, | 
					
						
							| 
									
										
										
										
											2024-07-22 15:29:39 +08:00
										 |  |  |                 environment_variables: [], | 
					
						
							| 
									
										
										
										
											2024-08-13 14:44:10 +08:00
										 |  |  |                 conversation_variables: [], | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  |               }, | 
					
						
							|  |  |  |             }).then((res) => { | 
					
						
							|  |  |  |               workflowStore.getState().setDraftUpdatedAt(res.updated_at) | 
					
						
							|  |  |  |               handleGetInitialWorkflowData() | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-10 14:48:20 +08:00
										 |  |  |   }, [appDetail, nodesTemplate, edgesTemplate, workflowStore, setSyncWorkflowDraftHash]) | 
					
						
							| 
									
										
										
										
											2024-04-25 14:02:06 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     handleGetInitialWorkflowData() | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const handleFetchPreloadData = useCallback(async () => { | 
					
						
							|  |  |  |     try { | 
					
						
							|  |  |  |       const nodesDefaultConfigsData = await fetchNodesDefaultConfigs(`/apps/${appDetail?.id}/workflows/default-workflow-block-configs`) | 
					
						
							|  |  |  |       const publishedWorkflow = await fetchPublishedWorkflow(`/apps/${appDetail?.id}/workflows/publish`) | 
					
						
							|  |  |  |       workflowStore.setState({ | 
					
						
							|  |  |  |         nodesDefaultConfigs: nodesDefaultConfigsData.reduce((acc, block) => { | 
					
						
							|  |  |  |           if (!acc[block.type]) | 
					
						
							|  |  |  |             acc[block.type] = { ...block.config } | 
					
						
							|  |  |  |           return acc | 
					
						
							|  |  |  |         }, {} as Record<string, any>), | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       workflowStore.getState().setPublishedAt(publishedWorkflow?.created_at) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     catch (e) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [workflowStore, appDetail]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     handleFetchPreloadData() | 
					
						
							|  |  |  |     handleFetchAllTools('builtin') | 
					
						
							|  |  |  |     handleFetchAllTools('custom') | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     handleFetchAllTools('workflow') | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   }, [handleFetchPreloadData, handleFetchAllTools]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |     if (data) { | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |       workflowStore.getState().setDraftUpdatedAt(data.updated_at) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |       workflowStore.getState().setToolPublished(data.tool_published) | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   }, [data, workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     data, | 
					
						
							|  |  |  |     isLoading, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useWorkflowReadOnly = () => { | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  |   const workflowRunningData = useStore(s => s.workflowRunningData) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const getWorkflowReadOnly = useCallback(() => { | 
					
						
							|  |  |  |     return workflowStore.getState().workflowRunningData?.result.status === WorkflowRunningStatus.Running | 
					
						
							|  |  |  |   }, [workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     workflowReadOnly: workflowRunningData?.result.status === WorkflowRunningStatus.Running, | 
					
						
							|  |  |  |     getWorkflowReadOnly, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | export const useNodesReadOnly = () => { | 
					
						
							|  |  |  |   const workflowStore = useWorkflowStore() | 
					
						
							|  |  |  |   const workflowRunningData = useStore(s => s.workflowRunningData) | 
					
						
							|  |  |  |   const historyWorkflowData = useStore(s => s.historyWorkflowData) | 
					
						
							|  |  |  |   const isRestoring = useStore(s => s.isRestoring) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const getNodesReadOnly = useCallback(() => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       workflowRunningData, | 
					
						
							|  |  |  |       historyWorkflowData, | 
					
						
							|  |  |  |       isRestoring, | 
					
						
							|  |  |  |     } = workflowStore.getState() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-04-28 17:09:56 +08:00
										 |  |  |     return workflowRunningData?.result.status === WorkflowRunningStatus.Running || historyWorkflowData || isRestoring | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   }, [workflowStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							| 
									
										
										
										
											2024-04-28 17:09:56 +08:00
										 |  |  |     nodesReadOnly: !!(workflowRunningData?.result.status === WorkflowRunningStatus.Running || historyWorkflowData || isRestoring), | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     getNodesReadOnly, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useToolIcon = (data: Node['data']) => { | 
					
						
							|  |  |  |   const buildInTools = useStore(s => s.buildInTools) | 
					
						
							|  |  |  |   const customTools = useStore(s => s.customTools) | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   const workflowTools = useStore(s => s.workflowTools) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |   const toolIcon = useMemo(() => { | 
					
						
							|  |  |  |     if (data.type === BlockEnum.Tool) { | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |       let targetTools = buildInTools | 
					
						
							|  |  |  |       if (data.provider_type === CollectionType.builtIn) | 
					
						
							|  |  |  |         targetTools = buildInTools | 
					
						
							|  |  |  |       else if (data.provider_type === CollectionType.custom) | 
					
						
							|  |  |  |         targetTools = customTools | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         targetTools = workflowTools | 
					
						
							|  |  |  |       return targetTools.find(toolWithProvider => toolWithProvider.id === data.provider_id)?.icon | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  |   }, [data, buildInTools, customTools, workflowTools]) | 
					
						
							| 
									
										
										
										
											2024-04-08 18:51:46 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return toolIcon | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-05-27 21:57:08 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | export const useIsNodeInIteration = (iterationId: string) => { | 
					
						
							|  |  |  |   const store = useStoreApi() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const isNodeInIteration = useCallback((nodeId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       getNodes, | 
					
						
							|  |  |  |     } = store.getState() | 
					
						
							|  |  |  |     const nodes = getNodes() | 
					
						
							|  |  |  |     const node = nodes.find(node => node.id === nodeId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!node) | 
					
						
							|  |  |  |       return false | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (node.parentId === iterationId) | 
					
						
							|  |  |  |       return true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false | 
					
						
							|  |  |  |   }, [iterationId, store]) | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     isNodeInIteration, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |