/* eslint-disable @typescript-eslint/prefer-nullish-coalescing */ /* eslint-disable @typescript-eslint/no-unused-vars */ /* eslint-disable no-restricted-syntax */ /* eslint-disable no-param-reassign */ /* eslint-disable @typescript-eslint/no-this-alias */ /* eslint-disable @typescript-eslint/no-unsafe-return */ /* eslint-disable no-prototype-builtins */ /* eslint-disable @typescript-eslint/no-non-null-assertion */ import { DateTime, Duration, Interval, Settings } from 'luxon'; import * as jmespath from 'jmespath'; // eslint-disable-next-line import/no-cycle import { ExpressionError, IDataObject, IExecuteData, INodeExecutionData, INodeParameters, IPairedItemData, IRunExecutionData, ISourceData, ITaskData, IWorkflowDataProxyAdditionalKeys, IWorkflowDataProxyData, NodeHelpers, NodeParameterValue, Workflow, WorkflowExecuteMode, } from '.'; export class WorkflowDataProxy { private workflow: Workflow; private runExecutionData: IRunExecutionData | null; private defaultReturnRunIndex: number; private runIndex: number; private itemIndex: number; private activeNodeName: string; private connectionInputData: INodeExecutionData[]; private siblingParameters: INodeParameters; private mode: WorkflowExecuteMode; private selfData: IDataObject; private additionalKeys: IWorkflowDataProxyAdditionalKeys; private executeData: IExecuteData | undefined; private defaultTimezone: string; private timezone: string; constructor( workflow: Workflow, runExecutionData: IRunExecutionData | null, runIndex: number, itemIndex: number, activeNodeName: string, connectionInputData: INodeExecutionData[], siblingParameters: INodeParameters, mode: WorkflowExecuteMode, defaultTimezone: string, additionalKeys: IWorkflowDataProxyAdditionalKeys, executeData?: IExecuteData, defaultReturnRunIndex = -1, selfData = {}, ) { this.workflow = workflow; this.runExecutionData = runExecutionData; this.defaultReturnRunIndex = defaultReturnRunIndex; this.runIndex = runIndex; this.itemIndex = itemIndex; this.activeNodeName = activeNodeName; this.connectionInputData = connectionInputData; this.siblingParameters = siblingParameters; this.mode = mode; this.defaultTimezone = defaultTimezone; this.timezone = (this.workflow.settings.timezone as string) || this.defaultTimezone; this.selfData = selfData; this.additionalKeys = additionalKeys; this.executeData = executeData; Settings.defaultZone = this.timezone; } /** * Returns a proxy which allows to query context data of a given node * * @private * @param {string} nodeName The name of the node to get the context from * @returns * @memberof WorkflowDataProxy */ private nodeContextGetter(nodeName: string) { const that = this; const node = this.workflow.nodes[nodeName]; return new Proxy( {}, { ownKeys(target) { if (Reflect.ownKeys(target).length === 0) { // Target object did not get set yet Object.assign(target, NodeHelpers.getContext(that.runExecutionData!, 'node', node)); } return Reflect.ownKeys(target); }, getOwnPropertyDescriptor(k) { return { enumerable: true, configurable: true, }; }, get(target, name, receiver) { // eslint-disable-next-line no-param-reassign name = name.toString(); const contextData = NodeHelpers.getContext(that.runExecutionData!, 'node', node); if (!contextData.hasOwnProperty(name)) { // Parameter does not exist on node throw new Error(`Could not find parameter "${name}" on context of node "${nodeName}"`); } return contextData[name]; }, }, ); } private selfGetter() { const that = this; return new Proxy( {}, { ownKeys(target) { return Reflect.ownKeys(target); }, // eslint-disable-next-line @typescript-eslint/no-unused-vars get(target, name, receiver) { name = name.toString(); return that.selfData[name]; }, }, ); } /** * Returns a proxy which allows to query parameter data of a given node * * @private * @param {string} nodeName The name of the node to query data from * @returns * @memberof WorkflowDataGetter */ private nodeParameterGetter(nodeName: string) { const that = this; const node = this.workflow.nodes[nodeName]; return new Proxy(node.parameters, { ownKeys(target) { return Reflect.ownKeys(target); }, getOwnPropertyDescriptor(k) { return { enumerable: true, configurable: true, }; }, get(target, name, receiver) { name = name.toString(); let returnValue: | INodeParameters | NodeParameterValue | NodeParameterValue[] | INodeParameters[]; if (name[0] === '&') { const key = name.slice(1); if (!that.siblingParameters.hasOwnProperty(key)) { throw new Error(`Could not find sibling parameter "${key}" on node "${nodeName}"`); } returnValue = that.siblingParameters[key]; } else { if (!node.parameters.hasOwnProperty(name)) { // Parameter does not exist on node return undefined; } returnValue = node.parameters[name]; } if (typeof returnValue === 'string' && returnValue.charAt(0) === '=') { // The found value is an expression so resolve it return that.workflow.expression.getParameterValue( returnValue, that.runExecutionData, that.runIndex, that.itemIndex, that.activeNodeName, that.connectionInputData, that.mode, that.timezone, that.additionalKeys, that.executeData, ); } return returnValue; }, }); } /** * Returns the node ExecutionData * * @private * @param {string} nodeName The name of the node query data from * @param {boolean} [shortSyntax=false] If short syntax got used * @param {number} [outputIndex] The index of the output, if not given the first one gets used * @param {number} [runIndex] The index of the run, if not given the current one does get used * @returns {INodeExecutionData[]} * @memberof WorkflowDataProxy */ private getNodeExecutionData( nodeName: string, shortSyntax = false, outputIndex?: number, runIndex?: number, ): INodeExecutionData[] { const that = this; let executionData: INodeExecutionData[]; if (!shortSyntax) { // Long syntax got used to return data from node in path if (that.runExecutionData === null) { throw new ExpressionError(`Workflow did not run so do not have any execution-data.`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } if (!that.runExecutionData.resultData.runData.hasOwnProperty(nodeName)) { if (that.workflow.getNode(nodeName)) { throw new ExpressionError( `The node "${nodeName}" hasn't been executed yet, so you can't reference its output data`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }, ); } throw new ExpressionError(`No node called "${nodeName}" in this workflow`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } runIndex = runIndex === undefined ? that.defaultReturnRunIndex : runIndex; runIndex = runIndex === -1 ? that.runExecutionData.resultData.runData[nodeName].length - 1 : runIndex; if (that.runExecutionData.resultData.runData[nodeName].length <= runIndex) { throw new ExpressionError(`Run ${runIndex} of node "${nodeName}" not found`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } const taskData = that.runExecutionData.resultData.runData[nodeName][runIndex].data!; if (taskData.main === null || !taskData.main.length || taskData.main[0] === null) { // throw new Error(`No data found for item-index: "${itemIndex}"`); throw new ExpressionError(`No data found from "main" input.`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } // Check from which output to read the data. // Depends on how the nodes are connected. // (example "IF" node. If node is connected to "true" or to "false" output) if (outputIndex === undefined) { const nodeConnection = that.workflow.getNodeConnectionIndexes( that.activeNodeName, nodeName, 'main', ); if (nodeConnection === undefined) { throw new ExpressionError( `The node "${that.activeNodeName}" is not connected with node "${nodeName}" so no data can get returned from it.`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }, ); } outputIndex = nodeConnection.sourceIndex; } if (outputIndex === undefined) { outputIndex = 0; } if (taskData.main.length <= outputIndex) { throw new ExpressionError(`Node "${nodeName}" has no branch with index ${outputIndex}.`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } executionData = taskData.main[outputIndex] as INodeExecutionData[]; } else { // Short syntax got used to return data from active node // TODO: Here have to generate connection Input data for the current node by itself // Data needed: // #- the run-index // - node which did send data (has to be the one from last recent execution) // - later also the name of the input and its index (currently not needed as it is always "main" and index "0") executionData = that.connectionInputData; } return executionData; } /** * Returns a proxy which allows to query data of a given node * * @private * @param {string} nodeName The name of the node query data from * @param {boolean} [shortSyntax=false] If short syntax got used * @returns * @memberof WorkflowDataGetter */ private nodeDataGetter(nodeName: string, shortSyntax = false) { const that = this; const node = this.workflow.nodes[nodeName]; if (!node) { return undefined; } return new Proxy( { binary: undefined, data: undefined, json: undefined }, { get(target, name, receiver) { name = name.toString(); if (['binary', 'data', 'json'].includes(name)) { const executionData = that.getNodeExecutionData(nodeName, shortSyntax, undefined); if (executionData.length <= that.itemIndex) { throw new ExpressionError(`No data found for item-index: "${that.itemIndex}"`, { runIndex: that.runIndex, itemIndex: that.itemIndex, }); } if (['data', 'json'].includes(name)) { // JSON-Data return executionData[that.itemIndex].json; } if (name === 'binary') { // Binary-Data const returnData: IDataObject = {}; if (!executionData[that.itemIndex].binary) { return returnData; } const binaryKeyData = executionData[that.itemIndex].binary!; for (const keyName of Object.keys(binaryKeyData)) { returnData[keyName] = {}; const binaryData = binaryKeyData[keyName]; for (const propertyName in binaryData) { if (propertyName === 'data') { // Skip the data property // eslint-disable-next-line no-continue continue; } (returnData[keyName] as IDataObject)[propertyName] = binaryData[propertyName]; } } return returnData; } } else if (name === 'context') { return that.nodeContextGetter(nodeName); } else if (name === 'parameter') { // Get node parameter data return that.nodeParameterGetter(nodeName); } else if (name === 'runIndex') { if ( that.runExecutionData === null || !that.runExecutionData.resultData.runData[nodeName] ) { return -1; } return that.runExecutionData.resultData.runData[nodeName].length - 1; } return Reflect.get(target, name, receiver); }, }, ); } /** * Returns a proxy to query data from the environment * * @private * @returns * @memberof WorkflowDataGetter */ private envGetter() { const that = this; return new Proxy( {}, { get(target, name, receiver) { if (process.env.N8N_BLOCK_ENV_ACCESS_IN_NODE === 'true') { throw new ExpressionError('Environment variable access got disabled', { causeDetailed: 'If you need access please contact the administrator to remove the environment variable ‘N8N_BLOCK_ENV_ACCESS_IN_NODE‘', runIndex: that.runIndex, itemIndex: that.itemIndex, failExecution: true, }); } return process.env[name.toString()]; }, }, ); } /** * Returns a proxy to query data from the workflow * * @private * @returns * @memberof WorkflowDataProxy */ private workflowGetter() { const allowedValues = ['active', 'id', 'name']; const that = this; return new Proxy( {}, { ownKeys(target) { return allowedValues; }, getOwnPropertyDescriptor(k) { return { enumerable: true, configurable: true, }; }, get(target, name, receiver) { if (!allowedValues.includes(name.toString())) { throw new Error(`The key "${name.toString()}" is not supported!`); } return that.workflow[name as keyof typeof target]; }, }, ); } /** * Returns a proxy to query data of all nodes * * @private * @returns * @memberof WorkflowDataGetter */ private nodeGetter() { const that = this; return new Proxy( {}, { get(target, name, receiver) { return that.nodeDataGetter(name.toString()); }, }, ); } /** * Returns the data proxy object which allows to query data from current run * * @returns * @memberof WorkflowDataGetter */ getDataProxy(): IWorkflowDataProxyData { const that = this; const getNodeOutput = (nodeName?: string, branchIndex?: number, runIndex?: number) => { let executionData: INodeExecutionData[]; if (nodeName === undefined) { executionData = that.connectionInputData; } else { branchIndex = branchIndex || 0; runIndex = runIndex === undefined ? -1 : runIndex; executionData = that.getNodeExecutionData(nodeName, false, branchIndex, runIndex); } return executionData; }; // replacing proxies with the actual data. const jmespathWrapper = (data: IDataObject | IDataObject[], query: string) => { if (!Array.isArray(data) && typeof data === 'object') { return jmespath.search({ ...data }, query); } return jmespath.search(data, query); }; const createExpressionError = ( message: string, context?: { causeDetailed?: string; description?: string; descriptionTemplate?: string; messageTemplate?: string; }, nodeName?: string, ) => { if (nodeName) { const pinData = this.workflow.getPinDataOfNode(nodeName); if (pinData) { if (!context) { context = {}; } message = `‘${nodeName}‘ must be unpinned to execute`; context.description = `To fetch the data the expression needs, The node ‘${nodeName}’ needs to execute without being pinned. Unpin it`; context.description = `To fetch the data for the expression, you must unpin the node '${nodeName}' and execute the workflow again.`; context.descriptionTemplate = `To fetch the data for the expression under '%%PARAMETER%%', you must unpin the node '${nodeName}' and execute the workflow again.`; } } return new ExpressionError(message, { runIndex: that.runIndex, itemIndex: that.itemIndex, failExecution: true, ...context, }); }; const getPairedItem = ( destinationNodeName: string, incomingSourceData: ISourceData | null, pairedItem: IPairedItemData, ): INodeExecutionData | null => { let taskData: ITaskData; let sourceData: ISourceData | null = incomingSourceData; if (typeof pairedItem === 'number') { pairedItem = { item: pairedItem, }; } let nodeBeforeLast: string | undefined; while (sourceData !== null && destinationNodeName !== sourceData.previousNode) { taskData = that.runExecutionData!.resultData.runData[sourceData.previousNode][ sourceData?.previousNodeRun || 0 ]; const previousNodeOutput = sourceData.previousNodeOutput || 0; if (previousNodeOutput >= taskData.data!.main.length) { // `Could not resolve as the defined node-output is not valid on node '${sourceData.previousNode}'.` throw createExpressionError( 'Can’t get data for expression', { messageTemplate: 'Can’t get data for expression under ‘%%PARAMETER%%’', description: `Apologies, this is an internal error. See details for more information`, causeDetailed: 'Referencing a non-existent output on a node, problem with source data', }, nodeBeforeLast, ); } if (pairedItem.item >= taskData.data!.main[previousNodeOutput]!.length) { // `Could not resolve as the defined item index is not valid on node '${sourceData.previousNode}'. throw createExpressionError( 'Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `Item points to an item which does not exist`, causeDetailed: `The pairedItem data points to an item ‘${pairedItem.item}‘ which does not exist on node ‘${sourceData.previousNode}‘ (output node did probably supply a wrong one)`, }, nodeBeforeLast, ); } const itemPreviousNode: INodeExecutionData = taskData.data!.main[previousNodeOutput]![pairedItem.item]; if (itemPreviousNode.pairedItem === undefined) { // `Could not resolve, as pairedItem data is missing on node '${sourceData.previousNode}'.`, throw createExpressionError( 'Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `To fetch the data from other nodes that this expression needs, more information is needed from the node ‘${sourceData.previousNode}’`, causeDetailed: `Missing pairedItem data (node ‘${sourceData.previousNode}’ did probably not supply it)`, }, sourceData.previousNode, ); } if (Array.isArray(itemPreviousNode.pairedItem)) { // Item is based on multiple items so check all of them const results = itemPreviousNode.pairedItem // eslint-disable-next-line @typescript-eslint/no-loop-func .map((item) => { try { const itemInput = item.input || 0; if (itemInput >= taskData.source.length) { // `Could not resolve pairedItem as the defined node input '${itemInput}' does not exist on node '${sourceData!.previousNode}'.` // Actual error does not matter as it gets caught below and `null` will be returned throw new Error('Not found'); } return getPairedItem(destinationNodeName, taskData.source[itemInput], item); } catch (error) { // Means pairedItem could not be found return null; } }) .filter((result) => result !== null); if (results.length !== 1) { throw createExpressionError('Invalid expression', { messageTemplate: 'Invalid expression under ‘%%PARAMETER%%’', description: `The expression uses data in node ‘${destinationNodeName}’ but there is more than one matching item in that node`, }); } return results[0]; } // pairedItem is not an array if (typeof itemPreviousNode.pairedItem === 'number') { pairedItem = { item: itemPreviousNode.pairedItem, }; } else { pairedItem = itemPreviousNode.pairedItem; } const itemInput = pairedItem.input || 0; if (itemInput >= taskData.source.length) { if (taskData.source.length === 0) { // A trigger node got reached, so looks like that that item can not be resolved throw createExpressionError('Invalid expression', { messageTemplate: 'Invalid expression under ‘%%PARAMETER%%’', description: `The expression uses data in node ‘${destinationNodeName}’ but there is no path back to it. Please check this node is connected to node ‘${that.activeNodeName}’ (there can be other nodes in between).`, }); } // `Could not resolve pairedItem as the defined node input '${itemInput}' does not exist on node '${sourceData.previousNode}'.` throw createExpressionError( 'Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `Item points to a node input which does not exist`, causeDetailed: `The pairedItem data points to a node input ‘${itemInput}‘ which does not exist on node ‘${sourceData.previousNode}‘ (node did probably supply a wrong one)`, }, nodeBeforeLast, ); } nodeBeforeLast = sourceData.previousNode; sourceData = taskData.source[pairedItem.input || 0] || null; } if (sourceData === null) { // 'Could not resolve, probably no pairedItem exists.' throw createExpressionError( 'Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `Could not resolve, probably no pairedItem exists`, }, nodeBeforeLast, ); } taskData = that.runExecutionData!.resultData.runData[sourceData.previousNode][ sourceData?.previousNodeRun || 0 ]; const previousNodeOutput = sourceData.previousNodeOutput || 0; if (previousNodeOutput >= taskData.data!.main.length) { // `Could not resolve pairedItem as the node output '${previousNodeOutput}' does not exist on node '${sourceData.previousNode}'` throw createExpressionError('Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `Item points to a node output which does not exist`, causeDetailed: `The sourceData points to a node output ‘${previousNodeOutput}‘ which does not exist on node ‘${sourceData.previousNode}‘ (output node did probably supply a wrong one)`, }); } if (pairedItem.item >= taskData.data!.main[previousNodeOutput]!.length) { // `Could not resolve pairedItem as the item with the index '${pairedItem.item}' does not exist on node '${sourceData.previousNode}'.` throw createExpressionError( 'Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `Item points to an item which does not exist`, causeDetailed: `The pairedItem data points to an item ‘${pairedItem.item}‘ which does not exist on node ‘${sourceData.previousNode}‘ (output node did probably supply a wrong one)`, }, nodeBeforeLast, ); } return taskData.data!.main[previousNodeOutput]![pairedItem.item]; }; const base = { $: (nodeName: string) => { if (!nodeName) { throw new ExpressionError('When calling $(), please specify a node', { runIndex: that.runIndex, itemIndex: that.itemIndex, failExecution: true, }); } return new Proxy( {}, { get(target, property, receiver) { if (property === 'pairedItem') { return (itemIndex?: number) => { if (itemIndex === undefined) { itemIndex = that.itemIndex; } const executionData = that.connectionInputData; // As we operate on the incoming item we can be sure that pairedItem is not an // array. After all can it only come from exactly one previous node via a certain // input. For that reason do we not have to consider the array case. const pairedItem = executionData[itemIndex].pairedItem as IPairedItemData; if (pairedItem === undefined) { throw new ExpressionError('Can’t get data for expression', { messageTemplate: `Can’t get data for expression under ‘%%PARAMETER%%’`, description: `To fetch the data from other nodes that this expression needs, more information is needed from the node ‘${that.activeNodeName}‘`, causeDetailed: `Missing pairedItem data (node ‘${that.activeNodeName}‘ did probably not supply it)`, runIndex: that.runIndex, itemIndex, failExecution: true, }); } if (!that.executeData?.source) { throw new ExpressionError('Can’t get data for expression', { messageTemplate: 'Can’t get data for expression under ‘%%PARAMETER%%’', description: `Apologies, this is an internal error. See details for more information`, causeDetailed: `Missing sourceData (probably an internal error)`, runIndex: that.runIndex, itemIndex, failExecution: true, }); } // Before resolving the pairedItem make sure that the requested node comes in the // graph before the current one const parentNodes = that.workflow.getParentNodes(that.activeNodeName); if (!parentNodes.includes(nodeName)) { throw new ExpressionError('Invalid expression', { messageTemplate: 'Invalid expression under ‘%%PARAMETER%%’', description: `The expression uses data in node ‘${nodeName}’ but there is no path back to it. Please check this node is connected to node ‘${that.activeNodeName}’ (there can be other nodes in between).`, runIndex: that.runIndex, itemIndex, failExecution: true, }); } const sourceData: ISourceData = that.executeData?.source.main![ pairedItem.input || 0 ] as ISourceData; return getPairedItem(nodeName, sourceData, pairedItem); }; } if (property === 'item') { return (itemIndex?: number, branchIndex?: number, runIndex?: number) => { if (itemIndex === undefined) { itemIndex = that.itemIndex; branchIndex = 0; runIndex = that.runIndex; } const executionData = getNodeOutput(nodeName, branchIndex, runIndex); if (executionData[itemIndex]) { return executionData[itemIndex]; } let errorMessage = ''; if (branchIndex === undefined && runIndex === undefined) { errorMessage = ` No item found at index ${itemIndex} (for node "${nodeName}")`; throw new Error(errorMessage); } if (branchIndex === undefined) { errorMessage = ` No item found at index ${itemIndex} in run ${runIndex || that.runIndex} (for node "${nodeName}")`; throw new Error(errorMessage); } if (runIndex === undefined) { errorMessage = ` No item found at index ${itemIndex} of branch ${branchIndex || 0} (for node "${nodeName}")`; throw new Error(errorMessage); } errorMessage = ` No item found at index ${itemIndex} of branch ${branchIndex || 0} in run ${runIndex || that.runIndex} (for node "${nodeName}")`; throw new Error(errorMessage); }; } if (property === 'first') { return (branchIndex?: number, runIndex?: number) => { const executionData = getNodeOutput(nodeName, branchIndex, runIndex); if (executionData[0]) return executionData[0]; return undefined; }; } if (property === 'last') { return (branchIndex?: number, runIndex?: number) => { const executionData = getNodeOutput(nodeName, branchIndex, runIndex); if (!executionData.length) return undefined; if (executionData[executionData.length - 1]) { return executionData[executionData.length - 1]; } return undefined; }; } if (property === 'all') { return (branchIndex?: number, runIndex?: number) => getNodeOutput(nodeName, branchIndex, runIndex); } if (property === 'context') { return that.nodeContextGetter(nodeName); } if (property === 'params') { return that.workflow.getNode(nodeName)?.parameters; } return Reflect.get(target, property, receiver); }, }, ); }, $input: new Proxy( {}, { get(target, property, receiver) { if (property === 'thisItem') { return that.connectionInputData[that.itemIndex]; } if (property === 'item') { return (itemIndex?: number) => { if (itemIndex === undefined) itemIndex = that.itemIndex; const result = that.connectionInputData; if (result[itemIndex]) { return result[itemIndex]; } return undefined; }; } if (property === 'first') { return () => { const result = that.connectionInputData; if (result[0]) { return result[0]; } return undefined; }; } if (property === 'last') { return () => { const result = that.connectionInputData; if (result.length && result[result.length - 1]) { return result[result.length - 1]; } return undefined; }; } if (property === 'all') { return () => { const result = that.connectionInputData; if (result.length) { return result; } return []; }; } return Reflect.get(target, property, receiver); }, }, ), $thisItem: that.connectionInputData[that.itemIndex], $binary: {}, // Placeholder $data: {}, // Placeholder $env: this.envGetter(), $evaluateExpression: (expression: string, itemIndex?: number) => { itemIndex = itemIndex || that.itemIndex; return that.workflow.expression.getParameterValue( `=${expression}`, that.runExecutionData, that.runIndex, itemIndex, that.activeNodeName, that.connectionInputData, that.mode, that.timezone, that.additionalKeys, that.executeData, ); }, $item: (itemIndex: number, runIndex?: number) => { const defaultReturnRunIndex = runIndex === undefined ? -1 : runIndex; const dataProxy = new WorkflowDataProxy( this.workflow, this.runExecutionData, this.runIndex, itemIndex, this.activeNodeName, this.connectionInputData, that.siblingParameters, that.mode, that.defaultTimezone, that.additionalKeys, that.executeData, defaultReturnRunIndex, ); return dataProxy.getDataProxy(); }, $items: (nodeName?: string, outputIndex?: number, runIndex?: number) => { let executionData: INodeExecutionData[]; if (nodeName === undefined) { executionData = that.connectionInputData; } else { outputIndex = outputIndex || 0; runIndex = runIndex === undefined ? -1 : runIndex; executionData = that.getNodeExecutionData(nodeName, false, outputIndex, runIndex); } return executionData; }, $json: {}, // Placeholder $node: this.nodeGetter(), $self: this.selfGetter(), $parameter: this.nodeParameterGetter(this.activeNodeName), $position: this.itemIndex, $runIndex: this.runIndex, $mode: this.mode, $workflow: this.workflowGetter(), $thisRunIndex: this.runIndex, $thisItemIndex: this.itemIndex, $now: DateTime.now(), $today: DateTime.now().set({ hour: 0, minute: 0, second: 0, millisecond: 0 }), $jmespath: jmespathWrapper, // eslint-disable-next-line @typescript-eslint/naming-convention DateTime, // eslint-disable-next-line @typescript-eslint/naming-convention Interval, // eslint-disable-next-line @typescript-eslint/naming-convention Duration, ...that.additionalKeys, }; return new Proxy(base, { get(target, name, receiver) { if (['$data', '$json'].includes(name as string)) { return that.nodeDataGetter(that.activeNodeName, true)?.json; } if (name === '$binary') { return that.nodeDataGetter(that.activeNodeName, true)?.binary; } return Reflect.get(target, name, receiver); }, }); } }