mirror of
https://github.com/langgenius/dify.git
synced 2026-03-31 10:58:48 +08:00
349 lines
12 KiB
TypeScript
349 lines
12 KiB
TypeScript
import type { ReactNode } from 'react'
|
|
import type { Features as FeaturesData } from '@/app/components/base/features/types'
|
|
import type { WorkflowProps } from '@/app/components/workflow'
|
|
import type { CollaborationUpdate } from '@/app/components/workflow/collaboration/types/collaboration'
|
|
import type { Shape as HooksStoreShape } from '@/app/components/workflow/hooks-store/store'
|
|
import type { Edge, Node } from '@/app/components/workflow/types'
|
|
import type { FetchWorkflowDraftResponse } from '@/types/workflow'
|
|
import {
|
|
useCallback,
|
|
useEffect,
|
|
useMemo,
|
|
useRef,
|
|
} from 'react'
|
|
import { useReactFlow } from 'reactflow'
|
|
import { useFeatures, useFeaturesStore } from '@/app/components/base/features/hooks'
|
|
import { FILE_EXTS } from '@/app/components/base/prompt-editor/constants'
|
|
import { WorkflowWithInnerContext } from '@/app/components/workflow'
|
|
import { collaborationManager, useCollaboration } from '@/app/components/workflow/collaboration'
|
|
import { useWorkflowUpdate } from '@/app/components/workflow/hooks/use-workflow-interactions'
|
|
import { MCPToolAvailabilityProvider } from '@/app/components/workflow/nodes/_base/components/mcp-tool-availability'
|
|
import { useStore, useWorkflowStore } from '@/app/components/workflow/store'
|
|
import { SupportUploadFileTypes } from '@/app/components/workflow/types'
|
|
import { fetchWorkflowDraft } from '@/service/workflow'
|
|
import {
|
|
useAvailableNodesMetaData,
|
|
useConfigsMap,
|
|
useDSL,
|
|
useGetRunAndTraceUrl,
|
|
useInspectVarsCrud,
|
|
useNodesSyncDraft,
|
|
useSetWorkflowVarsWithValue,
|
|
useWorkflowRefreshDraft,
|
|
useWorkflowRun,
|
|
useWorkflowStartRun,
|
|
} from '../hooks'
|
|
import WorkflowChildren from './workflow-children'
|
|
|
|
type WorkflowMainProps = Pick<WorkflowProps, 'nodes' | 'edges' | 'viewport'> & {
|
|
headerLeftSlot?: ReactNode
|
|
}
|
|
type WorkflowDataUpdatePayload = Pick<FetchWorkflowDraftResponse, 'features' | 'conversation_variables' | 'environment_variables'>
|
|
const WorkflowMain = ({
|
|
nodes,
|
|
edges,
|
|
viewport,
|
|
headerLeftSlot,
|
|
}: WorkflowMainProps) => {
|
|
const sandboxEnabled = useFeatures(state => state.features.sandbox?.enabled) ?? false
|
|
const featuresStore = useFeaturesStore()
|
|
const workflowStore = useWorkflowStore()
|
|
const appId = useStore(s => s.appId)
|
|
const containerRef = useRef<HTMLDivElement>(null)
|
|
const reactFlow = useReactFlow()
|
|
|
|
const reactFlowStore = useMemo(() => ({
|
|
getState: () => ({
|
|
getNodes: () => reactFlow.getNodes(),
|
|
setNodes: (nodesToSet: Node[]) => reactFlow.setNodes(nodesToSet),
|
|
getEdges: () => reactFlow.getEdges(),
|
|
setEdges: (edgesToSet: Edge[]) => reactFlow.setEdges(edgesToSet),
|
|
}),
|
|
}), [reactFlow])
|
|
const {
|
|
startCursorTracking,
|
|
stopCursorTracking,
|
|
onlineUsers,
|
|
cursors,
|
|
isConnected,
|
|
isEnabled: isCollaborationEnabled,
|
|
} = useCollaboration(appId || '', reactFlowStore)
|
|
const myUserId = useMemo(
|
|
() => (isCollaborationEnabled && isConnected ? 'current-user' : null),
|
|
[isCollaborationEnabled, isConnected],
|
|
)
|
|
|
|
const filteredCursors = Object.fromEntries(
|
|
Object.entries(cursors).filter(([userId]) => userId !== myUserId),
|
|
)
|
|
|
|
useEffect(() => {
|
|
if (!isCollaborationEnabled)
|
|
return
|
|
|
|
if (containerRef.current)
|
|
startCursorTracking(containerRef as React.RefObject<HTMLElement>, reactFlow)
|
|
|
|
return () => {
|
|
stopCursorTracking()
|
|
}
|
|
}, [startCursorTracking, stopCursorTracking, reactFlow, isCollaborationEnabled])
|
|
|
|
const handleWorkflowDataUpdate = useCallback((payload: WorkflowDataUpdatePayload) => {
|
|
const {
|
|
features,
|
|
conversation_variables,
|
|
environment_variables,
|
|
} = payload
|
|
if (features && featuresStore) {
|
|
const { setFeatures } = featuresStore.getState()
|
|
|
|
const transformedFeatures: FeaturesData = {
|
|
file: {
|
|
image: {
|
|
enabled: !!features.file_upload?.image?.enabled,
|
|
number_limits: features.file_upload?.image?.number_limits || 3,
|
|
transfer_methods: features.file_upload?.image?.transfer_methods || ['local_file', 'remote_url'],
|
|
},
|
|
enabled: !!(features.file_upload?.enabled || features.file_upload?.image?.enabled),
|
|
allowed_file_types: features.file_upload?.allowed_file_types || [SupportUploadFileTypes.image],
|
|
allowed_file_extensions: features.file_upload?.allowed_file_extensions || FILE_EXTS[SupportUploadFileTypes.image].map(ext => `.${ext}`),
|
|
allowed_file_upload_methods: features.file_upload?.allowed_file_upload_methods || features.file_upload?.image?.transfer_methods || ['local_file', 'remote_url'],
|
|
number_limits: features.file_upload?.number_limits || features.file_upload?.image?.number_limits || 3,
|
|
},
|
|
opening: {
|
|
enabled: !!features.opening_statement,
|
|
opening_statement: features.opening_statement,
|
|
suggested_questions: features.suggested_questions,
|
|
},
|
|
suggested: features.suggested_questions_after_answer || { enabled: false },
|
|
speech2text: features.speech_to_text || { enabled: false },
|
|
text2speech: features.text_to_speech || { enabled: false },
|
|
citation: features.retriever_resource || { enabled: false },
|
|
moderation: features.sensitive_word_avoidance || { enabled: false },
|
|
annotationReply: features.annotation_reply || { enabled: false },
|
|
sandbox: features.sandbox || { enabled: false },
|
|
}
|
|
|
|
setFeatures(transformedFeatures)
|
|
}
|
|
if (conversation_variables) {
|
|
const { setConversationVariables } = workflowStore.getState()
|
|
setConversationVariables(conversation_variables)
|
|
}
|
|
if (environment_variables) {
|
|
const { setEnvironmentVariables } = workflowStore.getState()
|
|
setEnvironmentVariables(environment_variables)
|
|
}
|
|
}, [featuresStore, workflowStore])
|
|
|
|
const {
|
|
doSyncWorkflowDraft,
|
|
syncWorkflowDraftWhenPageClose,
|
|
} = useNodesSyncDraft()
|
|
const { handleRefreshWorkflowDraft } = useWorkflowRefreshDraft()
|
|
const { handleUpdateWorkflowCanvas } = useWorkflowUpdate()
|
|
const {
|
|
handleBackupDraft,
|
|
handleLoadBackupDraft,
|
|
handleRestoreFromPublishedWorkflow,
|
|
handleRun,
|
|
handleStopRun,
|
|
} = useWorkflowRun()
|
|
|
|
useEffect(() => {
|
|
if (!appId || !isCollaborationEnabled)
|
|
return
|
|
|
|
const unsubscribe = collaborationManager.onVarsAndFeaturesUpdate(async (_update: CollaborationUpdate) => {
|
|
try {
|
|
const response = await fetchWorkflowDraft(`/apps/${appId}/workflows/draft`)
|
|
handleWorkflowDataUpdate(response)
|
|
}
|
|
catch (error) {
|
|
console.error('workflow vars and features update failed:', error)
|
|
}
|
|
})
|
|
|
|
return unsubscribe
|
|
}, [appId, handleWorkflowDataUpdate, isCollaborationEnabled])
|
|
|
|
// Listen for workflow updates from other users
|
|
useEffect(() => {
|
|
if (!appId || !isCollaborationEnabled)
|
|
return
|
|
|
|
const unsubscribe = collaborationManager.onWorkflowUpdate(async () => {
|
|
try {
|
|
const response = await fetchWorkflowDraft(`/apps/${appId}/workflows/draft`)
|
|
|
|
// Handle features, variables etc.
|
|
handleWorkflowDataUpdate(response)
|
|
|
|
// Update workflow canvas (nodes, edges, viewport)
|
|
if (response.graph) {
|
|
handleUpdateWorkflowCanvas({
|
|
nodes: response.graph.nodes || [],
|
|
edges: response.graph.edges || [],
|
|
viewport: response.graph.viewport || { x: 0, y: 0, zoom: 1 },
|
|
})
|
|
}
|
|
}
|
|
catch (error) {
|
|
console.error('Failed to fetch updated workflow:', error)
|
|
}
|
|
})
|
|
|
|
return unsubscribe
|
|
}, [appId, handleWorkflowDataUpdate, handleUpdateWorkflowCanvas, isCollaborationEnabled])
|
|
|
|
// Listen for sync requests from other users (only processed by leader)
|
|
useEffect(() => {
|
|
if (!appId || !isCollaborationEnabled)
|
|
return
|
|
|
|
const unsubscribe = collaborationManager.onSyncRequest(() => {
|
|
doSyncWorkflowDraft()
|
|
})
|
|
|
|
return unsubscribe
|
|
}, [appId, doSyncWorkflowDraft, isCollaborationEnabled])
|
|
const {
|
|
handleStartWorkflowRun,
|
|
handleWorkflowStartRunInChatflow,
|
|
handleWorkflowStartRunInWorkflow,
|
|
handleWorkflowTriggerScheduleRunInWorkflow,
|
|
handleWorkflowTriggerWebhookRunInWorkflow,
|
|
handleWorkflowTriggerPluginRunInWorkflow,
|
|
handleWorkflowRunAllTriggersInWorkflow,
|
|
} = useWorkflowStartRun()
|
|
const availableNodesMetaData = useAvailableNodesMetaData()
|
|
const { getWorkflowRunAndTraceUrl } = useGetRunAndTraceUrl()
|
|
const {
|
|
exportCheck,
|
|
handleExportDSL,
|
|
} = useDSL()
|
|
|
|
const configsMap = useConfigsMap()
|
|
|
|
const { fetchInspectVars } = useSetWorkflowVarsWithValue({
|
|
...configsMap,
|
|
})
|
|
const {
|
|
hasNodeInspectVars,
|
|
hasSetInspectVar,
|
|
fetchInspectVarValue,
|
|
editInspectVarValue,
|
|
renameInspectVarName,
|
|
appendNodeInspectVars,
|
|
deleteInspectVar,
|
|
deleteNodeInspectorVars,
|
|
deleteAllInspectorVars,
|
|
isInspectVarEdited,
|
|
resetToLastRunVar,
|
|
invalidateSysVarValues,
|
|
resetConversationVar,
|
|
invalidateConversationVarValues,
|
|
} = useInspectVarsCrud()
|
|
|
|
const hooksStore = useMemo<Partial<HooksStoreShape>>(() => {
|
|
return {
|
|
syncWorkflowDraftWhenPageClose,
|
|
doSyncWorkflowDraft,
|
|
handleRefreshWorkflowDraft,
|
|
handleBackupDraft,
|
|
handleLoadBackupDraft,
|
|
handleRestoreFromPublishedWorkflow,
|
|
handleRun,
|
|
handleStopRun,
|
|
handleStartWorkflowRun,
|
|
handleWorkflowStartRunInChatflow,
|
|
handleWorkflowStartRunInWorkflow,
|
|
handleWorkflowTriggerScheduleRunInWorkflow,
|
|
handleWorkflowTriggerWebhookRunInWorkflow,
|
|
handleWorkflowTriggerPluginRunInWorkflow,
|
|
handleWorkflowRunAllTriggersInWorkflow,
|
|
availableNodesMetaData,
|
|
getWorkflowRunAndTraceUrl,
|
|
exportCheck,
|
|
handleExportDSL,
|
|
fetchInspectVars,
|
|
hasNodeInspectVars,
|
|
hasSetInspectVar,
|
|
fetchInspectVarValue,
|
|
editInspectVarValue,
|
|
renameInspectVarName,
|
|
appendNodeInspectVars,
|
|
deleteInspectVar,
|
|
deleteNodeInspectorVars,
|
|
deleteAllInspectorVars,
|
|
isInspectVarEdited,
|
|
resetToLastRunVar,
|
|
invalidateSysVarValues,
|
|
resetConversationVar,
|
|
invalidateConversationVarValues,
|
|
configsMap,
|
|
}
|
|
}, [
|
|
syncWorkflowDraftWhenPageClose,
|
|
doSyncWorkflowDraft,
|
|
handleRefreshWorkflowDraft,
|
|
handleBackupDraft,
|
|
handleLoadBackupDraft,
|
|
handleRestoreFromPublishedWorkflow,
|
|
handleRun,
|
|
handleStopRun,
|
|
handleStartWorkflowRun,
|
|
handleWorkflowStartRunInChatflow,
|
|
handleWorkflowStartRunInWorkflow,
|
|
handleWorkflowTriggerScheduleRunInWorkflow,
|
|
handleWorkflowTriggerWebhookRunInWorkflow,
|
|
handleWorkflowTriggerPluginRunInWorkflow,
|
|
handleWorkflowRunAllTriggersInWorkflow,
|
|
availableNodesMetaData,
|
|
getWorkflowRunAndTraceUrl,
|
|
exportCheck,
|
|
handleExportDSL,
|
|
fetchInspectVars,
|
|
hasNodeInspectVars,
|
|
hasSetInspectVar,
|
|
fetchInspectVarValue,
|
|
editInspectVarValue,
|
|
renameInspectVarName,
|
|
appendNodeInspectVars,
|
|
deleteInspectVar,
|
|
deleteNodeInspectorVars,
|
|
deleteAllInspectorVars,
|
|
isInspectVarEdited,
|
|
resetToLastRunVar,
|
|
invalidateSysVarValues,
|
|
resetConversationVar,
|
|
invalidateConversationVarValues,
|
|
configsMap,
|
|
])
|
|
|
|
return (
|
|
<div
|
|
ref={containerRef}
|
|
className="relative h-full w-full"
|
|
>
|
|
<WorkflowWithInnerContext
|
|
nodes={nodes}
|
|
edges={edges}
|
|
viewport={viewport}
|
|
onWorkflowDataUpdate={handleWorkflowDataUpdate}
|
|
hooksStore={hooksStore}
|
|
cursors={filteredCursors}
|
|
myUserId={myUserId}
|
|
onlineUsers={onlineUsers}
|
|
>
|
|
<MCPToolAvailabilityProvider sandboxEnabled={sandboxEnabled}>
|
|
<WorkflowChildren headerLeftSlot={headerLeftSlot} />
|
|
</MCPToolAvailabilityProvider>
|
|
</WorkflowWithInnerContext>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default WorkflowMain
|