mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-31 19:03:09 +00:00 
			
		
		
		
	 dabfd74622
			
		
	
	
		dabfd74622
		
			
		
	
	
	
	
		
			
			Co-authored-by: StyleZhang <jasonapring2015@outlook.com> Co-authored-by: Yi <yxiaoisme@gmail.com> Co-authored-by: -LAN- <laipz8200@outlook.com>
		
			
				
	
	
		
			759 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			759 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {
 | |
|   Position,
 | |
|   getConnectedEdges,
 | |
|   getIncomers,
 | |
|   getOutgoers,
 | |
| } from 'reactflow'
 | |
| import dagre from '@dagrejs/dagre'
 | |
| import { v4 as uuid4 } from 'uuid'
 | |
| import {
 | |
|   cloneDeep,
 | |
|   groupBy,
 | |
|   isEqual,
 | |
|   uniqBy,
 | |
| } from 'lodash-es'
 | |
| import type {
 | |
|   Edge,
 | |
|   InputVar,
 | |
|   Node,
 | |
|   ToolWithProvider,
 | |
|   ValueSelector,
 | |
| } from './types'
 | |
| import { BlockEnum } from './types'
 | |
| import {
 | |
|   CUSTOM_NODE,
 | |
|   ITERATION_CHILDREN_Z_INDEX,
 | |
|   ITERATION_NODE_Z_INDEX,
 | |
|   NODE_WIDTH_X_OFFSET,
 | |
|   START_INITIAL_POSITION,
 | |
| } from './constants'
 | |
| import { CUSTOM_ITERATION_START_NODE } from './nodes/iteration-start/constants'
 | |
| import type { QuestionClassifierNodeType } from './nodes/question-classifier/types'
 | |
| import type { IfElseNodeType } from './nodes/if-else/types'
 | |
| import { branchNameCorrect } from './nodes/if-else/utils'
 | |
| import type { ToolNodeType } from './nodes/tool/types'
 | |
| import type { IterationNodeType } from './nodes/iteration/types'
 | |
| import { CollectionType } from '@/app/components/tools/types'
 | |
| import { toolParametersToFormSchemas } from '@/app/components/tools/utils/to-form-schema'
 | |
| 
 | |
| const WHITE = 'WHITE'
 | |
| const GRAY = 'GRAY'
 | |
| const BLACK = 'BLACK'
 | |
| 
 | |
| const isCyclicUtil = (nodeId: string, color: Record<string, string>, adjList: Record<string, string[]>, stack: string[]) => {
 | |
|   color[nodeId] = GRAY
 | |
|   stack.push(nodeId)
 | |
| 
 | |
|   for (let i = 0; i < adjList[nodeId].length; ++i) {
 | |
|     const childId = adjList[nodeId][i]
 | |
| 
 | |
|     if (color[childId] === GRAY) {
 | |
|       stack.push(childId)
 | |
|       return true
 | |
|     }
 | |
|     if (color[childId] === WHITE && isCyclicUtil(childId, color, adjList, stack))
 | |
|       return true
 | |
|   }
 | |
|   color[nodeId] = BLACK
 | |
|   if (stack.length > 0 && stack[stack.length - 1] === nodeId)
 | |
|     stack.pop()
 | |
|   return false
 | |
| }
 | |
| 
 | |
| const getCycleEdges = (nodes: Node[], edges: Edge[]) => {
 | |
|   const adjList: Record<string, string[]> = {}
 | |
|   const color: Record<string, string> = {}
 | |
|   const stack: string[] = []
 | |
| 
 | |
|   for (const node of nodes) {
 | |
|     color[node.id] = WHITE
 | |
|     adjList[node.id] = []
 | |
|   }
 | |
| 
 | |
|   for (const edge of edges)
 | |
|     adjList[edge.source]?.push(edge.target)
 | |
| 
 | |
|   for (let i = 0; i < nodes.length; i++) {
 | |
|     if (color[nodes[i].id] === WHITE)
 | |
|       isCyclicUtil(nodes[i].id, color, adjList, stack)
 | |
|   }
 | |
| 
 | |
|   const cycleEdges = []
 | |
|   if (stack.length > 0) {
 | |
|     const cycleNodes = new Set(stack)
 | |
|     for (const edge of edges) {
 | |
|       if (cycleNodes.has(edge.source) && cycleNodes.has(edge.target))
 | |
|         cycleEdges.push(edge)
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return cycleEdges
 | |
| }
 | |
| 
 | |
| export function getIterationStartNode(iterationId: string): Node {
 | |
|   return generateNewNode({
 | |
|     id: `${iterationId}start`,
 | |
|     type: CUSTOM_ITERATION_START_NODE,
 | |
|     data: {
 | |
|       title: '',
 | |
|       desc: '',
 | |
|       type: BlockEnum.IterationStart,
 | |
|       isInIteration: true,
 | |
|     },
 | |
|     position: {
 | |
|       x: 24,
 | |
|       y: 68,
 | |
|     },
 | |
|     zIndex: ITERATION_CHILDREN_Z_INDEX,
 | |
|     parentId: iterationId,
 | |
|     selectable: false,
 | |
|     draggable: false,
 | |
|   }).newNode
 | |
| }
 | |
| 
 | |
| export function generateNewNode({ data, position, id, zIndex, type, ...rest }: Omit<Node, 'id'> & { id?: string }): {
 | |
|   newNode: Node
 | |
|   newIterationStartNode?: Node
 | |
| } {
 | |
|   const newNode = {
 | |
|     id: id || `${Date.now()}`,
 | |
|     type: type || CUSTOM_NODE,
 | |
|     data,
 | |
|     position,
 | |
|     targetPosition: Position.Left,
 | |
|     sourcePosition: Position.Right,
 | |
|     zIndex: data.type === BlockEnum.Iteration ? ITERATION_NODE_Z_INDEX : zIndex,
 | |
|     ...rest,
 | |
|   } as Node
 | |
| 
 | |
|   if (data.type === BlockEnum.Iteration) {
 | |
|     const newIterationStartNode = getIterationStartNode(newNode.id);
 | |
|     (newNode.data as IterationNodeType).start_node_id = newIterationStartNode.id;
 | |
|     (newNode.data as IterationNodeType)._children = [newIterationStartNode.id]
 | |
|     return {
 | |
|       newNode,
 | |
|       newIterationStartNode,
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     newNode,
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const preprocessNodesAndEdges = (nodes: Node[], edges: Edge[]) => {
 | |
|   const hasIterationNode = nodes.some(node => node.data.type === BlockEnum.Iteration)
 | |
| 
 | |
|   if (!hasIterationNode) {
 | |
|     return {
 | |
|       nodes,
 | |
|       edges,
 | |
|     }
 | |
|   }
 | |
|   const nodesMap = nodes.reduce((prev, next) => {
 | |
|     prev[next.id] = next
 | |
|     return prev
 | |
|   }, {} as Record<string, Node>)
 | |
|   const iterationNodesWithStartNode = []
 | |
|   const iterationNodesWithoutStartNode = []
 | |
| 
 | |
|   for (let i = 0; i < nodes.length; i++) {
 | |
|     const currentNode = nodes[i] as Node<IterationNodeType>
 | |
| 
 | |
|     if (currentNode.data.type === BlockEnum.Iteration) {
 | |
|       if (currentNode.data.start_node_id) {
 | |
|         if (nodesMap[currentNode.data.start_node_id]?.type !== CUSTOM_ITERATION_START_NODE)
 | |
|           iterationNodesWithStartNode.push(currentNode)
 | |
|       }
 | |
|       else {
 | |
|         iterationNodesWithoutStartNode.push(currentNode)
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   const newIterationStartNodesMap = {} as Record<string, Node>
 | |
|   const newIterationStartNodes = [...iterationNodesWithStartNode, ...iterationNodesWithoutStartNode].map((iterationNode, index) => {
 | |
|     const newNode = getIterationStartNode(iterationNode.id)
 | |
|     newNode.id = newNode.id + index
 | |
|     newIterationStartNodesMap[iterationNode.id] = newNode
 | |
|     return newNode
 | |
|   })
 | |
|   const newEdges = iterationNodesWithStartNode.map((iterationNode) => {
 | |
|     const newNode = newIterationStartNodesMap[iterationNode.id]
 | |
|     const startNode = nodesMap[iterationNode.data.start_node_id]
 | |
|     const source = newNode.id
 | |
|     const sourceHandle = 'source'
 | |
|     const target = startNode.id
 | |
|     const targetHandle = 'target'
 | |
|     return {
 | |
|       id: `${source}-${sourceHandle}-${target}-${targetHandle}`,
 | |
|       type: 'custom',
 | |
|       source,
 | |
|       sourceHandle,
 | |
|       target,
 | |
|       targetHandle,
 | |
|       data: {
 | |
|         sourceType: newNode.data.type,
 | |
|         targetType: startNode.data.type,
 | |
|         isInIteration: true,
 | |
|         iteration_id: startNode.parentId,
 | |
|         _connectedNodeIsSelected: true,
 | |
|       },
 | |
|       zIndex: ITERATION_CHILDREN_Z_INDEX,
 | |
|     }
 | |
|   })
 | |
|   nodes.forEach((node) => {
 | |
|     if (node.data.type === BlockEnum.Iteration && newIterationStartNodesMap[node.id])
 | |
|       (node.data as IterationNodeType).start_node_id = newIterationStartNodesMap[node.id].id
 | |
|   })
 | |
| 
 | |
|   return {
 | |
|     nodes: [...nodes, ...newIterationStartNodes],
 | |
|     edges: [...edges, ...newEdges],
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const initialNodes = (originNodes: Node[], originEdges: Edge[]) => {
 | |
|   const { nodes, edges } = preprocessNodesAndEdges(cloneDeep(originNodes), cloneDeep(originEdges))
 | |
|   const firstNode = nodes[0]
 | |
| 
 | |
|   if (!firstNode?.position) {
 | |
|     nodes.forEach((node, index) => {
 | |
|       node.position = {
 | |
|         x: START_INITIAL_POSITION.x + index * NODE_WIDTH_X_OFFSET,
 | |
|         y: START_INITIAL_POSITION.y,
 | |
|       }
 | |
|     })
 | |
|   }
 | |
| 
 | |
|   const iterationNodeMap = nodes.reduce((acc, node) => {
 | |
|     if (node.parentId) {
 | |
|       if (acc[node.parentId])
 | |
|         acc[node.parentId].push(node.id)
 | |
|       else
 | |
|         acc[node.parentId] = [node.id]
 | |
|     }
 | |
|     return acc
 | |
|   }, {} as Record<string, string[]>)
 | |
| 
 | |
|   return nodes.map((node) => {
 | |
|     if (!node.type)
 | |
|       node.type = CUSTOM_NODE
 | |
| 
 | |
|     const connectedEdges = getConnectedEdges([node], edges)
 | |
|     node.data._connectedSourceHandleIds = connectedEdges.filter(edge => edge.source === node.id).map(edge => edge.sourceHandle || 'source')
 | |
|     node.data._connectedTargetHandleIds = connectedEdges.filter(edge => edge.target === node.id).map(edge => edge.targetHandle || 'target')
 | |
| 
 | |
|     if (node.data.type === BlockEnum.IfElse) {
 | |
|       const nodeData = node.data as IfElseNodeType
 | |
| 
 | |
|       if (!nodeData.cases && nodeData.logical_operator && nodeData.conditions) {
 | |
|         (node.data as IfElseNodeType).cases = [
 | |
|           {
 | |
|             case_id: 'true',
 | |
|             logical_operator: nodeData.logical_operator,
 | |
|             conditions: nodeData.conditions,
 | |
|           },
 | |
|         ]
 | |
|       }
 | |
|       node.data._targetBranches = branchNameCorrect([
 | |
|         ...(node.data as IfElseNodeType).cases.map(item => ({ id: item.case_id, name: '' })),
 | |
|         { id: 'false', name: '' },
 | |
|       ])
 | |
|     }
 | |
| 
 | |
|     if (node.data.type === BlockEnum.QuestionClassifier) {
 | |
|       node.data._targetBranches = (node.data as QuestionClassifierNodeType).classes.map((topic) => {
 | |
|         return topic
 | |
|       })
 | |
|     }
 | |
| 
 | |
|     if (node.data.type === BlockEnum.Iteration)
 | |
|       node.data._children = iterationNodeMap[node.id] || []
 | |
| 
 | |
|     return node
 | |
|   })
 | |
| }
 | |
| 
 | |
| export const initialEdges = (originEdges: Edge[], originNodes: Node[]) => {
 | |
|   const { nodes, edges } = preprocessNodesAndEdges(cloneDeep(originNodes), cloneDeep(originEdges))
 | |
|   let selectedNode: Node | null = null
 | |
|   const nodesMap = nodes.reduce((acc, node) => {
 | |
|     acc[node.id] = node
 | |
| 
 | |
|     if (node.data?.selected)
 | |
|       selectedNode = node
 | |
| 
 | |
|     return acc
 | |
|   }, {} as Record<string, Node>)
 | |
| 
 | |
|   const cycleEdges = getCycleEdges(nodes, edges)
 | |
|   return edges.filter((edge) => {
 | |
|     return !cycleEdges.find(cycEdge => cycEdge.source === edge.source && cycEdge.target === edge.target)
 | |
|   }).map((edge) => {
 | |
|     edge.type = 'custom'
 | |
| 
 | |
|     if (!edge.sourceHandle)
 | |
|       edge.sourceHandle = 'source'
 | |
| 
 | |
|     if (!edge.targetHandle)
 | |
|       edge.targetHandle = 'target'
 | |
| 
 | |
|     if (!edge.data?.sourceType && edge.source && nodesMap[edge.source]) {
 | |
|       edge.data = {
 | |
|         ...edge.data,
 | |
|         sourceType: nodesMap[edge.source].data.type!,
 | |
|       } as any
 | |
|     }
 | |
| 
 | |
|     if (!edge.data?.targetType && edge.target && nodesMap[edge.target]) {
 | |
|       edge.data = {
 | |
|         ...edge.data,
 | |
|         targetType: nodesMap[edge.target].data.type!,
 | |
|       } as any
 | |
|     }
 | |
| 
 | |
|     if (selectedNode) {
 | |
|       edge.data = {
 | |
|         ...edge.data,
 | |
|         _connectedNodeIsSelected: edge.source === selectedNode.id || edge.target === selectedNode.id,
 | |
|       } as any
 | |
|     }
 | |
| 
 | |
|     return edge
 | |
|   })
 | |
| }
 | |
| 
 | |
| export const getLayoutByDagre = (originNodes: Node[], originEdges: Edge[]) => {
 | |
|   const dagreGraph = new dagre.graphlib.Graph()
 | |
|   dagreGraph.setDefaultEdgeLabel(() => ({}))
 | |
|   const nodes = cloneDeep(originNodes).filter(node => !node.parentId && node.type === CUSTOM_NODE)
 | |
|   const edges = cloneDeep(originEdges).filter(edge => !edge.data?.isInIteration)
 | |
|   dagreGraph.setGraph({
 | |
|     rankdir: 'LR',
 | |
|     align: 'UL',
 | |
|     nodesep: 40,
 | |
|     ranksep: 60,
 | |
|     ranker: 'tight-tree',
 | |
|     marginx: 30,
 | |
|     marginy: 200,
 | |
|   })
 | |
|   nodes.forEach((node) => {
 | |
|     dagreGraph.setNode(node.id, {
 | |
|       width: node.width!,
 | |
|       height: node.height!,
 | |
|     })
 | |
|   })
 | |
| 
 | |
|   edges.forEach((edge) => {
 | |
|     dagreGraph.setEdge(edge.source, edge.target)
 | |
|   })
 | |
| 
 | |
|   dagre.layout(dagreGraph)
 | |
| 
 | |
|   return dagreGraph
 | |
| }
 | |
| 
 | |
| export const canRunBySingle = (nodeType: BlockEnum) => {
 | |
|   return nodeType === BlockEnum.LLM
 | |
|     || nodeType === BlockEnum.KnowledgeRetrieval
 | |
|     || nodeType === BlockEnum.Code
 | |
|     || nodeType === BlockEnum.TemplateTransform
 | |
|     || nodeType === BlockEnum.QuestionClassifier
 | |
|     || nodeType === BlockEnum.HttpRequest
 | |
|     || nodeType === BlockEnum.Tool
 | |
|     || nodeType === BlockEnum.ParameterExtractor
 | |
|     || nodeType === BlockEnum.Iteration
 | |
| }
 | |
| 
 | |
| type ConnectedSourceOrTargetNodesChange = {
 | |
|   type: string
 | |
|   edge: Edge
 | |
| }[]
 | |
| export const getNodesConnectedSourceOrTargetHandleIdsMap = (changes: ConnectedSourceOrTargetNodesChange, nodes: Node[]) => {
 | |
|   const nodesConnectedSourceOrTargetHandleIdsMap = {} as Record<string, any>
 | |
| 
 | |
|   changes.forEach((change) => {
 | |
|     const {
 | |
|       edge,
 | |
|       type,
 | |
|     } = change
 | |
|     const sourceNode = nodes.find(node => node.id === edge.source)!
 | |
|     if (sourceNode) {
 | |
|       nodesConnectedSourceOrTargetHandleIdsMap[sourceNode.id] = nodesConnectedSourceOrTargetHandleIdsMap[sourceNode.id] || {
 | |
|         _connectedSourceHandleIds: [...(sourceNode?.data._connectedSourceHandleIds || [])],
 | |
|         _connectedTargetHandleIds: [...(sourceNode?.data._connectedTargetHandleIds || [])],
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     const targetNode = nodes.find(node => node.id === edge.target)!
 | |
|     if (targetNode) {
 | |
|       nodesConnectedSourceOrTargetHandleIdsMap[targetNode.id] = nodesConnectedSourceOrTargetHandleIdsMap[targetNode.id] || {
 | |
|         _connectedSourceHandleIds: [...(targetNode?.data._connectedSourceHandleIds || [])],
 | |
|         _connectedTargetHandleIds: [...(targetNode?.data._connectedTargetHandleIds || [])],
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (sourceNode) {
 | |
|       if (type === 'remove') {
 | |
|         const index = nodesConnectedSourceOrTargetHandleIdsMap[sourceNode.id]._connectedSourceHandleIds.findIndex((handleId: string) => handleId === edge.sourceHandle)
 | |
|         nodesConnectedSourceOrTargetHandleIdsMap[sourceNode.id]._connectedSourceHandleIds.splice(index, 1)
 | |
|       }
 | |
| 
 | |
|       if (type === 'add')
 | |
|         nodesConnectedSourceOrTargetHandleIdsMap[sourceNode.id]._connectedSourceHandleIds.push(edge.sourceHandle || 'source')
 | |
|     }
 | |
| 
 | |
|     if (targetNode) {
 | |
|       if (type === 'remove') {
 | |
|         const index = nodesConnectedSourceOrTargetHandleIdsMap[targetNode.id]._connectedTargetHandleIds.findIndex((handleId: string) => handleId === edge.targetHandle)
 | |
|         nodesConnectedSourceOrTargetHandleIdsMap[targetNode.id]._connectedTargetHandleIds.splice(index, 1)
 | |
|       }
 | |
| 
 | |
|       if (type === 'add')
 | |
|         nodesConnectedSourceOrTargetHandleIdsMap[targetNode.id]._connectedTargetHandleIds.push(edge.targetHandle || 'target')
 | |
|     }
 | |
|   })
 | |
| 
 | |
|   return nodesConnectedSourceOrTargetHandleIdsMap
 | |
| }
 | |
| 
 | |
| export const genNewNodeTitleFromOld = (oldTitle: string) => {
 | |
|   const regex = /^(.+?)\s*\((\d+)\)\s*$/
 | |
|   const match = oldTitle.match(regex)
 | |
| 
 | |
|   if (match) {
 | |
|     const title = match[1]
 | |
|     const num = parseInt(match[2], 10)
 | |
|     return `${title} (${num + 1})`
 | |
|   }
 | |
|   else {
 | |
|     return `${oldTitle} (1)`
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const getValidTreeNodes = (nodes: Node[], edges: Edge[]) => {
 | |
|   const startNode = nodes.find(node => node.data.type === BlockEnum.Start)
 | |
| 
 | |
|   if (!startNode) {
 | |
|     return {
 | |
|       validNodes: [],
 | |
|       maxDepth: 0,
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   const list: Node[] = [startNode]
 | |
|   let maxDepth = 1
 | |
| 
 | |
|   const traverse = (root: Node, depth: number) => {
 | |
|     if (depth > maxDepth)
 | |
|       maxDepth = depth
 | |
| 
 | |
|     const outgoers = getOutgoers(root, nodes, edges)
 | |
| 
 | |
|     if (outgoers.length) {
 | |
|       outgoers.forEach((outgoer) => {
 | |
|         list.push(outgoer)
 | |
|         if (outgoer.data.type === BlockEnum.Iteration)
 | |
|           list.push(...nodes.filter(node => node.parentId === outgoer.id))
 | |
|         traverse(outgoer, depth + 1)
 | |
|       })
 | |
|     }
 | |
|     else {
 | |
|       list.push(root)
 | |
|       if (root.data.type === BlockEnum.Iteration)
 | |
|         list.push(...nodes.filter(node => node.parentId === root.id))
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   traverse(startNode, maxDepth)
 | |
| 
 | |
|   return {
 | |
|     validNodes: uniqBy(list, 'id'),
 | |
|     maxDepth,
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const getToolCheckParams = (
 | |
|   toolData: ToolNodeType,
 | |
|   buildInTools: ToolWithProvider[],
 | |
|   customTools: ToolWithProvider[],
 | |
|   workflowTools: ToolWithProvider[],
 | |
|   language: string,
 | |
| ) => {
 | |
|   const { provider_id, provider_type, tool_name } = toolData
 | |
|   const isBuiltIn = provider_type === CollectionType.builtIn
 | |
|   const currentTools = provider_type === CollectionType.builtIn ? buildInTools : provider_type === CollectionType.custom ? customTools : workflowTools
 | |
|   const currCollection = currentTools.find(item => item.id === provider_id)
 | |
|   const currTool = currCollection?.tools.find(tool => tool.name === tool_name)
 | |
|   const formSchemas = currTool ? toolParametersToFormSchemas(currTool.parameters) : []
 | |
|   const toolInputVarSchema = formSchemas.filter((item: any) => item.form === 'llm')
 | |
|   const toolSettingSchema = formSchemas.filter((item: any) => item.form !== 'llm')
 | |
| 
 | |
|   return {
 | |
|     toolInputsSchema: (() => {
 | |
|       const formInputs: InputVar[] = []
 | |
|       toolInputVarSchema.forEach((item: any) => {
 | |
|         formInputs.push({
 | |
|           label: item.label[language] || item.label.en_US,
 | |
|           variable: item.variable,
 | |
|           type: item.type,
 | |
|           required: item.required,
 | |
|         })
 | |
|       })
 | |
|       return formInputs
 | |
|     })(),
 | |
|     notAuthed: isBuiltIn && !!currCollection?.allow_delete && !currCollection?.is_team_authorization,
 | |
|     toolSettingSchema,
 | |
|     language,
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const changeNodesAndEdgesId = (nodes: Node[], edges: Edge[]) => {
 | |
|   const idMap = nodes.reduce((acc, node) => {
 | |
|     acc[node.id] = uuid4()
 | |
| 
 | |
|     return acc
 | |
|   }, {} as Record<string, string>)
 | |
| 
 | |
|   const newNodes = nodes.map((node) => {
 | |
|     return {
 | |
|       ...node,
 | |
|       id: idMap[node.id],
 | |
|     }
 | |
|   })
 | |
| 
 | |
|   const newEdges = edges.map((edge) => {
 | |
|     return {
 | |
|       ...edge,
 | |
|       source: idMap[edge.source],
 | |
|       target: idMap[edge.target],
 | |
|     }
 | |
|   })
 | |
| 
 | |
|   return [newNodes, newEdges] as [Node[], Edge[]]
 | |
| }
 | |
| 
 | |
| export const isMac = () => {
 | |
|   return navigator.userAgent.toUpperCase().includes('MAC')
 | |
| }
 | |
| 
 | |
| const specialKeysNameMap: Record<string, string | undefined> = {
 | |
|   ctrl: '⌘',
 | |
|   alt: '⌥',
 | |
| }
 | |
| 
 | |
| export const getKeyboardKeyNameBySystem = (key: string) => {
 | |
|   if (isMac())
 | |
|     return specialKeysNameMap[key] || key
 | |
| 
 | |
|   return key
 | |
| }
 | |
| 
 | |
| const specialKeysCodeMap: Record<string, string | undefined> = {
 | |
|   ctrl: 'meta',
 | |
| }
 | |
| 
 | |
| export const getKeyboardKeyCodeBySystem = (key: string) => {
 | |
|   if (isMac())
 | |
|     return specialKeysCodeMap[key] || key
 | |
| 
 | |
|   return key
 | |
| }
 | |
| 
 | |
| export const getTopLeftNodePosition = (nodes: Node[]) => {
 | |
|   let minX = Infinity
 | |
|   let minY = Infinity
 | |
| 
 | |
|   nodes.forEach((node) => {
 | |
|     if (node.position.x < minX)
 | |
|       minX = node.position.x
 | |
| 
 | |
|     if (node.position.y < minY)
 | |
|       minY = node.position.y
 | |
|   })
 | |
| 
 | |
|   return {
 | |
|     x: minX,
 | |
|     y: minY,
 | |
|   }
 | |
| }
 | |
| 
 | |
| export const isEventTargetInputArea = (target: HTMLElement) => {
 | |
|   if (target.tagName === 'INPUT' || target.tagName === 'TEXTAREA')
 | |
|     return true
 | |
| 
 | |
|   if (target.contentEditable === 'true')
 | |
|     return true
 | |
| }
 | |
| 
 | |
| export const variableTransformer = (v: ValueSelector | string) => {
 | |
|   if (typeof v === 'string')
 | |
|     return v.replace(/^{{#|#}}$/g, '').split('.')
 | |
| 
 | |
|   return `{{#${v.join('.')}#}}`
 | |
| }
 | |
| 
 | |
| type ParallelInfoItem = {
 | |
|   parallelNodeId: string
 | |
|   depth: number
 | |
|   isBranch?: boolean
 | |
| }
 | |
| type NodeParallelInfo = {
 | |
|   parallelNodeId: string
 | |
|   edgeHandleId: string
 | |
|   depth: number
 | |
| }
 | |
| type NodeHandle = {
 | |
|   node: Node
 | |
|   handle: string
 | |
| }
 | |
| type NodeStreamInfo = {
 | |
|   upstreamNodes: Set<string>
 | |
|   downstreamEdges: Set<string>
 | |
| }
 | |
| export const getParallelInfo = (nodes: Node[], edges: Edge[], parentNodeId?: string) => {
 | |
|   let startNode
 | |
| 
 | |
|   if (parentNodeId) {
 | |
|     const parentNode = nodes.find(node => node.id === parentNodeId)
 | |
|     if (!parentNode)
 | |
|       throw new Error('Parent node not found')
 | |
| 
 | |
|     startNode = nodes.find(node => node.id === (parentNode.data as IterationNodeType).start_node_id)
 | |
|   }
 | |
|   else {
 | |
|     startNode = nodes.find(node => node.data.type === BlockEnum.Start)
 | |
|   }
 | |
|   if (!startNode)
 | |
|     throw new Error('Start node not found')
 | |
| 
 | |
|   const parallelList = [] as ParallelInfoItem[]
 | |
|   const nextNodeHandles = [{ node: startNode, handle: 'source' }]
 | |
|   let hasAbnormalEdges = false
 | |
| 
 | |
|   const traverse = (firstNodeHandle: NodeHandle) => {
 | |
|     const nodeEdgesSet = {} as Record<string, Set<string>>
 | |
|     const totalEdgesSet = new Set<string>()
 | |
|     const nextHandles = [firstNodeHandle]
 | |
|     const streamInfo = {} as Record<string, NodeStreamInfo>
 | |
|     const parallelListItem = {
 | |
|       parallelNodeId: '',
 | |
|       depth: 0,
 | |
|     } as ParallelInfoItem
 | |
|     const nodeParallelInfoMap = {} as Record<string, NodeParallelInfo>
 | |
|     nodeParallelInfoMap[firstNodeHandle.node.id] = {
 | |
|       parallelNodeId: '',
 | |
|       edgeHandleId: '',
 | |
|       depth: 0,
 | |
|     }
 | |
| 
 | |
|     while (nextHandles.length) {
 | |
|       const currentNodeHandle = nextHandles.shift()!
 | |
|       const { node: currentNode, handle: currentHandle = 'source' } = currentNodeHandle
 | |
|       const currentNodeHandleKey = currentNode.id
 | |
|       const connectedEdges = edges.filter(edge => edge.source === currentNode.id && edge.sourceHandle === currentHandle)
 | |
|       const connectedEdgesLength = connectedEdges.length
 | |
|       const outgoers = nodes.filter(node => connectedEdges.some(edge => edge.target === node.id))
 | |
|       const incomers = getIncomers(currentNode, nodes, edges)
 | |
| 
 | |
|       if (!streamInfo[currentNodeHandleKey]) {
 | |
|         streamInfo[currentNodeHandleKey] = {
 | |
|           upstreamNodes: new Set<string>(),
 | |
|           downstreamEdges: new Set<string>(),
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (nodeEdgesSet[currentNodeHandleKey]?.size > 0 && incomers.length > 1) {
 | |
|         const newSet = new Set<string>()
 | |
|         for (const item of totalEdgesSet) {
 | |
|           if (!streamInfo[currentNodeHandleKey].downstreamEdges.has(item))
 | |
|             newSet.add(item)
 | |
|         }
 | |
|         if (isEqual(nodeEdgesSet[currentNodeHandleKey], newSet)) {
 | |
|           parallelListItem.depth = nodeParallelInfoMap[currentNode.id].depth
 | |
|           nextNodeHandles.push({ node: currentNode, handle: currentHandle })
 | |
|           break
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (nodeParallelInfoMap[currentNode.id].depth > parallelListItem.depth)
 | |
|         parallelListItem.depth = nodeParallelInfoMap[currentNode.id].depth
 | |
| 
 | |
|       outgoers.forEach((outgoer) => {
 | |
|         const outgoerConnectedEdges = getConnectedEdges([outgoer], edges).filter(edge => edge.source === outgoer.id)
 | |
|         const sourceEdgesGroup = groupBy(outgoerConnectedEdges, 'sourceHandle')
 | |
|         const incomers = getIncomers(outgoer, nodes, edges)
 | |
| 
 | |
|         if (outgoers.length > 1 && incomers.length > 1)
 | |
|           hasAbnormalEdges = true
 | |
| 
 | |
|         Object.keys(sourceEdgesGroup).forEach((sourceHandle) => {
 | |
|           nextHandles.push({ node: outgoer, handle: sourceHandle })
 | |
|         })
 | |
|         if (!outgoerConnectedEdges.length)
 | |
|           nextHandles.push({ node: outgoer, handle: 'source' })
 | |
| 
 | |
|         const outgoerKey = outgoer.id
 | |
|         if (!nodeEdgesSet[outgoerKey])
 | |
|           nodeEdgesSet[outgoerKey] = new Set<string>()
 | |
| 
 | |
|         if (nodeEdgesSet[currentNodeHandleKey]) {
 | |
|           for (const item of nodeEdgesSet[currentNodeHandleKey])
 | |
|             nodeEdgesSet[outgoerKey].add(item)
 | |
|         }
 | |
| 
 | |
|         if (!streamInfo[outgoerKey]) {
 | |
|           streamInfo[outgoerKey] = {
 | |
|             upstreamNodes: new Set<string>(),
 | |
|             downstreamEdges: new Set<string>(),
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         if (!nodeParallelInfoMap[outgoer.id]) {
 | |
|           nodeParallelInfoMap[outgoer.id] = {
 | |
|             ...nodeParallelInfoMap[currentNode.id],
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         if (connectedEdgesLength > 1) {
 | |
|           const edge = connectedEdges.find(edge => edge.target === outgoer.id)!
 | |
|           nodeEdgesSet[outgoerKey].add(edge.id)
 | |
|           totalEdgesSet.add(edge.id)
 | |
| 
 | |
|           streamInfo[currentNodeHandleKey].downstreamEdges.add(edge.id)
 | |
|           streamInfo[outgoerKey].upstreamNodes.add(currentNodeHandleKey)
 | |
| 
 | |
|           for (const item of streamInfo[currentNodeHandleKey].upstreamNodes)
 | |
|             streamInfo[item].downstreamEdges.add(edge.id)
 | |
| 
 | |
|           if (!parallelListItem.parallelNodeId)
 | |
|             parallelListItem.parallelNodeId = currentNode.id
 | |
| 
 | |
|           const prevDepth = nodeParallelInfoMap[currentNode.id].depth + 1
 | |
|           const currentDepth = nodeParallelInfoMap[outgoer.id].depth
 | |
| 
 | |
|           nodeParallelInfoMap[outgoer.id].depth = Math.max(prevDepth, currentDepth)
 | |
|         }
 | |
|         else {
 | |
|           for (const item of streamInfo[currentNodeHandleKey].upstreamNodes)
 | |
|             streamInfo[outgoerKey].upstreamNodes.add(item)
 | |
| 
 | |
|           nodeParallelInfoMap[outgoer.id].depth = nodeParallelInfoMap[currentNode.id].depth
 | |
|         }
 | |
|       })
 | |
|     }
 | |
| 
 | |
|     parallelList.push(parallelListItem)
 | |
|   }
 | |
| 
 | |
|   while (nextNodeHandles.length) {
 | |
|     const nodeHandle = nextNodeHandles.shift()!
 | |
|     traverse(nodeHandle)
 | |
|   }
 | |
| 
 | |
|   return {
 | |
|     parallelList,
 | |
|     hasAbnormalEdges,
 | |
|   }
 | |
| }
 |