Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448
  1. import { useCallback, useEffect, useRef, useState } from 'react'
  2. import produce from 'immer'
  3. import { EditionType, VarType } from '../../types'
  4. import type { Memory, PromptItem, ValueSelector, Var, Variable } from '../../types'
  5. import { useStore } from '../../store'
  6. import {
  7. useIsChatMode,
  8. useNodesReadOnly,
  9. } from '../../hooks'
  10. import useAvailableVarList from '../_base/hooks/use-available-var-list'
  11. import useConfigVision from '../../hooks/use-config-vision'
  12. import type { LLMNodeType, StructuredOutput } from './types'
  13. import { useModelList, useModelListAndDefaultModelAndCurrentProviderAndModel } from '@/app/components/header/account-setting/model-provider-page/hooks'
  14. import {
  15. ModelFeatureEnum,
  16. ModelTypeEnum,
  17. } from '@/app/components/header/account-setting/model-provider-page/declarations'
  18. import useNodeCrud from '@/app/components/workflow/nodes/_base/hooks/use-node-crud'
  19. import useOneStepRun from '@/app/components/workflow/nodes/_base/hooks/use-one-step-run'
  20. import { RETRIEVAL_OUTPUT_STRUCT } from '@/app/components/workflow/constants'
  21. import { checkHasContextBlock, checkHasHistoryBlock, checkHasQueryBlock } from '@/app/components/base/prompt-editor/constants'
  22. const useConfig = (id: string, payload: LLMNodeType) => {
  23. const { nodesReadOnly: readOnly } = useNodesReadOnly()
  24. const isChatMode = useIsChatMode()
  25. const defaultConfig = useStore(s => s.nodesDefaultConfigs)[payload.type]
  26. const [defaultRolePrefix, setDefaultRolePrefix] = useState<{ user: string; assistant: string }>({ user: '', assistant: '' })
  27. const { inputs, setInputs: doSetInputs } = useNodeCrud<LLMNodeType>(id, payload)
  28. const inputRef = useRef(inputs)
  29. const setInputs = useCallback((newInputs: LLMNodeType) => {
  30. if (newInputs.memory && !newInputs.memory.role_prefix) {
  31. const newPayload = produce(newInputs, (draft) => {
  32. draft.memory!.role_prefix = defaultRolePrefix
  33. })
  34. doSetInputs(newPayload)
  35. inputRef.current = newPayload
  36. return
  37. }
  38. doSetInputs(newInputs)
  39. inputRef.current = newInputs
  40. }, [doSetInputs, defaultRolePrefix])
  41. // model
  42. const model = inputs.model
  43. const modelMode = inputs.model?.mode
  44. const isChatModel = modelMode === 'chat'
  45. const isCompletionModel = !isChatModel
  46. const hasSetBlockStatus = (() => {
  47. const promptTemplate = inputs.prompt_template
  48. const hasSetContext = isChatModel ? (promptTemplate as PromptItem[]).some(item => checkHasContextBlock(item.text)) : checkHasContextBlock((promptTemplate as PromptItem).text)
  49. if (!isChatMode) {
  50. return {
  51. history: false,
  52. query: false,
  53. context: hasSetContext,
  54. }
  55. }
  56. if (isChatModel) {
  57. return {
  58. history: false,
  59. query: (promptTemplate as PromptItem[]).some(item => checkHasQueryBlock(item.text)),
  60. context: hasSetContext,
  61. }
  62. }
  63. else {
  64. return {
  65. history: checkHasHistoryBlock((promptTemplate as PromptItem).text),
  66. query: checkHasQueryBlock((promptTemplate as PromptItem).text),
  67. context: hasSetContext,
  68. }
  69. }
  70. })()
  71. const shouldShowContextTip = !hasSetBlockStatus.context && inputs.context.enabled
  72. const appendDefaultPromptConfig = useCallback((draft: LLMNodeType, defaultConfig: any, passInIsChatMode?: boolean) => {
  73. const promptTemplates = defaultConfig.prompt_templates
  74. if (passInIsChatMode === undefined ? isChatModel : passInIsChatMode) {
  75. draft.prompt_template = promptTemplates.chat_model.prompts
  76. }
  77. else {
  78. draft.prompt_template = promptTemplates.completion_model.prompt
  79. setDefaultRolePrefix({
  80. user: promptTemplates.completion_model.conversation_histories_role.user_prefix,
  81. assistant: promptTemplates.completion_model.conversation_histories_role.assistant_prefix,
  82. })
  83. }
  84. }, [isChatModel])
  85. useEffect(() => {
  86. const isReady = defaultConfig && Object.keys(defaultConfig).length > 0
  87. if (isReady && !inputs.prompt_template) {
  88. const newInputs = produce(inputs, (draft) => {
  89. appendDefaultPromptConfig(draft, defaultConfig)
  90. })
  91. setInputs(newInputs)
  92. }
  93. // eslint-disable-next-line react-hooks/exhaustive-deps
  94. }, [defaultConfig, isChatModel])
  95. const [modelChanged, setModelChanged] = useState(false)
  96. const {
  97. currentProvider,
  98. currentModel,
  99. } = useModelListAndDefaultModelAndCurrentProviderAndModel(ModelTypeEnum.textGeneration)
  100. const {
  101. isVisionModel,
  102. handleVisionResolutionEnabledChange,
  103. handleVisionResolutionChange,
  104. handleModelChanged: handleVisionConfigAfterModelChanged,
  105. } = useConfigVision(model, {
  106. payload: inputs.vision,
  107. onChange: (newPayload) => {
  108. const newInputs = produce(inputs, (draft) => {
  109. draft.vision = newPayload
  110. })
  111. setInputs(newInputs)
  112. },
  113. })
  114. const handleModelChanged = useCallback((model: { provider: string; modelId: string; mode?: string }) => {
  115. const newInputs = produce(inputRef.current, (draft) => {
  116. draft.model.provider = model.provider
  117. draft.model.name = model.modelId
  118. draft.model.mode = model.mode!
  119. const isModeChange = model.mode !== inputRef.current.model.mode
  120. if (isModeChange && defaultConfig && Object.keys(defaultConfig).length > 0)
  121. appendDefaultPromptConfig(draft, defaultConfig, model.mode === 'chat')
  122. })
  123. setInputs(newInputs)
  124. setModelChanged(true)
  125. }, [setInputs, defaultConfig, appendDefaultPromptConfig])
  126. useEffect(() => {
  127. if (currentProvider?.provider && currentModel?.model && !model.provider) {
  128. handleModelChanged({
  129. provider: currentProvider?.provider,
  130. modelId: currentModel?.model,
  131. mode: currentModel?.model_properties?.mode as string,
  132. })
  133. }
  134. }, [model.provider, currentProvider, currentModel, handleModelChanged])
  135. const handleCompletionParamsChange = useCallback((newParams: Record<string, any>) => {
  136. const newInputs = produce(inputs, (draft) => {
  137. draft.model.completion_params = newParams
  138. })
  139. setInputs(newInputs)
  140. }, [inputs, setInputs])
  141. // change to vision model to set vision enabled, else disabled
  142. useEffect(() => {
  143. if (!modelChanged)
  144. return
  145. setModelChanged(false)
  146. handleVisionConfigAfterModelChanged()
  147. // eslint-disable-next-line react-hooks/exhaustive-deps
  148. }, [isVisionModel, modelChanged])
  149. // variables
  150. const isShowVars = (() => {
  151. if (isChatModel)
  152. return (inputs.prompt_template as PromptItem[]).some(item => item.edition_type === EditionType.jinja2)
  153. return (inputs.prompt_template as PromptItem).edition_type === EditionType.jinja2
  154. })()
  155. const handleAddEmptyVariable = useCallback(() => {
  156. const newInputs = produce(inputRef.current, (draft) => {
  157. if (!draft.prompt_config) {
  158. draft.prompt_config = {
  159. jinja2_variables: [],
  160. }
  161. }
  162. if (!draft.prompt_config.jinja2_variables)
  163. draft.prompt_config.jinja2_variables = []
  164. draft.prompt_config.jinja2_variables.push({
  165. variable: '',
  166. value_selector: [],
  167. })
  168. })
  169. setInputs(newInputs)
  170. }, [setInputs])
  171. const handleAddVariable = useCallback((payload: Variable) => {
  172. const newInputs = produce(inputRef.current, (draft) => {
  173. if (!draft.prompt_config) {
  174. draft.prompt_config = {
  175. jinja2_variables: [],
  176. }
  177. }
  178. if (!draft.prompt_config.jinja2_variables)
  179. draft.prompt_config.jinja2_variables = []
  180. draft.prompt_config.jinja2_variables.push(payload)
  181. })
  182. setInputs(newInputs)
  183. }, [setInputs])
  184. const handleVarListChange = useCallback((newList: Variable[]) => {
  185. const newInputs = produce(inputRef.current, (draft) => {
  186. if (!draft.prompt_config) {
  187. draft.prompt_config = {
  188. jinja2_variables: [],
  189. }
  190. }
  191. if (!draft.prompt_config.jinja2_variables)
  192. draft.prompt_config.jinja2_variables = []
  193. draft.prompt_config.jinja2_variables = newList
  194. })
  195. setInputs(newInputs)
  196. }, [setInputs])
  197. const handleVarNameChange = useCallback((oldName: string, newName: string) => {
  198. const newInputs = produce(inputRef.current, (draft) => {
  199. if (isChatModel) {
  200. const promptTemplate = draft.prompt_template as PromptItem[]
  201. promptTemplate.filter(item => item.edition_type === EditionType.jinja2).forEach((item) => {
  202. item.jinja2_text = (item.jinja2_text || '').replaceAll(`{{ ${oldName} }}`, `{{ ${newName} }}`)
  203. })
  204. }
  205. else {
  206. if ((draft.prompt_template as PromptItem).edition_type !== EditionType.jinja2)
  207. return
  208. const promptTemplate = draft.prompt_template as PromptItem
  209. promptTemplate.jinja2_text = (promptTemplate.jinja2_text || '').replaceAll(`{{ ${oldName} }}`, `{{ ${newName} }}`)
  210. }
  211. })
  212. setInputs(newInputs)
  213. }, [isChatModel, setInputs])
  214. // context
  215. const handleContextVarChange = useCallback((newVar: ValueSelector | string) => {
  216. const newInputs = produce(inputs, (draft) => {
  217. draft.context.variable_selector = newVar as ValueSelector || []
  218. draft.context.enabled = !!(newVar && newVar.length > 0)
  219. })
  220. setInputs(newInputs)
  221. }, [inputs, setInputs])
  222. const handlePromptChange = useCallback((newPrompt: PromptItem[] | PromptItem) => {
  223. const newInputs = produce(inputRef.current, (draft) => {
  224. draft.prompt_template = newPrompt
  225. })
  226. setInputs(newInputs)
  227. }, [setInputs])
  228. const handleMemoryChange = useCallback((newMemory?: Memory) => {
  229. const newInputs = produce(inputs, (draft) => {
  230. draft.memory = newMemory
  231. })
  232. setInputs(newInputs)
  233. }, [inputs, setInputs])
  234. const handleSyeQueryChange = useCallback((newQuery: string) => {
  235. const newInputs = produce(inputs, (draft) => {
  236. if (!draft.memory) {
  237. draft.memory = {
  238. window: {
  239. enabled: false,
  240. size: 10,
  241. },
  242. query_prompt_template: newQuery,
  243. }
  244. }
  245. else {
  246. draft.memory.query_prompt_template = newQuery
  247. }
  248. })
  249. setInputs(newInputs)
  250. }, [inputs, setInputs])
  251. // structure output
  252. const { data: modelList } = useModelList(ModelTypeEnum.textGeneration)
  253. const isModelSupportStructuredOutput = modelList
  254. ?.find(provideItem => provideItem.provider === model?.provider)
  255. ?.models.find(modelItem => modelItem.model === model?.name)
  256. ?.features?.includes(ModelFeatureEnum.StructuredOutput)
  257. const [structuredOutputCollapsed, setStructuredOutputCollapsed] = useState(true)
  258. const handleStructureOutputEnableChange = useCallback((enabled: boolean) => {
  259. const newInputs = produce(inputs, (draft) => {
  260. draft.structured_output_enabled = enabled
  261. })
  262. setInputs(newInputs)
  263. if (enabled)
  264. setStructuredOutputCollapsed(false)
  265. }, [inputs, setInputs])
  266. const handleStructureOutputChange = useCallback((newOutput: StructuredOutput) => {
  267. const newInputs = produce(inputs, (draft) => {
  268. draft.structured_output = newOutput
  269. })
  270. setInputs(newInputs)
  271. }, [inputs, setInputs])
  272. const filterInputVar = useCallback((varPayload: Var) => {
  273. return [VarType.number, VarType.string, VarType.secret, VarType.arrayString, VarType.arrayNumber, VarType.file, VarType.arrayFile].includes(varPayload.type)
  274. }, [])
  275. const filterJinjia2InputVar = useCallback((varPayload: Var) => {
  276. return [VarType.number, VarType.string, VarType.secret, VarType.arrayString, VarType.arrayNumber].includes(varPayload.type)
  277. }, [])
  278. const filterMemoryPromptVar = useCallback((varPayload: Var) => {
  279. return [VarType.arrayObject, VarType.array, VarType.number, VarType.string, VarType.secret, VarType.arrayString, VarType.arrayNumber, VarType.file, VarType.arrayFile].includes(varPayload.type)
  280. }, [])
  281. const {
  282. availableVars,
  283. availableNodesWithParent,
  284. } = useAvailableVarList(id, {
  285. onlyLeafNodeVar: false,
  286. filterVar: filterMemoryPromptVar,
  287. })
  288. // single run
  289. const {
  290. isShowSingleRun,
  291. hideSingleRun,
  292. getInputVars,
  293. runningStatus,
  294. handleRun,
  295. handleStop,
  296. runInputData,
  297. runInputDataRef,
  298. setRunInputData,
  299. runResult,
  300. toVarInputs,
  301. } = useOneStepRun<LLMNodeType>({
  302. id,
  303. data: inputs,
  304. defaultRunInputData: {
  305. '#context#': [RETRIEVAL_OUTPUT_STRUCT],
  306. '#files#': [],
  307. },
  308. })
  309. const inputVarValues = (() => {
  310. const vars: Record<string, any> = {}
  311. Object.keys(runInputData)
  312. .filter(key => !['#context#', '#files#'].includes(key))
  313. .forEach((key) => {
  314. vars[key] = runInputData[key]
  315. })
  316. return vars
  317. })()
  318. const setInputVarValues = useCallback((newPayload: Record<string, any>) => {
  319. const newVars = {
  320. ...newPayload,
  321. '#context#': runInputDataRef.current['#context#'],
  322. '#files#': runInputDataRef.current['#files#'],
  323. }
  324. setRunInputData(newVars)
  325. }, [runInputDataRef, setRunInputData])
  326. const contexts = runInputData['#context#']
  327. const setContexts = useCallback((newContexts: string[]) => {
  328. setRunInputData({
  329. ...runInputDataRef.current,
  330. '#context#': newContexts,
  331. })
  332. }, [runInputDataRef, setRunInputData])
  333. const visionFiles = runInputData['#files#']
  334. const setVisionFiles = useCallback((newFiles: any[]) => {
  335. setRunInputData({
  336. ...runInputDataRef.current,
  337. '#files#': newFiles,
  338. })
  339. }, [runInputDataRef, setRunInputData])
  340. const allVarStrArr = (() => {
  341. const arr = isChatModel ? (inputs.prompt_template as PromptItem[]).filter(item => item.edition_type !== EditionType.jinja2).map(item => item.text) : [(inputs.prompt_template as PromptItem).text]
  342. if (isChatMode && isChatModel && !!inputs.memory) {
  343. arr.push('{{#sys.query#}}')
  344. arr.push(inputs.memory.query_prompt_template)
  345. }
  346. return arr
  347. })()
  348. const varInputs = (() => {
  349. const vars = getInputVars(allVarStrArr)
  350. if (isShowVars)
  351. return [...vars, ...toVarInputs(inputs.prompt_config?.jinja2_variables || [])]
  352. return vars
  353. })()
  354. return {
  355. readOnly,
  356. isChatMode,
  357. inputs,
  358. isChatModel,
  359. isCompletionModel,
  360. hasSetBlockStatus,
  361. shouldShowContextTip,
  362. isVisionModel,
  363. handleModelChanged,
  364. handleCompletionParamsChange,
  365. isShowVars,
  366. handleVarListChange,
  367. handleVarNameChange,
  368. handleAddVariable,
  369. handleAddEmptyVariable,
  370. handleContextVarChange,
  371. filterInputVar,
  372. filterVar: filterMemoryPromptVar,
  373. availableVars,
  374. availableNodesWithParent,
  375. handlePromptChange,
  376. handleMemoryChange,
  377. handleSyeQueryChange,
  378. handleVisionResolutionEnabledChange,
  379. handleVisionResolutionChange,
  380. isShowSingleRun,
  381. hideSingleRun,
  382. inputVarValues,
  383. setInputVarValues,
  384. visionFiles,
  385. setVisionFiles,
  386. contexts,
  387. setContexts,
  388. varInputs,
  389. runningStatus,
  390. isModelSupportStructuredOutput,
  391. handleStructureOutputChange,
  392. structuredOutputCollapsed,
  393. setStructuredOutputCollapsed,
  394. handleStructureOutputEnableChange,
  395. handleRun,
  396. handleStop,
  397. runResult,
  398. filterJinjia2InputVar,
  399. }
  400. }
  401. export default useConfig