mirror of
https://github.com/langgenius/dify.git
synced 2026-06-23 12:31:13 +08:00
Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: fatelei <fatelei@gmail.com> Co-authored-by: Copilot <198982749+Copilot@users.noreply.github.com> Co-authored-by: autofix-ci[bot] <114827586+autofix-ci[bot]@users.noreply.github.com> Co-authored-by: 盐粒 Yanli <yanli@dify.ai> Co-authored-by: Charles Yao <chongbinyao33@gmail.com> Co-authored-by: Claude Opus 4.8 <noreply@anthropic.com> Co-authored-by: yunlu.wen <yunlu.wen@dify.ai> Co-authored-by: yyh <92089059+lyzno1@users.noreply.github.com> Co-authored-by: Jingyi <jingyi.qi@dify.ai> Co-authored-by: yyh <yuanyouhuilyz@gmail.com> Co-authored-by: Joel <iamjoel007@gmail.com> Co-authored-by: hjlarry <hjlarry@163.com> Co-authored-by: Asuka Minato <i@asukaminato.eu.org> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Xiyuan Chen <52963600+GareArc@users.noreply.github.com> Co-authored-by: gigglewang <gigglewang@dify.ai> Co-authored-by: chariri <w@chariri.moe> Co-authored-by: Evan <2869018789@qq.com> Co-authored-by: zyssyz123 <916125788@qq.com>
221 lines
7.7 KiB
TypeScript
221 lines
7.7 KiB
TypeScript
import type { Edge, Node } from '@/app/components/workflow/types'
|
|
import type { FileUploadConfigResponse } from '@/models/common'
|
|
import type { FetchWorkflowDraftResponse } from '@/types/workflow'
|
|
import {
|
|
useCallback,
|
|
useEffect,
|
|
useMemo,
|
|
useState,
|
|
} from 'react'
|
|
import { useStore as useAppStore } from '@/app/components/app/store'
|
|
import {
|
|
useStore,
|
|
useWorkflowStore,
|
|
} from '@/app/components/workflow/store'
|
|
import { BlockEnum } from '@/app/components/workflow/types'
|
|
import { useSelector as useAppContextWithSelector } from '@/context/app-context'
|
|
import { useWorkflowConfig } from '@/service/use-workflow'
|
|
import {
|
|
fetchNodesDefaultConfigs,
|
|
fetchPublishedWorkflow,
|
|
fetchWorkflowDraft,
|
|
syncWorkflowDraft,
|
|
} from '@/service/workflow'
|
|
import { AppModeEnum } from '@/types/app'
|
|
import { getAppACLCapabilities } from '@/utils/permission'
|
|
import { useWorkflowDraftGraphForCanvas } from './use-workflow-draft-graph-for-canvas'
|
|
import { useWorkflowTemplate } from './use-workflow-template'
|
|
|
|
const emptyAccount = {
|
|
id: '',
|
|
name: '',
|
|
email: '',
|
|
}
|
|
|
|
const createLocalWorkflowDraft = (
|
|
graph: FetchWorkflowDraftResponse['graph'],
|
|
): FetchWorkflowDraftResponse => ({
|
|
id: '',
|
|
graph,
|
|
features: {
|
|
retriever_resource: { enabled: true },
|
|
},
|
|
created_at: 0,
|
|
created_by: emptyAccount,
|
|
hash: '',
|
|
updated_at: 0,
|
|
updated_by: emptyAccount,
|
|
tool_published: false,
|
|
environment_variables: [],
|
|
conversation_variables: [],
|
|
version: '',
|
|
marked_name: '',
|
|
marked_comment: '',
|
|
})
|
|
|
|
const hasConnectedUserInput = (nodes: Node[] = [], edges: Edge[] = []): boolean => {
|
|
const startNodeIds = nodes
|
|
.filter(node => node?.data?.type === BlockEnum.Start)
|
|
.map(node => node.id)
|
|
|
|
if (!startNodeIds.length)
|
|
return false
|
|
|
|
return edges.some(edge => startNodeIds.includes(edge.source))
|
|
}
|
|
|
|
export const useWorkflowInit = () => {
|
|
const workflowStore = useWorkflowStore()
|
|
const {
|
|
nodes: nodesTemplate,
|
|
edges: edgesTemplate,
|
|
} = useWorkflowTemplate()
|
|
const appDetail = useAppStore(state => state.appDetail)!
|
|
const currentUserId = useAppContextWithSelector(state => state.userProfile?.id)
|
|
const workspacePermissionKeys = useAppContextWithSelector(state => state.workspacePermissionKeys)
|
|
const appACLCapabilities = useMemo(() => getAppACLCapabilities(appDetail.permission_keys, {
|
|
currentUserId,
|
|
resourceMaintainer: appDetail.maintainer,
|
|
workspacePermissionKeys,
|
|
}), [appDetail.maintainer, appDetail.permission_keys, currentUserId, workspacePermissionKeys])
|
|
const { getWorkflowDraftGraphForCanvas } = useWorkflowDraftGraphForCanvas(appDetail.mode)
|
|
const setSyncWorkflowDraftHash = useStore(s => s.setSyncWorkflowDraftHash)
|
|
const [data, setData] = useState<FetchWorkflowDraftResponse>()
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
useEffect(() => {
|
|
workflowStore.setState({ appId: appDetail.id, appName: appDetail.name })
|
|
}, [appDetail.id, workflowStore])
|
|
|
|
const handleUpdateWorkflowFileUploadConfig = useCallback((config: FileUploadConfigResponse) => {
|
|
const { setFileUploadConfig } = workflowStore.getState()
|
|
setFileUploadConfig(config)
|
|
}, [workflowStore])
|
|
const {
|
|
data: fileUploadConfigResponse,
|
|
isLoading: isFileUploadConfigLoading,
|
|
} = useWorkflowConfig('/files/upload', handleUpdateWorkflowFileUploadConfig)
|
|
|
|
const handleGetInitialWorkflowData = useCallback(async () => {
|
|
try {
|
|
const res = await fetchWorkflowDraft(`/apps/${appDetail.id}/workflows/draft`)
|
|
const initialData = {
|
|
...res,
|
|
graph: getWorkflowDraftGraphForCanvas(res.graph, {
|
|
localStartPlaceholderNodes: nodesTemplate,
|
|
}),
|
|
}
|
|
|
|
setData(initialData)
|
|
workflowStore.setState({
|
|
envSecrets: (initialData.environment_variables || []).filter(env => env.value_type === 'secret').reduce((acc, env) => {
|
|
acc[env.id] = env.value
|
|
return acc
|
|
}, {} as Record<string, string>),
|
|
environmentVariables: initialData.environment_variables?.map(env => env.value_type === 'secret' ? { ...env, value: '[__HIDDEN__]' } : env) || [],
|
|
conversationVariables: initialData.conversation_variables || [],
|
|
isWorkflowDataLoaded: true,
|
|
})
|
|
setSyncWorkflowDraftHash(initialData.hash)
|
|
setIsLoading(false)
|
|
}
|
|
catch (error: any) {
|
|
if (error && error.json && !error.bodyUsed && appDetail) {
|
|
error.json().then((err: any) => {
|
|
if (err.code === 'draft_workflow_not_exist') {
|
|
const isAdvancedChat = appDetail.mode === AppModeEnum.ADVANCED_CHAT
|
|
const initialGraph = {
|
|
nodes: isAdvancedChat ? nodesTemplate : [],
|
|
edges: isAdvancedChat ? edgesTemplate : [],
|
|
}
|
|
workflowStore.setState({
|
|
notInitialWorkflow: true,
|
|
showOnboarding: false,
|
|
shouldAutoOpenStartNodeSelector: false,
|
|
hasSelectedStartNode: false,
|
|
hasShownOnboarding: !isAdvancedChat,
|
|
})
|
|
|
|
if (!appACLCapabilities.canEdit) {
|
|
const initialData = createLocalWorkflowDraft({
|
|
...getWorkflowDraftGraphForCanvas(initialGraph, {
|
|
localStartPlaceholderNodes: nodesTemplate,
|
|
}),
|
|
})
|
|
setData(initialData)
|
|
workflowStore.setState({
|
|
envSecrets: {},
|
|
environmentVariables: [],
|
|
conversationVariables: [],
|
|
isWorkflowDataLoaded: true,
|
|
})
|
|
setSyncWorkflowDraftHash(initialData.hash)
|
|
setIsLoading(false)
|
|
return
|
|
}
|
|
|
|
syncWorkflowDraft({
|
|
url: `/apps/${appDetail.id}/workflows/draft`,
|
|
params: {
|
|
graph: initialGraph,
|
|
features: {
|
|
retriever_resource: { enabled: true },
|
|
},
|
|
environment_variables: [],
|
|
conversation_variables: [],
|
|
},
|
|
}).then((res) => {
|
|
workflowStore.getState().setDraftUpdatedAt(res.updated_at)
|
|
setSyncWorkflowDraftHash(res.hash)
|
|
handleGetInitialWorkflowData()
|
|
})
|
|
}
|
|
})
|
|
}
|
|
}
|
|
}, [appACLCapabilities.canEdit, appDetail, getWorkflowDraftGraphForCanvas, nodesTemplate, edgesTemplate, workflowStore, setSyncWorkflowDraftHash])
|
|
|
|
useEffect(() => {
|
|
handleGetInitialWorkflowData()
|
|
}, [])
|
|
|
|
const handleFetchPreloadData = useCallback(async () => {
|
|
try {
|
|
const nodesDefaultConfigsData = await fetchNodesDefaultConfigs(`/apps/${appDetail?.id}/workflows/default-workflow-block-configs`)
|
|
const publishedWorkflow = await fetchPublishedWorkflow(`/apps/${appDetail?.id}/workflows/publish`)
|
|
workflowStore.setState({
|
|
nodesDefaultConfigs: nodesDefaultConfigsData.reduce((acc, block) => {
|
|
if (!acc[block.type])
|
|
acc[block.type] = { ...block.config }
|
|
return acc
|
|
}, {} as Record<string, any>),
|
|
})
|
|
workflowStore.getState().setPublishedAt(publishedWorkflow?.created_at ?? 0)
|
|
const graph = publishedWorkflow?.graph
|
|
workflowStore.getState().setLastPublishedHasUserInput(
|
|
hasConnectedUserInput(graph?.nodes, graph?.edges),
|
|
)
|
|
}
|
|
catch (e) {
|
|
console.error(e)
|
|
workflowStore.getState().setLastPublishedHasUserInput(false)
|
|
}
|
|
}, [workflowStore, appDetail])
|
|
|
|
useEffect(() => {
|
|
handleFetchPreloadData()
|
|
}, [handleFetchPreloadData])
|
|
|
|
useEffect(() => {
|
|
if (data) {
|
|
workflowStore.getState().setDraftUpdatedAt(data.updated_at)
|
|
workflowStore.getState().setToolPublished(data.tool_published)
|
|
}
|
|
}, [data, workflowStore])
|
|
|
|
return {
|
|
data,
|
|
isLoading: isLoading || isFileUploadConfigLoading,
|
|
fileUploadConfigResponse,
|
|
}
|
|
}
|