use-config.ts 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. import { useCallback } from 'react'
  2. import produce from 'immer'
  3. import { useBoolean } from 'ahooks'
  4. import {
  5. useIsChatMode,
  6. useIsNodeInIteration,
  7. useNodesReadOnly,
  8. useWorkflow,
  9. } from '../../hooks'
  10. import { VarType } from '../../types'
  11. import type { ValueSelector, Var } from '../../types'
  12. import useNodeCrud from '../_base/hooks/use-node-crud'
  13. import { getNodeInfoById, getNodeUsedVarPassToServerKey, getNodeUsedVars, isSystemVar, toNodeOutputVars } from '../_base/components/variable/utils'
  14. import useOneStepRun from '../_base/hooks/use-one-step-run'
  15. import type { IterationNodeType } from './types'
  16. import type { VarType as VarKindType } from '@/app/components/workflow/nodes/tool/types'
  17. const DELIMITER = '@@@@@'
  18. const useConfig = (id: string, payload: IterationNodeType) => {
  19. const { nodesReadOnly: readOnly } = useNodesReadOnly()
  20. const { isNodeInIteration } = useIsNodeInIteration(id)
  21. const isChatMode = useIsChatMode()
  22. const { inputs, setInputs } = useNodeCrud<IterationNodeType>(id, payload)
  23. const filterInputVar = useCallback((varPayload: Var) => {
  24. return [VarType.array, VarType.arrayString, VarType.arrayNumber, VarType.arrayObject, VarType.arrayFile].includes(varPayload.type)
  25. }, [])
  26. const handleInputChange = useCallback((input: ValueSelector | string) => {
  27. const newInputs = produce(inputs, (draft) => {
  28. draft.iterator_selector = input as ValueSelector || []
  29. })
  30. setInputs(newInputs)
  31. }, [inputs, setInputs])
  32. // output
  33. const { getIterationNodeChildren, getBeforeNodesInSameBranch } = useWorkflow()
  34. const beforeNodes = getBeforeNodesInSameBranch(id)
  35. const iterationChildrenNodes = getIterationNodeChildren(id)
  36. const canChooseVarNodes = [...beforeNodes, ...iterationChildrenNodes]
  37. const childrenNodeVars = toNodeOutputVars(iterationChildrenNodes, isChatMode)
  38. const handleOutputVarChange = useCallback((output: ValueSelector | string, _varKindType: VarKindType, varInfo?: Var) => {
  39. const newInputs = produce(inputs, (draft) => {
  40. draft.output_selector = output as ValueSelector || []
  41. const outputItemType = varInfo?.type || VarType.string
  42. draft.output_type = ({
  43. [VarType.string]: VarType.arrayString,
  44. [VarType.number]: VarType.arrayNumber,
  45. [VarType.object]: VarType.arrayObject,
  46. [VarType.file]: VarType.arrayFile,
  47. } as Record<VarType, VarType>)[outputItemType] || VarType.arrayString
  48. })
  49. setInputs(newInputs)
  50. }, [inputs, setInputs])
  51. // single run
  52. const iteratorInputKey = `${id}.input_selector`
  53. const {
  54. isShowSingleRun,
  55. showSingleRun,
  56. hideSingleRun,
  57. toVarInputs,
  58. runningStatus,
  59. handleRun: doHandleRun,
  60. handleStop,
  61. runInputData,
  62. setRunInputData,
  63. runResult,
  64. iterationRunResult,
  65. } = useOneStepRun<IterationNodeType>({
  66. id,
  67. data: inputs,
  68. iteratorInputKey,
  69. defaultRunInputData: {
  70. [iteratorInputKey]: [''],
  71. },
  72. })
  73. const [isShowIterationDetail, {
  74. setTrue: doShowIterationDetail,
  75. setFalse: doHideIterationDetail,
  76. }] = useBoolean(false)
  77. const hideIterationDetail = useCallback(() => {
  78. hideSingleRun()
  79. doHideIterationDetail()
  80. }, [doHideIterationDetail, hideSingleRun])
  81. const showIterationDetail = useCallback(() => {
  82. doShowIterationDetail()
  83. }, [doShowIterationDetail])
  84. const backToSingleRun = useCallback(() => {
  85. hideIterationDetail()
  86. showSingleRun()
  87. }, [hideIterationDetail, showSingleRun])
  88. const { usedOutVars, allVarObject } = (() => {
  89. const vars: ValueSelector[] = []
  90. const varObjs: Record<string, boolean> = {}
  91. const allVarObject: Record<string, {
  92. inSingleRunPassedKey: string
  93. }> = {}
  94. iterationChildrenNodes.forEach((node) => {
  95. const nodeVars = getNodeUsedVars(node).filter(item => item && item.length > 0)
  96. nodeVars.forEach((varSelector) => {
  97. if (varSelector[0] === id) { // skip iteration node itself variable: item, index
  98. return
  99. }
  100. const isInIteration = isNodeInIteration(varSelector[0])
  101. if (isInIteration) // not pass iteration inner variable
  102. return
  103. const varSectorStr = varSelector.join('.')
  104. if (!varObjs[varSectorStr]) {
  105. varObjs[varSectorStr] = true
  106. vars.push(varSelector)
  107. }
  108. let passToServerKeys = getNodeUsedVarPassToServerKey(node, varSelector)
  109. if (typeof passToServerKeys === 'string')
  110. passToServerKeys = [passToServerKeys]
  111. passToServerKeys.forEach((key: string, index: number) => {
  112. allVarObject[[varSectorStr, node.id, index].join(DELIMITER)] = {
  113. inSingleRunPassedKey: key,
  114. }
  115. })
  116. })
  117. })
  118. const res = toVarInputs(vars.map((item) => {
  119. const varInfo = getNodeInfoById(canChooseVarNodes, item[0])
  120. return {
  121. label: {
  122. nodeType: varInfo?.data.type,
  123. nodeName: varInfo?.data.title || canChooseVarNodes[0]?.data.title, // default start node title
  124. variable: isSystemVar(item) ? item.join('.') : item[item.length - 1],
  125. },
  126. variable: `${item.join('.')}`,
  127. value_selector: item,
  128. }
  129. }))
  130. return {
  131. usedOutVars: res,
  132. allVarObject,
  133. }
  134. })()
  135. const handleRun = useCallback((data: Record<string, any>) => {
  136. const formattedData: Record<string, any> = {}
  137. Object.keys(allVarObject).forEach((key) => {
  138. const [varSectorStr, nodeId] = key.split(DELIMITER)
  139. formattedData[`${nodeId}.${allVarObject[key].inSingleRunPassedKey}`] = data[varSectorStr]
  140. })
  141. formattedData[iteratorInputKey] = data[iteratorInputKey]
  142. doHandleRun(formattedData)
  143. }, [allVarObject, doHandleRun, iteratorInputKey])
  144. const inputVarValues = (() => {
  145. const vars: Record<string, any> = {}
  146. Object.keys(runInputData)
  147. .filter(key => ![iteratorInputKey].includes(key))
  148. .forEach((key) => {
  149. vars[key] = runInputData[key]
  150. })
  151. return vars
  152. })()
  153. const setInputVarValues = useCallback((newPayload: Record<string, any>) => {
  154. const newVars = {
  155. ...newPayload,
  156. [iteratorInputKey]: runInputData[iteratorInputKey],
  157. }
  158. setRunInputData(newVars)
  159. }, [iteratorInputKey, runInputData, setRunInputData])
  160. const iterator = runInputData[iteratorInputKey]
  161. const setIterator = useCallback((newIterator: string[]) => {
  162. setRunInputData({
  163. ...runInputData,
  164. [iteratorInputKey]: newIterator,
  165. })
  166. }, [iteratorInputKey, runInputData, setRunInputData])
  167. return {
  168. readOnly,
  169. inputs,
  170. filterInputVar,
  171. handleInputChange,
  172. childrenNodeVars,
  173. iterationChildrenNodes,
  174. handleOutputVarChange,
  175. isShowSingleRun,
  176. showSingleRun,
  177. hideSingleRun,
  178. isShowIterationDetail,
  179. showIterationDetail,
  180. hideIterationDetail,
  181. backToSingleRun,
  182. runningStatus,
  183. handleRun,
  184. handleStop,
  185. runResult,
  186. inputVarValues,
  187. setInputVarValues,
  188. usedOutVars,
  189. iterator,
  190. setIterator,
  191. iteratorInputKey,
  192. iterationRunResult,
  193. }
  194. }
  195. export default useConfig