mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-31 02:42:59 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			380 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			380 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import {
 | |
|   useCallback,
 | |
|   useRef,
 | |
| } from 'react'
 | |
| import produce from 'immer'
 | |
| import { useBoolean } from 'ahooks'
 | |
| import { v4 as uuid4 } from 'uuid'
 | |
| import {
 | |
|   useIsChatMode,
 | |
|   useIsNodeInLoop,
 | |
|   useNodesReadOnly,
 | |
|   useWorkflow,
 | |
| } from '../../hooks'
 | |
| import { ValueType, VarType } from '../../types'
 | |
| import type { ErrorHandleMode, ValueSelector, Var } from '../../types'
 | |
| import useNodeCrud from '../_base/hooks/use-node-crud'
 | |
| import { getNodeInfoById, getNodeUsedVarPassToServerKey, getNodeUsedVars, isSystemVar, toNodeOutputVars } from '../_base/components/variable/utils'
 | |
| import useOneStepRun from '../_base/hooks/use-one-step-run'
 | |
| import { getOperators } from './utils'
 | |
| import { LogicalOperator } from './types'
 | |
| import type { HandleAddCondition, HandleAddSubVariableCondition, HandleRemoveCondition, HandleToggleConditionLogicalOperator, HandleToggleSubVariableConditionLogicalOperator, HandleUpdateCondition, HandleUpdateSubVariableCondition, LoopNodeType } from './types'
 | |
| import useIsVarFileAttribute from './use-is-var-file-attribute'
 | |
| import { useStore } from '@/app/components/workflow/store'
 | |
| 
 | |
| const DELIMITER = '@@@@@'
 | |
| const useConfig = (id: string, payload: LoopNodeType) => {
 | |
|   const { nodesReadOnly: readOnly } = useNodesReadOnly()
 | |
|   const { isNodeInLoop } = useIsNodeInLoop(id)
 | |
|   const isChatMode = useIsChatMode()
 | |
|   const conversationVariables = useStore(s => s.conversationVariables)
 | |
| 
 | |
|   const { inputs, setInputs } = useNodeCrud<LoopNodeType>(id, payload)
 | |
|   const inputsRef = useRef(inputs)
 | |
|   const handleInputsChange = useCallback((newInputs: LoopNodeType) => {
 | |
|     inputsRef.current = newInputs
 | |
|     setInputs(newInputs)
 | |
|   }, [setInputs])
 | |
| 
 | |
|   const filterInputVar = useCallback((varPayload: Var) => {
 | |
|     return [VarType.array, VarType.arrayString, VarType.arrayNumber, VarType.arrayObject, VarType.arrayFile].includes(varPayload.type)
 | |
|   }, [])
 | |
| 
 | |
|   // output
 | |
|   const { getLoopNodeChildren, getBeforeNodesInSameBranch } = useWorkflow()
 | |
|   const beforeNodes = getBeforeNodesInSameBranch(id)
 | |
|   const loopChildrenNodes = [{ id, data: payload } as any, ...getLoopNodeChildren(id)]
 | |
|   const canChooseVarNodes = [...beforeNodes, ...loopChildrenNodes]
 | |
|   const childrenNodeVars = toNodeOutputVars(loopChildrenNodes, isChatMode, undefined, [], conversationVariables)
 | |
| 
 | |
|   // single run
 | |
|   const loopInputKey = `${id}.input_selector`
 | |
|   const {
 | |
|     isShowSingleRun,
 | |
|     showSingleRun,
 | |
|     hideSingleRun,
 | |
|     toVarInputs,
 | |
|     runningStatus,
 | |
|     handleRun: doHandleRun,
 | |
|     handleStop,
 | |
|     runInputData,
 | |
|     setRunInputData,
 | |
|     runResult,
 | |
|     loopRunResult,
 | |
|   } = useOneStepRun<LoopNodeType>({
 | |
|     id,
 | |
|     data: inputs,
 | |
|     loopInputKey,
 | |
|     defaultRunInputData: {
 | |
|       [loopInputKey]: [''],
 | |
|     },
 | |
|   })
 | |
| 
 | |
|   const [isShowLoopDetail, {
 | |
|     setTrue: doShowLoopDetail,
 | |
|     setFalse: doHideLoopDetail,
 | |
|   }] = useBoolean(false)
 | |
| 
 | |
|   const hideLoopDetail = useCallback(() => {
 | |
|     hideSingleRun()
 | |
|     doHideLoopDetail()
 | |
|   }, [doHideLoopDetail, hideSingleRun])
 | |
| 
 | |
|   const showLoopDetail = useCallback(() => {
 | |
|     doShowLoopDetail()
 | |
|   }, [doShowLoopDetail])
 | |
| 
 | |
|   const backToSingleRun = useCallback(() => {
 | |
|     hideLoopDetail()
 | |
|     showSingleRun()
 | |
|   }, [hideLoopDetail, showSingleRun])
 | |
| 
 | |
|   const {
 | |
|     getIsVarFileAttribute,
 | |
|   } = useIsVarFileAttribute({
 | |
|     nodeId: id,
 | |
|   })
 | |
| 
 | |
|   const { usedOutVars, allVarObject } = (() => {
 | |
|     const vars: ValueSelector[] = []
 | |
|     const varObjs: Record<string, boolean> = {}
 | |
|     const allVarObject: Record<string, {
 | |
|       inSingleRunPassedKey: string
 | |
|     }> = {}
 | |
|     loopChildrenNodes.forEach((node) => {
 | |
|       const nodeVars = getNodeUsedVars(node).filter(item => item && item.length > 0)
 | |
|       nodeVars.forEach((varSelector) => {
 | |
|         if (varSelector[0] === id) { // skip Loop node itself variable: item, index
 | |
|           return
 | |
|         }
 | |
|         const isInLoop = isNodeInLoop(varSelector[0])
 | |
|         if (isInLoop) // not pass loop inner variable
 | |
|           return
 | |
| 
 | |
|         const varSectorStr = varSelector.join('.')
 | |
|         if (!varObjs[varSectorStr]) {
 | |
|           varObjs[varSectorStr] = true
 | |
|           vars.push(varSelector)
 | |
|         }
 | |
|         let passToServerKeys = getNodeUsedVarPassToServerKey(node, varSelector)
 | |
|         if (typeof passToServerKeys === 'string')
 | |
|           passToServerKeys = [passToServerKeys]
 | |
| 
 | |
|         passToServerKeys.forEach((key: string, index: number) => {
 | |
|           allVarObject[[varSectorStr, node.id, index].join(DELIMITER)] = {
 | |
|             inSingleRunPassedKey: key,
 | |
|           }
 | |
|         })
 | |
|       })
 | |
|     })
 | |
|     const res = toVarInputs(vars.map((item) => {
 | |
|       const varInfo = getNodeInfoById(canChooseVarNodes, item[0])
 | |
|       return {
 | |
|         label: {
 | |
|           nodeType: varInfo?.data.type,
 | |
|           nodeName: varInfo?.data.title || canChooseVarNodes[0]?.data.title, // default start node title
 | |
|           variable: isSystemVar(item) ? item.join('.') : item[item.length - 1],
 | |
|         },
 | |
|         variable: `${item.join('.')}`,
 | |
|         value_selector: item,
 | |
|       }
 | |
|     }))
 | |
|     return {
 | |
|       usedOutVars: res,
 | |
|       allVarObject,
 | |
|     }
 | |
|   })()
 | |
| 
 | |
|   const handleRun = useCallback((data: Record<string, any>) => {
 | |
|     const formattedData: Record<string, any> = {}
 | |
|     Object.keys(allVarObject).forEach((key) => {
 | |
|       const [varSectorStr, nodeId] = key.split(DELIMITER)
 | |
|       formattedData[`${nodeId}.${allVarObject[key].inSingleRunPassedKey}`] = data[varSectorStr]
 | |
|     })
 | |
|     formattedData[loopInputKey] = data[loopInputKey]
 | |
|     doHandleRun(formattedData)
 | |
|   }, [allVarObject, doHandleRun, loopInputKey])
 | |
| 
 | |
|   const inputVarValues = (() => {
 | |
|     const vars: Record<string, any> = {}
 | |
|     Object.keys(runInputData)
 | |
|       .filter(key => ![loopInputKey].includes(key))
 | |
|       .forEach((key) => {
 | |
|         vars[key] = runInputData[key]
 | |
|       })
 | |
|     return vars
 | |
|   })()
 | |
| 
 | |
|   const setInputVarValues = useCallback((newPayload: Record<string, any>) => {
 | |
|     const newVars = {
 | |
|       ...newPayload,
 | |
|       [loopInputKey]: runInputData[loopInputKey],
 | |
|     }
 | |
|     setRunInputData(newVars)
 | |
|   }, [loopInputKey, runInputData, setRunInputData])
 | |
| 
 | |
|   const loop = runInputData[loopInputKey]
 | |
|   const setLoop = useCallback((newLoop: string[]) => {
 | |
|     setRunInputData({
 | |
|       ...runInputData,
 | |
|       [loopInputKey]: newLoop,
 | |
|     })
 | |
|   }, [loopInputKey, runInputData, setRunInputData])
 | |
| 
 | |
|   const changeErrorResponseMode = useCallback((item: { value: unknown }) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       draft.error_handle_mode = item.value as ErrorHandleMode
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleAddCondition = useCallback<HandleAddCondition>((valueSelector, varItem) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       if (!draft.break_conditions)
 | |
|         draft.break_conditions = []
 | |
| 
 | |
|       draft.break_conditions?.push({
 | |
|         id: uuid4(),
 | |
|         varType: varItem.type,
 | |
|         variable_selector: valueSelector,
 | |
|         comparison_operator: getOperators(varItem.type, getIsVarFileAttribute(valueSelector) ? { key: valueSelector.slice(-1)[0] } : undefined)[0],
 | |
|         value: '',
 | |
|       })
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [getIsVarFileAttribute, inputs, setInputs])
 | |
| 
 | |
|   const handleRemoveCondition = useCallback<HandleRemoveCondition>((conditionId) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       draft.break_conditions = draft.break_conditions?.filter(item => item.id !== conditionId)
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleUpdateCondition = useCallback<HandleUpdateCondition>((conditionId, newCondition) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       const targetCondition = draft.break_conditions?.find(item => item.id === conditionId)
 | |
|       if (targetCondition)
 | |
|         Object.assign(targetCondition, newCondition)
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleToggleConditionLogicalOperator = useCallback<HandleToggleConditionLogicalOperator>(() => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       draft.logical_operator = draft.logical_operator === LogicalOperator.and ? LogicalOperator.or : LogicalOperator.and
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleAddSubVariableCondition = useCallback<HandleAddSubVariableCondition>((conditionId: string, key?: string) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       const condition = draft.break_conditions?.find(item => item.id === conditionId)
 | |
|       if (!condition)
 | |
|         return
 | |
|       if (!condition?.sub_variable_condition) {
 | |
|         condition.sub_variable_condition = {
 | |
|           logical_operator: LogicalOperator.and,
 | |
|           conditions: [],
 | |
|         }
 | |
|       }
 | |
|       const subVarCondition = condition.sub_variable_condition
 | |
|       if (subVarCondition) {
 | |
|         if (!subVarCondition.conditions)
 | |
|           subVarCondition.conditions = []
 | |
| 
 | |
|         const svcComparisonOperators = getOperators(VarType.string, { key: key || '' })
 | |
| 
 | |
|         subVarCondition.conditions.push({
 | |
|           id: uuid4(),
 | |
|           key: key || '',
 | |
|           varType: VarType.string,
 | |
|           comparison_operator: (svcComparisonOperators && svcComparisonOperators.length) ? svcComparisonOperators[0] : undefined,
 | |
|           value: '',
 | |
|         })
 | |
|       }
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleRemoveSubVariableCondition = useCallback((conditionId: string, subConditionId: string) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       const condition = draft.break_conditions?.find(item => item.id === conditionId)
 | |
|       if (!condition)
 | |
|         return
 | |
|       if (!condition?.sub_variable_condition)
 | |
|         return
 | |
|       const subVarCondition = condition.sub_variable_condition
 | |
|       if (subVarCondition)
 | |
|         subVarCondition.conditions = subVarCondition.conditions.filter(item => item.id !== subConditionId)
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleUpdateSubVariableCondition = useCallback<HandleUpdateSubVariableCondition>((conditionId, subConditionId, newSubCondition) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       const targetCondition = draft.break_conditions?.find(item => item.id === conditionId)
 | |
|       if (targetCondition && targetCondition.sub_variable_condition) {
 | |
|         const targetSubCondition = targetCondition.sub_variable_condition.conditions.find(item => item.id === subConditionId)
 | |
|         if (targetSubCondition)
 | |
|           Object.assign(targetSubCondition, newSubCondition)
 | |
|       }
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleToggleSubVariableConditionLogicalOperator = useCallback<HandleToggleSubVariableConditionLogicalOperator>((conditionId) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       const targetCondition = draft.break_conditions?.find(item => item.id === conditionId)
 | |
|       if (targetCondition && targetCondition.sub_variable_condition)
 | |
|         targetCondition.sub_variable_condition.logical_operator = targetCondition.sub_variable_condition.logical_operator === LogicalOperator.and ? LogicalOperator.or : LogicalOperator.and
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleUpdateLoopCount = useCallback((value: number) => {
 | |
|     const newInputs = produce(inputs, (draft) => {
 | |
|       draft.loop_count = value
 | |
|     })
 | |
|     setInputs(newInputs)
 | |
|   }, [inputs, setInputs])
 | |
| 
 | |
|   const handleAddLoopVariable = useCallback(() => {
 | |
|     const newInputs = produce(inputsRef.current, (draft) => {
 | |
|       if (!draft.loop_variables)
 | |
|         draft.loop_variables = []
 | |
| 
 | |
|       draft.loop_variables.push({
 | |
|         id: uuid4(),
 | |
|         label: '',
 | |
|         var_type: VarType.string,
 | |
|         value_type: ValueType.constant,
 | |
|         value: '',
 | |
|       })
 | |
|     })
 | |
|     handleInputsChange(newInputs)
 | |
|   }, [handleInputsChange])
 | |
| 
 | |
|   const handleRemoveLoopVariable = useCallback((id: string) => {
 | |
|     const newInputs = produce(inputsRef.current, (draft) => {
 | |
|       draft.loop_variables = draft.loop_variables?.filter(item => item.id !== id)
 | |
|     })
 | |
|     handleInputsChange(newInputs)
 | |
|   }, [handleInputsChange])
 | |
| 
 | |
|   const handleUpdateLoopVariable = useCallback((id: string, updateData: any) => {
 | |
|     const loopVariables = inputsRef.current.loop_variables || []
 | |
|     const index = loopVariables.findIndex(item => item.id === id)
 | |
|     const newInputs = produce(inputsRef.current, (draft) => {
 | |
|       if (index > -1) {
 | |
|         draft.loop_variables![index] = {
 | |
|           ...draft.loop_variables![index],
 | |
|           ...updateData,
 | |
|         }
 | |
|       }
 | |
|     })
 | |
|     handleInputsChange(newInputs)
 | |
|   }, [handleInputsChange])
 | |
| 
 | |
|   return {
 | |
|     readOnly,
 | |
|     inputs,
 | |
|     filterInputVar,
 | |
|     childrenNodeVars,
 | |
|     loopChildrenNodes,
 | |
|     isShowSingleRun,
 | |
|     showSingleRun,
 | |
|     hideSingleRun,
 | |
|     isShowLoopDetail,
 | |
|     showLoopDetail,
 | |
|     hideLoopDetail,
 | |
|     backToSingleRun,
 | |
|     runningStatus,
 | |
|     handleRun,
 | |
|     handleStop,
 | |
|     runResult,
 | |
|     inputVarValues,
 | |
|     setInputVarValues,
 | |
|     usedOutVars,
 | |
|     loop,
 | |
|     setLoop,
 | |
|     loopInputKey,
 | |
|     loopRunResult,
 | |
|     handleAddCondition,
 | |
|     handleRemoveCondition,
 | |
|     handleUpdateCondition,
 | |
|     handleToggleConditionLogicalOperator,
 | |
|     handleAddSubVariableCondition,
 | |
|     handleUpdateSubVariableCondition,
 | |
|     handleRemoveSubVariableCondition,
 | |
|     handleToggleSubVariableConditionLogicalOperator,
 | |
|     handleUpdateLoopCount,
 | |
|     changeErrorResponseMode,
 | |
|     handleAddLoopVariable,
 | |
|     handleRemoveLoopVariable,
 | |
|     handleUpdateLoopVariable,
 | |
|   }
 | |
| }
 | |
| 
 | |
| export default useConfig
 | 
