From 11417d8019db59190c96c1ff5264e25a732329c9 Mon Sep 17 00:00:00 2001 From: Jan Oberhauser Date: Wed, 7 Feb 2024 17:03:53 +0100 Subject: [PATCH 1/5] fix(core): Fix pairedItem issue with partial manual executions --- packages/cli/src/Interfaces.ts | 4 +-- packages/cli/src/WorkflowHelpers.ts | 4 +-- packages/cli/src/commands/execute.ts | 2 +- packages/cli/src/commands/executeBatch.ts | 2 +- .../cli/src/workflows/workflow.request.ts | 3 ++- .../workflows/workflowExecution.service.ts | 8 ++++-- packages/core/src/Interfaces.ts | 6 +++++ packages/core/src/WorkflowExecute.ts | 11 ++++---- packages/editor-ui/src/Interface.ts | 8 +++++- packages/editor-ui/src/mixins/workflowRun.ts | 25 ++++++++++++++++--- 10 files changed, 54 insertions(+), 19 deletions(-) diff --git a/packages/cli/src/Interfaces.ts b/packages/cli/src/Interfaces.ts index abcfa0bccb5a4..a1dbb30fd5493 100644 --- a/packages/cli/src/Interfaces.ts +++ b/packages/cli/src/Interfaces.ts @@ -27,7 +27,7 @@ import type { import type { ActiveWorkflowRunner } from '@/ActiveWorkflowRunner'; -import type { WorkflowExecute } from 'n8n-core'; +import type { StartNodeData, WorkflowExecute } from 'n8n-core'; import type PCancelable from 'p-cancelable'; @@ -532,7 +532,7 @@ export interface IWorkflowExecutionDataProcess { pinData?: IPinData; retryOf?: string; sessionId?: string; - startNodes?: string[]; + startNodes?: StartNodeData[]; workflowData: IWorkflowBase; userId: string; } diff --git a/packages/cli/src/WorkflowHelpers.ts b/packages/cli/src/WorkflowHelpers.ts index 9ea4e8335db10..f95498fb9d77b 100644 --- a/packages/cli/src/WorkflowHelpers.ts +++ b/packages/cli/src/WorkflowHelpers.ts @@ -227,9 +227,9 @@ export function getExecutionStartNode(data: IWorkflowExecutionDataProcess, workf let startNode; if ( data.startNodes?.length === 1 && - Object.keys(data.pinData ?? {}).includes(data.startNodes[0]) + Object.keys(data.pinData ?? {}).includes(data.startNodes[0].name) ) { - startNode = workflow.getNode(data.startNodes[0]) ?? undefined; + startNode = workflow.getNode(data.startNodes[0].name) ?? undefined; } return startNode; diff --git a/packages/cli/src/commands/execute.ts b/packages/cli/src/commands/execute.ts index a5c60a3e208ff..d3f0eb469d869 100644 --- a/packages/cli/src/commands/execute.ts +++ b/packages/cli/src/commands/execute.ts @@ -101,7 +101,7 @@ export class Execute extends BaseCommand { const user = await Container.get(OwnershipService).getInstanceOwner(); const runData: IWorkflowExecutionDataProcess = { executionMode: 'cli', - startNodes: [startingNode.name], + startNodes: [{ name: startingNode.name, sourceData: null }], workflowData, userId: user.id, }; diff --git a/packages/cli/src/commands/executeBatch.ts b/packages/cli/src/commands/executeBatch.ts index 483b4a3dc4abc..9d29ee4adc1ec 100644 --- a/packages/cli/src/commands/executeBatch.ts +++ b/packages/cli/src/commands/executeBatch.ts @@ -620,7 +620,7 @@ export class ExecuteBatch extends BaseCommand { const runData: IWorkflowExecutionDataProcess = { executionMode: 'cli', - startNodes: [startingNode.name], + startNodes: [{ name: startingNode.name, sourceData: null }], workflowData, userId: ExecuteBatch.instanceOwner.id, }; diff --git a/packages/cli/src/workflows/workflow.request.ts b/packages/cli/src/workflows/workflow.request.ts index e576f50f3625c..2bdf8764f31f0 100644 --- a/packages/cli/src/workflows/workflow.request.ts +++ b/packages/cli/src/workflows/workflow.request.ts @@ -1,5 +1,6 @@ import type { IWorkflowDb } from '@/Interfaces'; import type { AuthenticatedRequest } from '@/requests'; +import type { StartNodeData } from 'n8n-core'; import type { INode, IConnections, IWorkflowSettings, IRunData, IPinData } from 'n8n-workflow'; export declare namespace WorkflowRequest { @@ -19,7 +20,7 @@ export declare namespace WorkflowRequest { workflowData: IWorkflowDb; runData: IRunData; pinData: IPinData; - startNodes?: string[]; + startNodes?: StartNodeData[]; destinationNode?: string; }; diff --git a/packages/cli/src/workflows/workflowExecution.service.ts b/packages/cli/src/workflows/workflowExecution.service.ts index 913a3bbc44ace..65e947ee47df6 100644 --- a/packages/cli/src/workflows/workflowExecution.service.ts +++ b/packages/cli/src/workflows/workflowExecution.service.ts @@ -101,7 +101,11 @@ export class WorkflowExecutionService { user: User, sessionId?: string, ) { - const pinnedTrigger = this.selectPinnedActivatorStarter(workflowData, startNodes, pinData); + const pinnedTrigger = this.selectPinnedActivatorStarter( + workflowData, + startNodes?.map((nodeData) => nodeData.name), + pinData, + ); // If webhooks nodes exist and are active we have to wait for till we receive a call if ( @@ -143,7 +147,7 @@ export class WorkflowExecutionService { const hasRunData = (node: INode) => runData !== undefined && !!runData[node.name]; if (pinnedTrigger && !hasRunData(pinnedTrigger)) { - data.startNodes = [pinnedTrigger.name]; + data.startNodes = [{ name: pinnedTrigger.name, sourceData: null }]; } const executionId = await this.workflowRunner.run(data); diff --git a/packages/core/src/Interfaces.ts b/packages/core/src/Interfaces.ts index 000f2a8125b07..afaa63704f14f 100644 --- a/packages/core/src/Interfaces.ts +++ b/packages/core/src/Interfaces.ts @@ -1,5 +1,6 @@ import type { IPollResponse, + ISourceData, ITriggerResponse, IWorkflowSettings as IWorkflowSettingsWorkflow, ValidationResult, @@ -7,6 +8,11 @@ import type { export type Class = new (...args: A) => T; +export interface StartNodeData { + name: string; + sourceData: ISourceData | null; +} + export interface IResponseError extends Error { statusCode?: number; } diff --git a/packages/core/src/WorkflowExecute.ts b/packages/core/src/WorkflowExecute.ts index 542704e6fef91..29cc685d71cbf 100644 --- a/packages/core/src/WorkflowExecute.ts +++ b/packages/core/src/WorkflowExecute.ts @@ -43,6 +43,7 @@ import { } from 'n8n-workflow'; import get from 'lodash/get'; import * as NodeExecuteFunctions from './NodeExecuteFunctions'; +import type { StartNodeData } from './Interfaces'; export class WorkflowExecute { private status: ExecutionStatus = 'new'; @@ -157,7 +158,7 @@ export class WorkflowExecute { runPartialWorkflow( workflow: Workflow, runData: IRunData, - startNodes: string[], + startNodes: StartNodeData[], destinationNode?: string, pinData?: IPinData, ): PCancelable { @@ -175,7 +176,7 @@ export class WorkflowExecute { const waitingExecution: IWaitingForExecution = {}; const waitingExecutionSource: IWaitingForExecutionSource = {}; for (const startNode of startNodes) { - incomingNodeConnections = workflow.connectionsByDestinationNode[startNode]; + incomingNodeConnections = workflow.connectionsByDestinationNode[startNode.name]; const incomingData: INodeExecutionData[][] = []; let incomingSourceData: ITaskDataConnectionsSource | null = null; @@ -210,15 +211,13 @@ export class WorkflowExecute { } } - incomingSourceData.main.push({ - previousNode: connection.node, - }); + incomingSourceData.main.push(startNode.sourceData); } } } const executeData: IExecuteData = { - node: workflow.getNode(startNode) as INode, + node: workflow.getNode(startNode.name) as INode, data: { main: incomingData, }, diff --git a/packages/editor-ui/src/Interface.ts b/packages/editor-ui/src/Interface.ts index 6fd65e1cf7a2f..f2b8783b6be6f 100644 --- a/packages/editor-ui/src/Interface.ts +++ b/packages/editor-ui/src/Interface.ts @@ -47,6 +47,7 @@ import { type INodeProperties, type NodeConnectionType, type INodeCredentialsDetails, + type ISourceData, } from 'n8n-workflow'; import type { BulkCommand, Undoable } from '@/models/history'; import type { PartialBy, TupleToUnion } from '@/utils/typeHelpers'; @@ -189,9 +190,14 @@ export interface IAiData { runIndex: number; } +export interface StartNodeData { + name: string; + sourceData: ISourceData | null; +} + export interface IStartRunData { workflowData: IWorkflowData; - startNodes?: string[]; + startNodes?: StartNodeData[]; destinationNode?: string; runData?: IRunData; pinData?: IPinData; diff --git a/packages/editor-ui/src/mixins/workflowRun.ts b/packages/editor-ui/src/mixins/workflowRun.ts index 58842f6a137d2..63e6598a03fce 100644 --- a/packages/editor-ui/src/mixins/workflowRun.ts +++ b/packages/editor-ui/src/mixins/workflowRun.ts @@ -1,6 +1,12 @@ -import type { IExecutionPushResponse, IExecutionResponse, IStartRunData } from '@/Interface'; +import type { + IExecutionPushResponse, + IExecutionResponse, + IStartRunData, + StartNodeData, +} from '@/Interface'; import { mapStores } from 'pinia'; import { defineComponent } from 'vue'; +import { get } from 'lodash-es'; import type { IDataObject, @@ -236,11 +242,25 @@ export const workflowRun = defineComponent({ const workflowData = await this.getWorkflowDataToSave(); + const startNodesData: StartNodeData[] = startNodes.map((name) => { + // Find for each start node the source data + let sourceData = get(runData, [name, 0, 'source', 0], null); + if (sourceData === null) { + const parentNodes = workflow.getParentNodes(name, NodeConnectionType.Main, 1); + const executeData = this.executeData(parentNodes, name, NodeConnectionType.Main, 0); + sourceData = get(executeData, ['source', NodeConnectionType.Main, 0], null); + } + return { + name, + sourceData, + }; + }); + const startRunData: IStartRunData = { workflowData, runData: newRunData, pinData: workflowData.pinData, - startNodes, + startNodes: startNodesData, }; if ('destinationNode' in options) { startRunData.destinationNode = options.destinationNode; @@ -261,7 +281,6 @@ export const workflowRun = defineComponent({ resultData: { runData: newRunData || {}, pinData: workflowData.pinData, - startNodes, workflowData, }, } as IRunExecutionData, From 9159ba89834e1d5f45de07e1a21a6f224dd4ed48 Mon Sep 17 00:00:00 2001 From: Danny Martini Date: Thu, 22 Feb 2024 15:25:31 +0100 Subject: [PATCH 2/5] add test --- cypress/e2e/19-execution.cy.ts | 23 +++ ...flow_pairedItem_incomplete_manual_bug.json | 160 ++++++++++++++++++ cypress/pages/workflow.ts | 5 +- 3 files changed, 187 insertions(+), 1 deletion(-) create mode 100644 cypress/fixtures/Test_Workflow_pairedItem_incomplete_manual_bug.json diff --git a/cypress/e2e/19-execution.cy.ts b/cypress/e2e/19-execution.cy.ts index cc8a55e89ceb8..6bc3658fab45d 100644 --- a/cypress/e2e/19-execution.cy.ts +++ b/cypress/e2e/19-execution.cy.ts @@ -569,4 +569,27 @@ describe('Execution', () => { expect(interception.request.body.runData).to.include.all.keys(expectedRunDataKeys); }); }); + + it('should successful execute partial executions with nodes attached to the second output', () => { + cy.createFixtureWorkflow( + 'Test_Workflow_pairedItem_incomplete_manual_bug.json', + 'My test workflow', + ); + + cy.intercept('POST', '/rest/workflows/run').as('workflowRun'); + + workflowPage.getters.zoomToFitButton().click(); + workflowPage.getters.executeWorkflowButton().click(); + workflowPage.getters + .canvasNodeByName('Test Expression') + .findChildByTestId('execute-node-button') + .click({ force: true }); + + // Check toast (works because Cypress waits enough for the element to show after the http request node has finished) + // Wait for the execution to return. + cy.wait('@workflowRun'); + // Wait again for the websocket message to arrive and the UI to update. + cy.wait(100); + workflowPage.getters.errorToast({ timeout: 1 }).should('not.exist'); + }); }); diff --git a/cypress/fixtures/Test_Workflow_pairedItem_incomplete_manual_bug.json b/cypress/fixtures/Test_Workflow_pairedItem_incomplete_manual_bug.json new file mode 100644 index 0000000000000..f740bc1df63d1 --- /dev/null +++ b/cypress/fixtures/Test_Workflow_pairedItem_incomplete_manual_bug.json @@ -0,0 +1,160 @@ +{ + "name": "Test Workflow pairedItem incomplete manual bug", + "nodes": [ + { + "parameters": {}, + "id": "f26332f3-c61a-4843-94bd-64a73ad161ff", + "name": "When clicking \"Test workflow\"", + "type": "n8n-nodes-base.manualTrigger", + "typeVersion": 1, + "position": [ + 860, + 340 + ] + }, + { + "parameters": { + "assignments": { + "assignments": [ + { + "id": "bd522794-d056-48b8-9204-26f7d68288d9", + "name": "test", + "value": "a", + "type": "string" + } + ] + }, + "options": {} + }, + "id": "fae0c907-e2bf-4ecf-82be-f9caa209f925", + "name": "Init Data", + "type": "n8n-nodes-base.set", + "typeVersion": 3.3, + "position": [ + 1080, + 340 + ] + }, + { + "parameters": { + "conditions": { + "options": { + "caseSensitive": true, + "leftValue": "", + "typeValidation": "strict" + }, + "conditions": [ + { + "id": "8db21b4b-1675-4e63-b092-7fcc45a86547", + "leftValue": "={{ $json.test }}", + "rightValue": "b", + "operator": { + "type": "string", + "operation": "equals", + "name": "filter.operator.equals" + } + } + ], + "combinator": "and" + }, + "options": {} + }, + "id": "f7990edd-2c0f-42e6-b3ce-74c7df02b6a4", + "name": "If", + "type": "n8n-nodes-base.if", + "typeVersion": 2, + "position": [ + 1300, + 340 + ] + }, + { + "parameters": {}, + "id": "850d48f5-0689-4cab-b30c-30e179577c82", + "name": "NoOp1", + "type": "n8n-nodes-base.noOp", + "typeVersion": 1, + "position": [ + 1540, + 200 + ] + }, + { + "parameters": { + "assignments": { + "assignments": [ + { + "id": "bd522794-d056-48b8-9204-26f7d68288d9", + "name": "test2", + "value": "={{ $('Init Data').item.json.test }}", + "type": "string" + } + ] + }, + "options": {} + }, + "id": "91d93c3a-a557-465e-812b-266d6277b279", + "name": "Test Expression", + "type": "n8n-nodes-base.set", + "typeVersion": 3.3, + "position": [ + 1540, + 440 + ] + } + ], + "pinData": {}, + "connections": { + "When clicking \"Test workflow\"": { + "main": [ + [ + { + "node": "Init Data", + "type": "main", + "index": 0 + } + ] + ] + }, + "Init Data": { + "main": [ + [ + { + "node": "If", + "type": "main", + "index": 0 + } + ] + ] + }, + "If": { + "main": [ + [ + { + "node": "NoOp1", + "type": "main", + "index": 0 + } + ], + [ + { + "node": "Test Expression", + "type": "main", + "index": 0 + } + ] + ] + } + }, + "active": false, + "settings": { + "executionOrder": "v1" + }, + "versionId": "765a6d9b-d667-4a59-9bd7-b0bc2627b008", + "meta": { + "templateCredsSetupCompleted": true, + "instanceId": "021d3c82ba2d3bc090cbf4fc81c9312668bcc34297e022bb3438c5c88a43a5ff" + }, + "id": "qnGQYw8TD58xs214", + "tags": [] +} diff --git a/cypress/pages/workflow.ts b/cypress/pages/workflow.ts index f3994b4b42f9b..5014cdbc0916e 100644 --- a/cypress/pages/workflow.ts +++ b/cypress/pages/workflow.ts @@ -3,6 +3,8 @@ import { BasePage } from './base'; import { getVisibleSelect } from '../utils'; import { NodeCreator } from './features/node-creator'; +type CyGetOptions = Parameters<(typeof cy)['get']>[1]; + const nodeCreator = new NodeCreator(); export class WorkflowPage extends BasePage { url = '/workflow/new'; @@ -48,7 +50,8 @@ export class WorkflowPage extends BasePage { }, successToast: () => cy.get('.el-notification:has(.el-notification--success)'), warningToast: () => cy.get('.el-notification:has(.el-notification--warning)'), - errorToast: () => cy.get('.el-notification:has(.el-notification--error)'), + errorToast: (options?: CyGetOptions) => + cy.get('.el-notification:has(.el-notification--error)', options), infoToast: () => cy.get('.el-notification:has(.el-notification--info)'), activatorSwitch: () => cy.getByTestId('workflow-activate-switch'), workflowMenu: () => cy.getByTestId('workflow-menu'), From 952f9f4c4a707dfe3025ef4b51db494a2516434c Mon Sep 17 00:00:00 2001 From: Danny Martini Date: Thu, 22 Feb 2024 16:59:34 +0100 Subject: [PATCH 3/5] fix test setup --- packages/cli/test/unit/WorkflowHelpers.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/cli/test/unit/WorkflowHelpers.test.ts b/packages/cli/test/unit/WorkflowHelpers.test.ts index 998e2b8b42809..1b5da0a4de1bc 100644 --- a/packages/cli/test/unit/WorkflowHelpers.test.ts +++ b/packages/cli/test/unit/WorkflowHelpers.test.ts @@ -25,7 +25,7 @@ describe('WorkflowHelpers', () => { node1: {}, node2: {}, }, - startNodes: ['node2'], + startNodes: [{ name: 'node2' }], } as unknown as IWorkflowExecutionDataProcess; const workflow = { getNode(nodeName: string) { From c059faebc8c0cf44c90f9eabbd46473a4e6ccdd9 Mon Sep 17 00:00:00 2001 From: Danny Martini Date: Fri, 23 Feb 2024 09:45:24 +0100 Subject: [PATCH 4/5] improve test name --- cypress/e2e/19-execution.cy.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cypress/e2e/19-execution.cy.ts b/cypress/e2e/19-execution.cy.ts index 6bc3658fab45d..73e2a897f6b8e 100644 --- a/cypress/e2e/19-execution.cy.ts +++ b/cypress/e2e/19-execution.cy.ts @@ -570,7 +570,7 @@ describe('Execution', () => { }); }); - it('should successful execute partial executions with nodes attached to the second output', () => { + it('should successfully execute partial executions with nodes attached to the second output', () => { cy.createFixtureWorkflow( 'Test_Workflow_pairedItem_incomplete_manual_bug.json', 'My test workflow', From a1b415917001f78ed2906ea29991d0d9dbe15d0a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E0=A4=95=E0=A4=BE=E0=A4=B0=E0=A4=A4=E0=A5=8B=E0=A4=AB?= =?UTF-8?q?=E0=A5=8D=E0=A4=AB=E0=A5=87=E0=A4=B2=E0=A4=B8=E0=A5=8D=E0=A4=95?= =?UTF-8?q?=E0=A5=8D=E0=A4=B0=E0=A4=BF=E0=A4=AA=E0=A5=8D=E0=A4=9F=E2=84=A2?= Date: Fri, 23 Feb 2024 11:17:24 +0100 Subject: [PATCH 5/5] move StartNodeData to n8n-workflow --- packages/cli/src/Interfaces.ts | 3 +- .../cli/src/workflows/workflow.request.ts | 10 +++++-- packages/core/src/Interfaces.ts | 6 ---- packages/core/src/WorkflowExecute.ts | 2 +- packages/editor-ui/src/Interface.ts | 7 +---- packages/editor-ui/src/mixins/workflowRun.ts | 28 ++++++++----------- packages/workflow/src/Interfaces.ts | 5 ++++ 7 files changed, 29 insertions(+), 32 deletions(-) diff --git a/packages/cli/src/Interfaces.ts b/packages/cli/src/Interfaces.ts index a1dbb30fd5493..6bb0d1b933fcb 100644 --- a/packages/cli/src/Interfaces.ts +++ b/packages/cli/src/Interfaces.ts @@ -23,11 +23,12 @@ import type { INodeProperties, IUserSettings, IHttpRequestMethods, + StartNodeData, } from 'n8n-workflow'; import type { ActiveWorkflowRunner } from '@/ActiveWorkflowRunner'; -import type { StartNodeData, WorkflowExecute } from 'n8n-core'; +import type { WorkflowExecute } from 'n8n-core'; import type PCancelable from 'p-cancelable'; diff --git a/packages/cli/src/workflows/workflow.request.ts b/packages/cli/src/workflows/workflow.request.ts index 2bdf8764f31f0..77d653a2a052c 100644 --- a/packages/cli/src/workflows/workflow.request.ts +++ b/packages/cli/src/workflows/workflow.request.ts @@ -1,7 +1,13 @@ import type { IWorkflowDb } from '@/Interfaces'; import type { AuthenticatedRequest } from '@/requests'; -import type { StartNodeData } from 'n8n-core'; -import type { INode, IConnections, IWorkflowSettings, IRunData, IPinData } from 'n8n-workflow'; +import type { + INode, + IConnections, + IWorkflowSettings, + IRunData, + IPinData, + StartNodeData, +} from 'n8n-workflow'; export declare namespace WorkflowRequest { type CreateUpdatePayload = Partial<{ diff --git a/packages/core/src/Interfaces.ts b/packages/core/src/Interfaces.ts index afaa63704f14f..000f2a8125b07 100644 --- a/packages/core/src/Interfaces.ts +++ b/packages/core/src/Interfaces.ts @@ -1,6 +1,5 @@ import type { IPollResponse, - ISourceData, ITriggerResponse, IWorkflowSettings as IWorkflowSettingsWorkflow, ValidationResult, @@ -8,11 +7,6 @@ import type { export type Class = new (...args: A) => T; -export interface StartNodeData { - name: string; - sourceData: ISourceData | null; -} - export interface IResponseError extends Error { statusCode?: number; } diff --git a/packages/core/src/WorkflowExecute.ts b/packages/core/src/WorkflowExecute.ts index 8c9e62d753ea8..8feae910e5167 100644 --- a/packages/core/src/WorkflowExecute.ts +++ b/packages/core/src/WorkflowExecute.ts @@ -33,6 +33,7 @@ import type { IWorkflowExecuteAdditionalData, WorkflowExecuteMode, CloseFunction, + StartNodeData, } from 'n8n-workflow'; import { LoggerProxy as Logger, @@ -43,7 +44,6 @@ import { } from 'n8n-workflow'; import get from 'lodash/get'; import * as NodeExecuteFunctions from './NodeExecuteFunctions'; -import type { StartNodeData } from './Interfaces'; export class WorkflowExecute { private status: ExecutionStatus = 'new'; diff --git a/packages/editor-ui/src/Interface.ts b/packages/editor-ui/src/Interface.ts index fec6c861ccd22..5f295a8d18154 100644 --- a/packages/editor-ui/src/Interface.ts +++ b/packages/editor-ui/src/Interface.ts @@ -47,7 +47,7 @@ import { type INodeProperties, type NodeConnectionType, type INodeCredentialsDetails, - type ISourceData, + type StartNodeData, } from 'n8n-workflow'; import type { BulkCommand, Undoable } from '@/models/history'; import type { PartialBy, TupleToUnion } from '@/utils/typeHelpers'; @@ -187,11 +187,6 @@ export interface IAiData { runIndex: number; } -export interface StartNodeData { - name: string; - sourceData: ISourceData | null; -} - export interface IStartRunData { workflowData: IWorkflowData; startNodes?: StartNodeData[]; diff --git a/packages/editor-ui/src/mixins/workflowRun.ts b/packages/editor-ui/src/mixins/workflowRun.ts index 9244f682358e1..c873b2d754f10 100644 --- a/packages/editor-ui/src/mixins/workflowRun.ts +++ b/packages/editor-ui/src/mixins/workflowRun.ts @@ -1,9 +1,4 @@ -import type { - IExecutionPushResponse, - IExecutionResponse, - IStartRunData, - StartNodeData, -} from '@/Interface'; +import type { IExecutionPushResponse, IExecutionResponse, IStartRunData } from '@/Interface'; import { mapStores } from 'pinia'; import { defineComponent } from 'vue'; import { get } from 'lodash-es'; @@ -16,6 +11,7 @@ import type { IPinData, IWorkflowBase, Workflow, + StartNodeData, } from 'n8n-workflow'; import { NodeHelpers, @@ -43,8 +39,8 @@ export const consolidateRunDataAndStartNodes = ( runData: IRunData | null, pinData: IPinData | undefined, workflow: Workflow, -): { runData: IRunData | undefined; startNodes: string[] } => { - const startNodes: string[] = []; +): { runData: IRunData | undefined; startNodeNames: string[] } => { + const startNodeNames: string[] = []; let newRunData: IRunData | undefined; if (runData !== null && Object.keys(runData).length !== 0) { @@ -65,7 +61,7 @@ export const consolidateRunDataAndStartNodes = ( // When we hit a node which has no data we stop and set it // as a start node the execution from and then go on with other // direct input nodes - startNodes.push(parentNode); + startNodeNames.push(parentNode); break; } if (runData[parentNode]) { @@ -81,7 +77,7 @@ export const consolidateRunDataAndStartNodes = ( } } - return { runData: newRunData, startNodes }; + return { runData: newRunData, startNodeNames }; }; export const workflowRun = defineComponent({ @@ -249,18 +245,18 @@ export const workflowRun = defineComponent({ workflow, ); - const { startNodes } = consolidatedData; + const { startNodeNames } = consolidatedData; let { runData: newRunData } = consolidatedData; let executedNode: string | undefined; if ( - startNodes.length === 0 && + startNodeNames.length === 0 && 'destinationNode' in options && options.destinationNode !== undefined ) { executedNode = options.destinationNode; - startNodes.push(options.destinationNode); + startNodeNames.push(options.destinationNode); } else if ('triggerNode' in options && 'nodeData' in options) { - startNodes.push( + startNodeNames.push( ...workflow.getChildNodes(options.triggerNode, NodeConnectionType.Main, 1), ); newRunData = { @@ -269,7 +265,7 @@ export const workflowRun = defineComponent({ executedNode = options.triggerNode; } - const startNodesData: StartNodeData[] = startNodes.map((name) => { + const startNodes: StartNodeData[] = startNodeNames.map((name) => { // Find for each start node the source data let sourceData = get(runData, [name, 0, 'source', 0], null); if (sourceData === null) { @@ -292,7 +288,7 @@ export const workflowRun = defineComponent({ workflowData, runData: newRunData, pinData: workflowData.pinData, - startNodes: startNodesData, + startNodes, }; if ('destinationNode' in options) { startRunData.destinationNode = options.destinationNode; diff --git a/packages/workflow/src/Interfaces.ts b/packages/workflow/src/Interfaces.ts index 674781669dd8a..132850b47ca43 100644 --- a/packages/workflow/src/Interfaces.ts +++ b/packages/workflow/src/Interfaces.ts @@ -1937,6 +1937,11 @@ export interface ISourceData { previousNodeRun?: number; // If undefined "0" gets used } +export interface StartNodeData { + name: string; + sourceData: ISourceData | null; +} + // The data for all the different kind of connections (like main) and all the indexes export interface ITaskDataConnections { // Key for each input type and because there can be multiple inputs of the same type it is an array