mirror of
https://github.com/langgenius/dify.git
synced 2026-04-22 03:37:44 +08:00
Frontend: - Migrate deprecated imports: modal→dialog, toast→ui/toast, tooltip→tooltip-plus, portal-to-follow-elem→portal-to-follow-elem-plus, select→ui/select, confirm→alert-dialog - Replace next/* with @/next/* wrapper modules - Convert TypeScript enums to const objects (erasable-syntax-only) - Replace all `any` types with `unknown` or specific types in workflow types - Fix unused vars, react-hooks-extra, react-refresh/only-export-components - Extract InteractionMode to separate module, tool-block commands to commands.ts Backend: - Fix pyrefly errors: type narrowing, null guards, getattr patterns - Remove unused TYPE_CHECKING imports in LLM node - Add ignore_imports entries to .importlinter for dify_graph boundary violations Made-with: Cursor
146 lines
6.0 KiB
TypeScript
146 lines
6.0 KiB
TypeScript
import type { AvailableNodesMetaData } from '@/app/components/workflow/hooks-store/store'
|
|
import type { CommonNodeType, NodeDefault, NodeDefaultBase } from '@/app/components/workflow/types'
|
|
import type { DocPathWithoutLang } from '@/types/doc-paths'
|
|
import { useCallback, useMemo } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { useStore as useAppStore } from '@/app/components/app/store'
|
|
import { useFeatures } from '@/app/components/base/features/hooks'
|
|
import { WORKFLOW_COMMON_NODES } from '@/app/components/workflow/constants/node'
|
|
import {
|
|
buildNodeSelectorAvailabilityContext,
|
|
filterNodesForSelector,
|
|
NodeSelectorSandboxMode,
|
|
NodeSelectorScene,
|
|
} from '@/app/components/workflow/constants/node-availability'
|
|
import AnswerDefault from '@/app/components/workflow/nodes/answer/default'
|
|
import EndDefault from '@/app/components/workflow/nodes/end/default'
|
|
import StartDefault from '@/app/components/workflow/nodes/start/default'
|
|
import TriggerPluginDefault from '@/app/components/workflow/nodes/trigger-plugin/default'
|
|
import TriggerScheduleDefault from '@/app/components/workflow/nodes/trigger-schedule/default'
|
|
import TriggerWebhookDefault from '@/app/components/workflow/nodes/trigger-webhook/default'
|
|
import { BlockEnum } from '@/app/components/workflow/types'
|
|
import { useDocLink } from '@/context/i18n'
|
|
import { useIsChatMode } from './use-is-chat-mode'
|
|
|
|
const NODE_HELP_LINK_OVERRIDES: Partial<Record<BlockEnum, string>> = {
|
|
[BlockEnum.FileUpload]: 'upload-file-to-sandbox',
|
|
}
|
|
|
|
type WorkflowAppScene = typeof NodeSelectorScene.Workflow | typeof NodeSelectorScene.Chatflow
|
|
|
|
const WORKFLOW_APP_SCENE_NODES = {
|
|
[NodeSelectorScene.Workflow]: [
|
|
EndDefault,
|
|
TriggerWebhookDefault,
|
|
TriggerScheduleDefault,
|
|
TriggerPluginDefault,
|
|
],
|
|
[NodeSelectorScene.Chatflow]: [AnswerDefault],
|
|
} as const
|
|
|
|
export const useAvailableNodesMetaData = () => {
|
|
const { t } = useTranslation()
|
|
const isChatMode = useIsChatMode()
|
|
const isSandboxFeatureEnabled = useFeatures(s => s.features.sandbox?.enabled) ?? false
|
|
const isSandboxRuntime = useAppStore(s => s.appDetail?.runtime_type === 'sandboxed')
|
|
const scene: WorkflowAppScene = isChatMode ? NodeSelectorScene.Chatflow : NodeSelectorScene.Workflow
|
|
const nodeAvailabilityContext = useMemo(() => buildNodeSelectorAvailabilityContext({
|
|
scene,
|
|
isSandboxRuntime,
|
|
isSandboxFeatureEnabled,
|
|
}), [scene, isSandboxFeatureEnabled, isSandboxRuntime])
|
|
const isSandboxed = nodeAvailabilityContext.sandboxMode === NodeSelectorSandboxMode.Enabled
|
|
const docLink = useDocLink()
|
|
|
|
const startNodeMetaData = useMemo(() => ({
|
|
...StartDefault,
|
|
metaData: {
|
|
...StartDefault.metaData,
|
|
isUndeletable: isChatMode, // start node is undeletable in chat mode, @use-nodes-interactions: handleNodeDelete function
|
|
isTypeFixed: isChatMode, // start node (user input) is immutable in chatflow mode
|
|
},
|
|
}), [isChatMode])
|
|
|
|
const availableWorkflowCommonNodes = useMemo(() => {
|
|
return filterNodesForSelector(WORKFLOW_COMMON_NODES, nodeAvailabilityContext)
|
|
}, [nodeAvailabilityContext])
|
|
|
|
const mergedNodesMetaData = useMemo(() => [
|
|
...availableWorkflowCommonNodes,
|
|
startNodeMetaData,
|
|
...WORKFLOW_APP_SCENE_NODES[scene],
|
|
] as AvailableNodesMetaData['nodes'], [availableWorkflowCommonNodes, scene, startNodeMetaData])
|
|
|
|
const getHelpLinkSlug = useCallback((nodeType: BlockEnum, helpLinkUri?: string) => {
|
|
if (isSandboxed && nodeType === BlockEnum.LLM)
|
|
return BlockEnum.Agent
|
|
|
|
return NODE_HELP_LINK_OVERRIDES[nodeType] || helpLinkUri || nodeType
|
|
}, [isSandboxed])
|
|
|
|
const availableNodesMetaData = useMemo<NodeDefaultBase[]>(() => {
|
|
const toNodeDefaultBase = (
|
|
node: NodeDefault<CommonNodeType>,
|
|
metaData: NodeDefaultBase['metaData'],
|
|
defaultValue: Partial<CommonNodeType>,
|
|
): NodeDefaultBase => {
|
|
return {
|
|
...node,
|
|
metaData,
|
|
defaultValue,
|
|
checkValid: (payload: CommonNodeType, translator, moreDataForCheckValid) => {
|
|
// normalize validator signature for shared metadata storage.
|
|
return node.checkValid(payload, translator, moreDataForCheckValid)
|
|
},
|
|
getOutputVars: node.getOutputVars
|
|
? (payload: CommonNodeType, allPluginInfoList, ragVariables, utils) => {
|
|
// normalize output var signature for shared metadata storage.
|
|
return node.getOutputVars!(payload, allPluginInfoList, ragVariables, utils)
|
|
}
|
|
: undefined,
|
|
}
|
|
}
|
|
|
|
return mergedNodesMetaData.map((node) => {
|
|
// normalize per-node defaults into a shared metadata shape.
|
|
const typedNode = node as NodeDefault<CommonNodeType>
|
|
const { metaData } = typedNode
|
|
const title = isSandboxed && metaData.type === BlockEnum.LLM
|
|
? t('blocks.agent', { ns: 'workflow' })
|
|
: t(`blocks.${metaData.type}` as const, { ns: 'workflow' })
|
|
const iconTypeOverride = isSandboxed && metaData.type === BlockEnum.LLM
|
|
? BlockEnum.Agent
|
|
: undefined
|
|
const description = t(`blocksAbout.${metaData.type}`, { ns: 'workflow' })
|
|
const helpLinkPath = `/use-dify/nodes/${getHelpLinkSlug(metaData.type, metaData.helpLinkUri)}` as DocPathWithoutLang
|
|
return toNodeDefaultBase(typedNode, {
|
|
...metaData,
|
|
iconType: iconTypeOverride,
|
|
title,
|
|
description,
|
|
helpLinkUri: docLink(helpLinkPath),
|
|
}, {
|
|
...typedNode.defaultValue,
|
|
type: metaData.type,
|
|
title,
|
|
_iconTypeOverride: iconTypeOverride,
|
|
})
|
|
})
|
|
}, [mergedNodesMetaData, t, docLink, isSandboxed, getHelpLinkSlug])
|
|
|
|
const availableNodesMetaDataMap = useMemo(() => availableNodesMetaData.reduce((acc, node) => {
|
|
acc![node.metaData.type] = node
|
|
return acc
|
|
}, {} as AvailableNodesMetaData['nodesMap']), [availableNodesMetaData])
|
|
|
|
return useMemo(() => {
|
|
return {
|
|
nodes: availableNodesMetaData,
|
|
nodesMap: {
|
|
...availableNodesMetaDataMap,
|
|
[BlockEnum.VariableAssigner]: availableNodesMetaDataMap?.[BlockEnum.VariableAggregator],
|
|
},
|
|
}
|
|
}, [availableNodesMetaData, availableNodesMetaDataMap])
|
|
}
|