| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332 | 
							- import { useCallback } from 'react'
 
- import produce from 'immer'
 
- import { useBoolean } from 'ahooks'
 
- import { uuid4 } from '@sentry/utils'
 
- import {
 
-   useIsChatMode,
 
-   useIsNodeInLoop,
 
-   useNodesReadOnly,
 
-   useWorkflow,
 
- } from '../../hooks'
 
- import { 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 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 = 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])
 
-   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,
 
-   }
 
- }
 
- export default useConfig
 
 
  |