You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

use-config.ts 6.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. import { useCallback, useEffect, useRef, useState } from 'react'
  2. import produce from 'immer'
  3. import { BlockEnum, VarType } from '../../types'
  4. import type { Memory, ValueSelector, Var } from '../../types'
  5. import {
  6. useIsChatMode, useNodesReadOnly,
  7. useWorkflow,
  8. } from '../../hooks'
  9. import { useStore } from '../../store'
  10. import useAvailableVarList from '../_base/hooks/use-available-var-list'
  11. import useConfigVision from '../../hooks/use-config-vision'
  12. import type { QuestionClassifierNodeType } from './types'
  13. import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud'
  14. import { useModelListAndDefaultModelAndCurrentProviderAndModel } from '@/app/components/header/account-setting/model-provider-page/hooks'
  15. import { ModelTypeEnum } from '@/app/components/header/account-setting/model-provider-page/declarations'
  16. import { checkHasQueryBlock } from '@/app/components/base/prompt-editor/constants'
  17. const useConfig = (id: string, payload: QuestionClassifierNodeType) => {
  18. const { nodesReadOnly: readOnly } = useNodesReadOnly()
  19. const isChatMode = useIsChatMode()
  20. const defaultConfig = useStore(s => s.nodesDefaultConfigs)[payload.type]
  21. const { getBeforeNodesInSameBranch } = useWorkflow()
  22. const startNode = getBeforeNodesInSameBranch(id).find(node => node.data.type === BlockEnum.Start)
  23. const startNodeId = startNode?.id
  24. const { inputs, setInputs } = useNodeCrud<QuestionClassifierNodeType>(id, payload)
  25. const inputRef = useRef(inputs)
  26. useEffect(() => {
  27. inputRef.current = inputs
  28. }, [inputs])
  29. const [modelChanged, setModelChanged] = useState(false)
  30. const {
  31. currentProvider,
  32. currentModel,
  33. } = useModelListAndDefaultModelAndCurrentProviderAndModel(ModelTypeEnum.textGeneration)
  34. const model = inputs.model
  35. const modelMode = inputs.model?.mode
  36. const isChatModel = modelMode === 'chat'
  37. const {
  38. isVisionModel,
  39. handleVisionResolutionEnabledChange,
  40. handleVisionResolutionChange,
  41. handleModelChanged: handleVisionConfigAfterModelChanged,
  42. } = useConfigVision(model, {
  43. payload: inputs.vision,
  44. onChange: (newPayload) => {
  45. const newInputs = produce(inputs, (draft) => {
  46. draft.vision = newPayload
  47. })
  48. setInputs(newInputs)
  49. },
  50. })
  51. const handleModelChanged = useCallback((model: { provider: string; modelId: string; mode?: string }) => {
  52. const newInputs = produce(inputRef.current, (draft) => {
  53. draft.model.provider = model.provider
  54. draft.model.name = model.modelId
  55. draft.model.mode = model.mode!
  56. })
  57. setInputs(newInputs)
  58. setModelChanged(true)
  59. }, [setInputs])
  60. useEffect(() => {
  61. if (currentProvider?.provider && currentModel?.model && !model.provider) {
  62. handleModelChanged({
  63. provider: currentProvider?.provider,
  64. modelId: currentModel?.model,
  65. mode: currentModel?.model_properties?.mode as string,
  66. })
  67. }
  68. }, [model.provider, currentProvider, currentModel, handleModelChanged])
  69. const handleCompletionParamsChange = useCallback((newParams: Record<string, any>) => {
  70. const newInputs = produce(inputs, (draft) => {
  71. draft.model.completion_params = newParams
  72. })
  73. setInputs(newInputs)
  74. }, [inputs, setInputs])
  75. // change to vision model to set vision enabled, else disabled
  76. useEffect(() => {
  77. if (!modelChanged)
  78. return
  79. setModelChanged(false)
  80. handleVisionConfigAfterModelChanged()
  81. // eslint-disable-next-line react-hooks/exhaustive-deps
  82. }, [isVisionModel, modelChanged])
  83. const handleQueryVarChange = useCallback((newVar: ValueSelector | string) => {
  84. const newInputs = produce(inputs, (draft) => {
  85. draft.query_variable_selector = newVar as ValueSelector
  86. })
  87. setInputs(newInputs)
  88. }, [inputs, setInputs])
  89. useEffect(() => {
  90. const isReady = defaultConfig && Object.keys(defaultConfig).length > 0
  91. if (isReady) {
  92. let query_variable_selector: ValueSelector = []
  93. if (isChatMode && inputs.query_variable_selector.length === 0 && startNodeId)
  94. query_variable_selector = [startNodeId, 'sys.query']
  95. setInputs({
  96. ...inputs,
  97. ...defaultConfig,
  98. query_variable_selector: inputs.query_variable_selector.length > 0 ? inputs.query_variable_selector : query_variable_selector,
  99. })
  100. }
  101. // eslint-disable-next-line react-hooks/exhaustive-deps
  102. }, [defaultConfig])
  103. const handleClassesChange = useCallback((newClasses: any) => {
  104. const newInputs = produce(inputs, (draft) => {
  105. draft.classes = newClasses
  106. draft._targetBranches = newClasses
  107. })
  108. setInputs(newInputs)
  109. }, [inputs, setInputs])
  110. const filterInputVar = useCallback((varPayload: Var) => {
  111. return [VarType.number, VarType.string].includes(varPayload.type)
  112. }, [])
  113. const filterVisionInputVar = useCallback((varPayload: Var) => {
  114. return [VarType.file, VarType.arrayFile].includes(varPayload.type)
  115. }, [])
  116. const {
  117. availableVars,
  118. availableNodesWithParent,
  119. } = useAvailableVarList(id, {
  120. onlyLeafNodeVar: false,
  121. filterVar: filterInputVar,
  122. })
  123. const {
  124. availableVars: availableVisionVars,
  125. } = useAvailableVarList(id, {
  126. onlyLeafNodeVar: false,
  127. filterVar: filterVisionInputVar,
  128. })
  129. const hasSetBlockStatus = {
  130. history: false,
  131. query: isChatMode ? checkHasQueryBlock(inputs.instruction) : false,
  132. context: false,
  133. }
  134. const handleInstructionChange = useCallback((instruction: string) => {
  135. const newInputs = produce(inputs, (draft) => {
  136. draft.instruction = instruction
  137. })
  138. setInputs(newInputs)
  139. }, [inputs, setInputs])
  140. const handleMemoryChange = useCallback((memory?: Memory) => {
  141. const newInputs = produce(inputs, (draft) => {
  142. draft.memory = memory
  143. })
  144. setInputs(newInputs)
  145. }, [inputs, setInputs])
  146. const filterVar = useCallback((varPayload: Var) => {
  147. return varPayload.type === VarType.string
  148. }, [])
  149. return {
  150. readOnly,
  151. inputs,
  152. handleModelChanged,
  153. isChatMode,
  154. isChatModel,
  155. handleCompletionParamsChange,
  156. handleQueryVarChange,
  157. filterVar,
  158. handleTopicsChange: handleClassesChange,
  159. hasSetBlockStatus,
  160. availableVars,
  161. availableNodesWithParent,
  162. availableVisionVars,
  163. handleInstructionChange,
  164. handleMemoryChange,
  165. isVisionModel,
  166. handleVisionResolutionEnabledChange,
  167. handleVisionResolutionChange,
  168. }
  169. }
  170. export default useConfig