Files
dify/web/app/components/workflow-app/hooks/use-available-nodes-meta-data.ts
Novice 499d237b7e fix: pass all CI quality checks - ESLint, TypeScript, basedpyright, pyrefly, lint-imports
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
2026-03-24 10:54:58 +08:00

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])
}