From 5f6f8a1bddfd76b586c08da821e8b59070f449fc Mon Sep 17 00:00:00 2001 From: Alex Grozav Date: Fri, 29 Nov 2024 12:02:16 +0200 Subject: [PATCH] fix(editor): Fix pin data showing up in production executions on new canvas (#11951) --- .../composables/useCanvasOperations.test.ts | 64 ++++++++++++++++++- .../src/composables/useCanvasOperations.ts | 28 ++++++++ packages/editor-ui/src/views/NodeView.v2.vue | 25 ++------ 3 files changed, 98 insertions(+), 19 deletions(-) diff --git a/packages/editor-ui/src/composables/useCanvasOperations.test.ts b/packages/editor-ui/src/composables/useCanvasOperations.test.ts index c64230aa829bd..6776dda3cdd29 100644 --- a/packages/editor-ui/src/composables/useCanvasOperations.test.ts +++ b/packages/editor-ui/src/composables/useCanvasOperations.test.ts @@ -5,12 +5,13 @@ import type { IWebhookDescription, Workflow, INodeConnections, + WorkflowExecuteMode, } from 'n8n-workflow'; import { NodeConnectionType, NodeHelpers } from 'n8n-workflow'; import { useCanvasOperations } from '@/composables/useCanvasOperations'; import type { CanvasConnection, CanvasNode } from '@/types'; import { CanvasConnectionMode } from '@/types'; -import type { ICredentialsResponse, INodeUi, IWorkflowDb } from '@/Interface'; +import type { ICredentialsResponse, IExecutionResponse, INodeUi, IWorkflowDb } from '@/Interface'; import { RemoveNodeCommand } from '@/models/history'; import { useWorkflowsStore } from '@/stores/workflows.store'; import { useUIStore } from '@/stores/ui.store'; @@ -2174,6 +2175,67 @@ describe('useCanvasOperations', () => { }); }); }); + + describe('openExecution', () => { + it('should initialize workspace and set execution data when execution is found', async () => { + const workflowsStore = mockedStore(useWorkflowsStore); + const uiStore = mockedStore(useUIStore); + const { openExecution } = useCanvasOperations({ router }); + + const executionId = '123'; + const executionData: IExecutionResponse = { + id: executionId, + finished: true, + status: 'success', + startedAt: new Date(), + createdAt: new Date(), + workflowData: createTestWorkflow(), + mode: 'manual' as WorkflowExecuteMode, + }; + + workflowsStore.getExecution.mockResolvedValue(executionData); + + const result = await openExecution(executionId); + + expect(workflowsStore.setWorkflowExecutionData).toHaveBeenCalledWith(executionData); + expect(uiStore.stateIsDirty).toBe(false); + expect(result).toEqual(executionData); + }); + + it('should throw error when execution data is undefined', async () => { + const workflowsStore = mockedStore(useWorkflowsStore); + const executionId = '123'; + const { openExecution } = useCanvasOperations({ router }); + + workflowsStore.getExecution.mockResolvedValue(undefined); + + await expect(openExecution(executionId)).rejects.toThrow( + `Execution with id "${executionId}" could not be found!`, + ); + }); + + it('should clear workflow pin data if execution mode is not manual', async () => { + const workflowsStore = mockedStore(useWorkflowsStore); + const { openExecution } = useCanvasOperations({ router }); + + const executionId = '123'; + const executionData: IExecutionResponse = { + id: executionId, + finished: true, + status: 'success', + startedAt: new Date(), + createdAt: new Date(), + workflowData: createTestWorkflow(), + mode: 'trigger' as WorkflowExecuteMode, + }; + + workflowsStore.getExecution.mockResolvedValue(executionData); + + await openExecution(executionId); + + expect(workflowsStore.setWorkflowPinData).toHaveBeenCalledWith({}); + }); + }); }); function buildImportNodes() { diff --git a/packages/editor-ui/src/composables/useCanvasOperations.ts b/packages/editor-ui/src/composables/useCanvasOperations.ts index b30bab1892da7..b8857ebd4526a 100644 --- a/packages/editor-ui/src/composables/useCanvasOperations.ts +++ b/packages/editor-ui/src/composables/useCanvasOperations.ts @@ -5,6 +5,7 @@ import type { AddedNodesAndConnections, + IExecutionResponse, INodeUi, ITag, IUsedCredential, @@ -1849,6 +1850,32 @@ export function useCanvasOperations({ router }: { router: ReturnType