/* eslint-disable @typescript-eslint/no-unsafe-argument */ /* eslint-disable @typescript-eslint/no-explicit-any */ /* eslint-disable @typescript-eslint/no-unused-vars */ /* eslint-disable @typescript-eslint/prefer-nullish-coalescing */ /* eslint-disable @typescript-eslint/naming-convention */ /* eslint-disable new-cap */ /* eslint-disable @typescript-eslint/no-unsafe-call */ /* eslint-disable @typescript-eslint/no-unsafe-assignment */ /* eslint-disable @typescript-eslint/no-unsafe-return */ /* eslint-disable @typescript-eslint/no-unsafe-member-access */ /* eslint-disable @typescript-eslint/explicit-module-boundary-types */ /* eslint-disable @typescript-eslint/no-non-null-assertion */ /* eslint-disable @typescript-eslint/no-shadow */ /* eslint-disable no-param-reassign */ import type { GenericValue, IAdditionalCredentialOptions, IAllExecuteFunctions, IBinaryData, IContextObject, ICredentialDataDecryptedObject, ICredentialsExpressionResolveValues, IDataObject, IExecuteResponsePromiseData, IExecuteWorkflowInfo, IHttpRequestOptions, IN8nHttpFullResponse, IN8nHttpResponse, INode, INodeCredentialDescription, INodeCredentialsDetails, INodeExecutionData, IOAuth2Options, IRunExecutionData, ISourceData, ITaskDataConnections, IWebhookData, IWebhookDescription, IWorkflowDataProxyAdditionalKeys, IWorkflowDataProxyData, IWorkflowExecuteAdditionalData, Workflow, WorkflowActivateMode, WorkflowExecuteMode, IExecuteData, IGetNodeParameterOptions, NodeParameterValueType, NodeExecutionWithMetadata, IPairedItemData, ICredentialTestFunctions, BinaryHelperFunctions, NodeHelperFunctions, RequestHelperFunctions, FunctionsBase, IExecuteFunctions, IExecuteSingleFunctions, IHookFunctions, ILoadOptionsFunctions, IPollFunctions, ITriggerFunctions, IWebhookFunctions, BinaryMetadata, FileSystemHelperFunctions, } from 'n8n-workflow'; import { createDeferredPromise, NodeApiError, NodeHelpers, NodeOperationError, WorkflowDataProxy, LoggerProxy as Logger, OAuth2GrantType, deepCopy, fileTypeFromMimeType, ExpressionError, } from 'n8n-workflow'; import pick from 'lodash.pick'; import { Agent } from 'https'; import { IncomingMessage } from 'http'; import { stringify } from 'qs'; import type { Token } from 'oauth-1.0a'; import clientOAuth1 from 'oauth-1.0a'; import clientOAuth2 from 'client-oauth2'; import crypto, { createHmac } from 'crypto'; import get from 'lodash.get'; import type { Request, Response } from 'express'; import FormData from 'form-data'; import path from 'path'; import type { OptionsWithUri, OptionsWithUrl, RequestCallback, RequiredUriUrl } from 'request'; import type { RequestPromiseOptions } from 'request-promise-native'; import requestPromise from 'request-promise-native'; import FileType from 'file-type'; import { lookup, extension } from 'mime-types'; import type { IncomingHttpHeaders } from 'http'; import type { AxiosError, AxiosPromise, AxiosProxyConfig, AxiosRequestConfig, AxiosResponse, Method, } from 'axios'; import axios from 'axios'; import url, { URL, URLSearchParams } from 'url'; import { Readable } from 'stream'; import { access as fsAccess } from 'fs/promises'; import { createReadStream } from 'fs'; import { BinaryDataManager } from './BinaryDataManager'; import type { IResponseError, IWorkflowSettings } from './Interfaces'; import { extractValue } from './ExtractValue'; import { getClientCredentialsToken } from './OAuth2Helper'; import { PLACEHOLDER_EMPTY_EXECUTION_ID } from './Constants'; import { binaryToBuffer } from './BinaryDataManager/utils'; import { getAllWorkflowExecutionMetadata, getWorkflowExecutionMetadata, setAllWorkflowExecutionMetadata, setWorkflowExecutionMetadata, } from './WorkflowExecutionMetadata'; axios.defaults.timeout = 300000; // Prevent axios from adding x-form-www-urlencoded headers by default axios.defaults.headers.post = {}; axios.defaults.headers.put = {}; axios.defaults.headers.patch = {}; axios.defaults.paramsSerializer = (params) => { if (params instanceof URLSearchParams) { return params.toString(); } return stringify(params, { arrayFormat: 'indices' }); }; const requestPromiseWithDefaults = requestPromise.defaults({ timeout: 300000, // 5 minutes }); const pushFormDataValue = (form: FormData, key: string, value: any) => { if (value?.hasOwnProperty('value') && value.hasOwnProperty('options')) { form.append(key, value.value, value.options); } else { form.append(key, value); } }; const createFormDataObject = (data: Record) => { const formData = new FormData(); const keys = Object.keys(data); keys.forEach((key) => { const formField = data[key]; if (formField instanceof Array) { formField.forEach((item) => { pushFormDataValue(formData, key, item); }); } else { pushFormDataValue(formData, key, formField); } }); return formData; }; function searchForHeader(headers: IDataObject, headerName: string) { if (headers === undefined) { return undefined; } const headerNames = Object.keys(headers); headerName = headerName.toLowerCase(); return headerNames.find((thisHeader) => thisHeader.toLowerCase() === headerName); } async function generateContentLengthHeader(formData: FormData, headers: IDataObject) { if (!formData?.getLength) { return; } try { const length = await new Promise((res, rej) => { formData.getLength((error: Error | null, length: number) => { if (error) { rej(error); return; } res(length); }); }); headers = Object.assign(headers, { 'content-length': length, }); } catch (error) { Logger.error('Unable to calculate form data length', { error }); } } async function parseRequestObject(requestObject: IDataObject) { // This function is a temporary implementation // That translates all http requests done via // the request library to axios directly // We are not using n8n's interface as it would // an unnecessary step, considering the `request` // helper can be deprecated and removed. const axiosConfig: AxiosRequestConfig = {}; if (requestObject.headers !== undefined) { axiosConfig.headers = requestObject.headers as string; } // Let's start parsing the hardest part, which is the request body. // The process here is as following? // - Check if we have a `content-type` header. If this was set, // we will follow // - Check if the `form` property was set. If yes, then it's x-www-form-urlencoded // - Check if the `formData` property exists. If yes, then it's multipart/form-data // - Lastly, we should have a regular `body` that is probably a JSON. const contentTypeHeaderKeyName = axiosConfig.headers && Object.keys(axiosConfig.headers).find( (headerName) => headerName.toLowerCase() === 'content-type', ); const contentType = contentTypeHeaderKeyName && (axiosConfig.headers[contentTypeHeaderKeyName] as string | undefined); if (contentType === 'application/x-www-form-urlencoded' && requestObject.formData === undefined) { // there are nodes incorrectly created, informing the content type header // and also using formData. Request lib takes precedence for the formData. // We will do the same. // Merge body and form properties. if (typeof requestObject.body === 'string') { axiosConfig.data = requestObject.body; } else { const allData = Object.assign(requestObject.body || {}, requestObject.form || {}) as Record< string, string >; if (requestObject.useQuerystring === true) { axiosConfig.data = stringify(allData, { arrayFormat: 'repeat' }); } else { axiosConfig.data = stringify(allData); } } } else if (contentType && contentType.includes('multipart/form-data') !== false) { if (requestObject.formData !== undefined && requestObject.formData instanceof FormData) { axiosConfig.data = requestObject.formData; } else { const allData: Partial = { ...(requestObject.body as object | undefined), ...(requestObject.formData as object | undefined), }; axiosConfig.data = createFormDataObject(allData); } // replace the existing header with a new one that // contains the boundary property. delete axiosConfig.headers[contentTypeHeaderKeyName]; const headers = axiosConfig.data.getHeaders(); axiosConfig.headers = Object.assign(axiosConfig.headers || {}, headers); await generateContentLengthHeader(axiosConfig.data, axiosConfig.headers); } else { // When using the `form` property it means the content should be x-www-form-urlencoded. if (requestObject.form !== undefined && requestObject.body === undefined) { // If we have only form axiosConfig.data = typeof requestObject.form === 'string' ? stringify(requestObject.form, { format: 'RFC3986' }) : stringify(requestObject.form).toString(); if (axiosConfig.headers !== undefined) { const headerName = searchForHeader(axiosConfig.headers, 'content-type'); if (headerName) { delete axiosConfig.headers[headerName]; } axiosConfig.headers['Content-Type'] = 'application/x-www-form-urlencoded'; } else { axiosConfig.headers = { 'Content-Type': 'application/x-www-form-urlencoded', }; } } else if (requestObject.formData !== undefined) { // remove any "content-type" that might exist. if (axiosConfig.headers !== undefined) { const headers = Object.keys(axiosConfig.headers); headers.forEach((header) => header.toLowerCase() === 'content-type' ? delete axiosConfig.headers[header] : null, ); } if (requestObject.formData instanceof FormData) { axiosConfig.data = requestObject.formData; } else { axiosConfig.data = createFormDataObject(requestObject.formData as Record); } // Mix in headers as FormData creates the boundary. const headers = axiosConfig.data.getHeaders(); axiosConfig.headers = Object.assign(axiosConfig.headers || {}, headers); await generateContentLengthHeader(axiosConfig.data, axiosConfig.headers); } else if (requestObject.body !== undefined) { // If we have body and possibly form if (requestObject.form !== undefined && requestObject.body) { // merge both objects when exist. requestObject.body = Object.assign(requestObject.body, requestObject.form); } axiosConfig.data = requestObject.body as FormData | GenericValue | GenericValue[]; } } if (requestObject.uri !== undefined) { axiosConfig.url = requestObject.uri?.toString() as string; } if (requestObject.url !== undefined) { axiosConfig.url = requestObject.url?.toString() as string; } if (requestObject.baseURL !== undefined) { axiosConfig.baseURL = requestObject.baseURL?.toString() as string; } if (requestObject.method !== undefined) { axiosConfig.method = requestObject.method as Method; } if (requestObject.qs !== undefined && Object.keys(requestObject.qs as object).length > 0) { axiosConfig.params = requestObject.qs as IDataObject; } function hasArrayFormatOptions( arg: IDataObject, ): arg is IDataObject & { qsStringifyOptions: { arrayFormat: 'repeat' | 'brackets' } } { if ( typeof arg.qsStringifyOptions === 'object' && arg.qsStringifyOptions !== null && !Array.isArray(arg.qsStringifyOptions) && 'arrayFormat' in arg.qsStringifyOptions ) { return true; } return false; } if ( requestObject.useQuerystring === true || (hasArrayFormatOptions(requestObject) && requestObject.qsStringifyOptions.arrayFormat === 'repeat') ) { axiosConfig.paramsSerializer = (params) => { return stringify(params, { arrayFormat: 'repeat' }); }; } else if (requestObject.useQuerystring === false) { axiosConfig.paramsSerializer = (params) => { return stringify(params, { arrayFormat: 'indices' }); }; } if ( hasArrayFormatOptions(requestObject) && requestObject.qsStringifyOptions.arrayFormat === 'brackets' ) { axiosConfig.paramsSerializer = (params) => { return stringify(params, { arrayFormat: 'brackets' }); }; } if (requestObject.auth !== undefined) { // Check support for sendImmediately if ((requestObject.auth as IDataObject).bearer !== undefined) { axiosConfig.headers = Object.assign(axiosConfig.headers || {}, { // eslint-disable-next-line @typescript-eslint/restrict-template-expressions Authorization: `Bearer ${(requestObject.auth as IDataObject).bearer}`, }); } else { const authObj = requestObject.auth as IDataObject; // Request accepts both user/username and pass/password axiosConfig.auth = { username: (authObj.user || authObj.username) as string, password: (authObj.password || authObj.pass) as string, }; } } // Only set header if we have a body, otherwise it may fail if (requestObject.json === true) { // Add application/json headers - do not set charset as it breaks a lot of stuff // only add if no other accept headers was sent. const acceptHeaderExists = axiosConfig.headers === undefined ? false : Object.keys(axiosConfig.headers) .map((headerKey) => headerKey.toLowerCase()) .includes('accept'); if (!acceptHeaderExists) { axiosConfig.headers = Object.assign(axiosConfig.headers || {}, { Accept: 'application/json', }); } } if (requestObject.json === false || requestObject.json === undefined) { // Prevent json parsing axiosConfig.transformResponse = (res) => res; } // Axios will follow redirects by default, so we simply tell it otherwise if needed. if ( requestObject.followRedirect === false && ((requestObject.method as string | undefined) || 'get').toLowerCase() === 'get' ) { axiosConfig.maxRedirects = 0; } if ( requestObject.followAllRedirects === false && ((requestObject.method as string | undefined) || 'get').toLowerCase() !== 'get' ) { axiosConfig.maxRedirects = 0; } if (requestObject.rejectUnauthorized === false) { axiosConfig.httpsAgent = new Agent({ rejectUnauthorized: false, }); } if (requestObject.timeout !== undefined) { axiosConfig.timeout = requestObject.timeout as number; } if (requestObject.proxy !== undefined) { // try our best to parse the url provided. if (typeof requestObject.proxy === 'string') { try { const url = new URL(requestObject.proxy); axiosConfig.proxy = { host: url.hostname, port: parseInt(url.port, 10), protocol: url.protocol, }; if (!url.port) { // Sets port to a default if not informed if (url.protocol === 'http') { axiosConfig.proxy.port = 80; } else if (url.protocol === 'https') { axiosConfig.proxy.port = 443; } } if (url.username || url.password) { axiosConfig.proxy.auth = { username: url.username, password: url.password, }; } } catch (error) { // Not a valid URL. We will try to simply parse stuff // such as user:pass@host:port without protocol (we'll assume http) if (requestObject.proxy.includes('@')) { const [userpass, hostport] = requestObject.proxy.split('@'); const [username, password] = userpass.split(':'); const [hostname, port] = hostport.split(':'); axiosConfig.proxy = { host: hostname, port: parseInt(port, 10), protocol: 'http', auth: { username, password, }, }; } else if (requestObject.proxy.includes(':')) { const [hostname, port] = requestObject.proxy.split(':'); axiosConfig.proxy = { host: hostname, port: parseInt(port, 10), protocol: 'http', }; } else { axiosConfig.proxy = { host: requestObject.proxy, port: 80, protocol: 'http', }; } } } else { axiosConfig.proxy = requestObject.proxy as AxiosProxyConfig; } } if (requestObject.useStream) { axiosConfig.responseType = 'stream'; } else if (requestObject.encoding === null) { // When downloading files, return an arrayBuffer. axiosConfig.responseType = 'arraybuffer'; } // If we don't set an accept header // Axios forces "application/json, text/plan, */*" // Which causes some nodes like NextCloud to break // as the service returns XML unless requested otherwise. const allHeaders = axiosConfig.headers ? Object.keys(axiosConfig.headers) : []; if (!allHeaders.some((headerKey) => headerKey.toLowerCase() === 'accept')) { axiosConfig.headers = Object.assign(axiosConfig.headers || {}, { accept: '*/*' }); } if ( requestObject.json !== false && axiosConfig.data !== undefined && axiosConfig.data !== '' && !(axiosConfig.data instanceof Buffer) && !allHeaders.some((headerKey) => headerKey.toLowerCase() === 'content-type') ) { // Use default header for application/json // If we don't specify this here, axios will add // application/json; charset=utf-8 // and this breaks a lot of stuff axiosConfig.headers = Object.assign(axiosConfig.headers || {}, { 'content-type': 'application/json', }); } if (requestObject.simple === false) { axiosConfig.validateStatus = () => true; } /** * Missing properties: * encoding (need testing) * gzip (ignored - default already works) * resolveWithFullResponse (implemented elsewhere) */ return axiosConfig; } function digestAuthAxiosConfig( axiosConfig: AxiosRequestConfig, response: AxiosResponse, auth: AxiosRequestConfig['auth'], ): AxiosRequestConfig { const authDetails = response.headers['www-authenticate'] .split(',') .map((v: string) => v.split('=')); if (authDetails) { const nonceCount = '000000001'; const cnonce = crypto.randomBytes(24).toString('hex'); const realm: string = authDetails .find((el: any) => el[0].toLowerCase().indexOf('realm') > -1)[1] .replace(/"/g, ''); // If authDetails does not have opaque, we should not add it to authorization. const opaqueKV = authDetails.find((el: any) => el[0].toLowerCase().indexOf('opaque') > -1); const opaque: string = opaqueKV ? opaqueKV[1].replace(/"/g, '') : undefined; const nonce: string = authDetails .find((el: any) => el[0].toLowerCase().indexOf('nonce') > -1)[1] .replace(/"/g, ''); const ha1 = crypto .createHash('md5') .update(`${auth?.username as string}:${realm}:${auth?.password as string}`) .digest('hex'); const urlURL = new url.URL(axios.getUri(axiosConfig)); const path = urlURL.pathname + urlURL.search; const ha2 = crypto .createHash('md5') .update(`${axiosConfig.method ?? 'GET'}:${path}`) .digest('hex'); const response = crypto .createHash('md5') .update(`${ha1}:${nonce}:${nonceCount}:${cnonce}:auth:${ha2}`) .digest('hex'); let authorization = `Digest username="${auth?.username as string}",realm="${realm}",` + `nonce="${nonce}",uri="${path}",qop="auth",algorithm="MD5",` + `response="${response}",nc="${nonceCount}",cnonce="${cnonce}"`; // Only when opaque exists, add it to authorization. if (opaque) { authorization += `,opaque="${opaque}"`; } if (axiosConfig.headers) { axiosConfig.headers.authorization = authorization; } else { axiosConfig.headers = { authorization }; } } return axiosConfig; } async function proxyRequestToAxios( workflow: Workflow, additionalData: IWorkflowExecuteAdditionalData, node: INode, uriOrObject: string | IDataObject, options?: IDataObject, ): Promise { // Check if there's a better way of getting this config here if (process.env.N8N_USE_DEPRECATED_REQUEST_LIB) { return requestPromiseWithDefaults.call( null, uriOrObject as unknown as RequiredUriUrl & RequestPromiseOptions, options as unknown as RequestCallback, ); } let axiosConfig: AxiosRequestConfig = { maxBodyLength: Infinity, maxContentLength: Infinity, }; type ConfigObject = { auth?: { sendImmediately: boolean }; resolveWithFullResponse?: boolean; simple?: boolean; }; let configObject: ConfigObject; if (uriOrObject !== undefined && typeof uriOrObject === 'string') { axiosConfig.url = uriOrObject; } if (uriOrObject !== undefined && typeof uriOrObject === 'object') { configObject = uriOrObject; } else { configObject = options || {}; } axiosConfig = Object.assign(axiosConfig, await parseRequestObject(configObject)); Logger.debug( 'Proxying request to axios', // { // originalConfig: configObject, // parsedConfig: axiosConfig, // } ); let requestFn: () => AxiosPromise; if (configObject.auth?.sendImmediately === false) { // for digest-auth requestFn = async () => { try { return await axios(axiosConfig); } catch (error) { const { response } = error; if (response?.status !== 401 || !response.headers['www-authenticate']?.includes('nonce')) { throw error; } const { auth } = axiosConfig; delete axiosConfig.auth; axiosConfig = digestAuthAxiosConfig(axiosConfig, response, auth); return await axios(axiosConfig); } }; } else { requestFn = async () => axios(axiosConfig); } try { const response = await requestFn(); if (configObject.resolveWithFullResponse === true) { let body = response.data; if (response.data === '') { if (axiosConfig.responseType === 'arraybuffer') { body = Buffer.alloc(0); } else { body = undefined; } } await additionalData.hooks?.executeHookFunctions('nodeFetchedData', [workflow.id, node]); return { body, headers: response.headers, statusCode: response.status, statusMessage: response.statusText, request: response.request, }; } else { let body = response.data; if (response.data === '') { if (axiosConfig.responseType === 'arraybuffer') { body = Buffer.alloc(0); } else { body = undefined; } } await additionalData.hooks?.executeHookFunctions('nodeFetchedData', [workflow.id, node]); return body; } } catch (error) { const { config, response } = error; // Axios hydrates the original error with more data. We extract them. // https://github.com/axios/axios/blob/master/lib/core/enhanceError.js // Note: `code` is ignored as it's an expected part of the errorData. if (error.isAxiosError) { if (response) { Logger.debug('Request proxied to Axios failed', { status: response.status }); let responseData = response.data; if (Buffer.isBuffer(responseData) || responseData instanceof Readable) { responseData = await binaryToBuffer(responseData).then((buffer) => buffer.toString('utf-8'), ); } if (configObject.simple === false) { if (configObject.resolveWithFullResponse) { return { body: responseData, headers: response.headers, statusCode: response.status, statusMessage: response.statusText, }; } else { return responseData; } } const message = `${response.status as number} - ${JSON.stringify(responseData)}`; throw Object.assign(new Error(message, { cause: error }), { statusCode: response.status, options: pick(config ?? {}, ['url', 'method', 'data', 'headers']), }); } else { throw Object.assign(new Error(error.message, { cause: error }), { options: pick(config ?? {}, ['url', 'method', 'data', 'headers']), }); } } throw error; } } function isIterator(obj: unknown): boolean { return obj instanceof Object && Symbol.iterator in obj; } function convertN8nRequestToAxios(n8nRequest: IHttpRequestOptions): AxiosRequestConfig { // Destructure properties with the same name first. const { headers, method, timeout, auth, proxy, url } = n8nRequest; const axiosRequest = { headers: headers ?? {}, method, timeout, auth, proxy, url, } as AxiosRequestConfig; axiosRequest.params = n8nRequest.qs; if (n8nRequest.baseURL !== undefined) { axiosRequest.baseURL = n8nRequest.baseURL; } if (n8nRequest.disableFollowRedirect === true) { axiosRequest.maxRedirects = 0; } if (n8nRequest.encoding !== undefined) { axiosRequest.responseType = n8nRequest.encoding; } if (n8nRequest.skipSslCertificateValidation === true) { axiosRequest.httpsAgent = new Agent({ rejectUnauthorized: false, }); } if (n8nRequest.arrayFormat !== undefined) { axiosRequest.paramsSerializer = (params) => { return stringify(params, { arrayFormat: n8nRequest.arrayFormat }); }; } const { body } = n8nRequest; if (body) { // Let's add some useful header standards here. const existingContentTypeHeaderKey = searchForHeader(axiosRequest.headers, 'content-type'); if (existingContentTypeHeaderKey === undefined) { axiosRequest.headers = axiosRequest.headers || {}; // We are only setting content type headers if the user did // not set it already manually. We're not overriding, even if it's wrong. if (body instanceof FormData) { axiosRequest.headers = { ...axiosRequest.headers, ...body.getHeaders(), }; } else if (body instanceof URLSearchParams) { axiosRequest.headers['Content-Type'] = 'application/x-www-form-urlencoded'; } } else if ( axiosRequest.headers[existingContentTypeHeaderKey] === 'application/x-www-form-urlencoded' ) { axiosRequest.data = new URLSearchParams(n8nRequest.body as Record); } // if there is a body and it's empty (does not have properties), // make sure not to send anything in it as some services fail when // sending GET request with empty body. if (isIterator(body) || Object.keys(body).length > 0) { axiosRequest.data = body; } } if (n8nRequest.json) { const key = searchForHeader(axiosRequest.headers, 'accept'); // If key exists, then the user has set both accept // header and the json flag. Header should take precedence. if (!key) { axiosRequest.headers.Accept = 'application/json'; } } const userAgentHeader = searchForHeader(axiosRequest.headers, 'user-agent'); // If key exists, then the user has set both accept // header and the json flag. Header should take precedence. if (!userAgentHeader) { axiosRequest.headers['User-Agent'] = 'n8n'; } if (n8nRequest.ignoreHttpStatusErrors) { axiosRequest.validateStatus = () => true; } return axiosRequest; } async function httpRequest( requestOptions: IHttpRequestOptions, ): Promise { const axiosRequest = convertN8nRequestToAxios(requestOptions); if ( axiosRequest.data === undefined || (axiosRequest.method !== undefined && axiosRequest.method.toUpperCase() === 'GET') ) { delete axiosRequest.data; } const result = await axios(axiosRequest); if (requestOptions.returnFullResponse) { return { body: result.data, headers: result.headers, statusCode: result.status, statusMessage: result.statusText, }; } return result.data; } /** * Returns binary file metadata */ export async function getBinaryMetadata(binaryDataId: string): Promise { return BinaryDataManager.getInstance().getBinaryMetadata(binaryDataId); } /** * Returns binary file stream for piping */ export function getBinaryStream(binaryDataId: string, chunkSize?: number): Readable { return BinaryDataManager.getInstance().getBinaryStream(binaryDataId, chunkSize); } export function assertBinaryData( inputData: ITaskDataConnections, node: INode, itemIndex: number, propertyName: string, inputIndex: number, ): IBinaryData { const binaryKeyData = inputData.main[inputIndex]![itemIndex]!.binary; if (binaryKeyData === undefined) { throw new NodeOperationError(node, 'No binary data exists on item!', { itemIndex, }); } const binaryPropertyData = binaryKeyData[propertyName]; if (binaryPropertyData === undefined) { throw new NodeOperationError(node, `Item has no binary property called "${propertyName}"`, { itemIndex, }); } return binaryPropertyData; } /** * Returns binary data buffer for given item index and property name. */ export async function getBinaryDataBuffer( inputData: ITaskDataConnections, itemIndex: number, propertyName: string, inputIndex: number, ): Promise { const binaryData = inputData.main[inputIndex]![itemIndex]!.binary![propertyName]!; return BinaryDataManager.getInstance().retrieveBinaryData(binaryData); } /** * Store an incoming IBinaryData & related buffer using the configured binary data manager. * * @export * @param {IBinaryData} data * @param {Buffer | Readable} binaryData * @returns {Promise} */ export async function setBinaryDataBuffer( data: IBinaryData, binaryData: Buffer | Readable, executionId: string, ): Promise { return BinaryDataManager.getInstance().storeBinaryData(data, binaryData, executionId); } export async function copyBinaryFile( executionId: string, filePath: string, fileName: string, mimeType?: string, ): Promise { let fileExtension: string | undefined; if (!mimeType) { // If no mime type is given figure it out if (filePath) { // Use file path to guess mime type const mimeTypeLookup = lookup(filePath); if (mimeTypeLookup) { mimeType = mimeTypeLookup; } } if (!mimeType) { // read the first bytes of the file to guess mime type const fileTypeData = await FileType.fromFile(filePath); if (fileTypeData) { mimeType = fileTypeData.mime; fileExtension = fileTypeData.ext; } } } if (!fileExtension && mimeType) { fileExtension = extension(mimeType) || undefined; } if (!mimeType) { // Fall back to text mimeType = 'text/plain'; } const returnData: IBinaryData = { mimeType, fileType: fileTypeFromMimeType(mimeType), fileExtension, data: '', }; if (fileName) { returnData.fileName = fileName; } else if (filePath) { returnData.fileName = path.parse(filePath).base; } return BinaryDataManager.getInstance().copyBinaryFile(returnData, filePath, executionId); } /** * Takes a buffer and converts it into the format n8n uses. It encodes the binary data as * base64 and adds metadata. */ async function prepareBinaryData( binaryData: Buffer | Readable, executionId: string, filePath?: string, mimeType?: string, ): Promise { let fileExtension: string | undefined; if (!mimeType) { // If no mime type is given figure it out if (filePath) { // Use file path to guess mime type const mimeTypeLookup = lookup(filePath); if (mimeTypeLookup) { mimeType = mimeTypeLookup; } } if (!mimeType) { if (Buffer.isBuffer(binaryData)) { // Use buffer to guess mime type const fileTypeData = await FileType.fromBuffer(binaryData); if (fileTypeData) { mimeType = fileTypeData.mime; fileExtension = fileTypeData.ext; } } else if (binaryData instanceof IncomingMessage) { mimeType = binaryData.headers['content-type']; } else { // TODO: detect filetype from other kind of streams } } } if (!fileExtension && mimeType) { fileExtension = extension(mimeType) || undefined; } if (!mimeType) { // Fall back to text mimeType = 'text/plain'; } const returnData: IBinaryData = { mimeType, fileType: fileTypeFromMimeType(mimeType), fileExtension, data: '', }; if (filePath) { if (filePath.includes('?')) { // Remove maybe present query parameters filePath = filePath.split('?').shift(); } const filePathParts = path.parse(filePath as string); if (filePathParts.dir !== '') { returnData.directory = filePathParts.dir; } returnData.fileName = filePathParts.base; // Remove the dot const fileExtension = filePathParts.ext.slice(1); if (fileExtension) { returnData.fileExtension = fileExtension; } } return setBinaryDataBuffer(returnData, binaryData, executionId); } /** * Makes a request using OAuth data for authentication * * @param {(OptionsWithUri | requestPromise.RequestPromiseOptions)} requestOptions * */ export async function requestOAuth2( this: IAllExecuteFunctions, credentialsType: string, requestOptions: OptionsWithUri | requestPromise.RequestPromiseOptions | IHttpRequestOptions, node: INode, additionalData: IWorkflowExecuteAdditionalData, oAuth2Options?: IOAuth2Options, isN8nRequest = false, ) { const credentials = await this.getCredentials(credentialsType); // Only the OAuth2 with authorization code grant needs connection if ( credentials.grantType === OAuth2GrantType.authorizationCode && credentials.oauthTokenData === undefined ) { throw new Error('OAuth credentials not connected!'); } const oAuthClient = new clientOAuth2({ clientId: credentials.clientId as string, clientSecret: credentials.clientSecret as string, accessTokenUri: credentials.accessTokenUrl as string, scopes: (credentials.scope as string).split(' '), }); let oauthTokenData = credentials.oauthTokenData as clientOAuth2.Data; // if it's the first time using the credentials, get the access token and save it into the DB. if ( credentials.grantType === OAuth2GrantType.clientCredentials && (oauthTokenData === undefined || Object.keys(oauthTokenData).length === 0) ) { const { data } = await getClientCredentialsToken(oAuthClient, credentials); // Find the credentials if (!node.credentials?.[credentialsType]) { throw new Error( `The node "${node.name}" does not have credentials of type "${credentialsType}"!`, ); } const nodeCredentials = node.credentials[credentialsType]; // Save the refreshed token await additionalData.credentialsHelper.updateCredentials( nodeCredentials, credentialsType, Object.assign(credentials, { oauthTokenData: data }), ); oauthTokenData = data; } const token = oAuthClient.createToken( get(oauthTokenData, oAuth2Options?.property as string) || oauthTokenData.accessToken, oauthTokenData.refreshToken, oAuth2Options?.tokenType || oauthTokenData.tokenType, oauthTokenData, ); // Signs the request by adding authorization headers or query parameters depending // on the token-type used. const newRequestOptions = token.sign(requestOptions as clientOAuth2.RequestObject); const newRequestHeaders = (newRequestOptions.headers = newRequestOptions.headers ?? {}); // If keep bearer is false remove the it from the authorization header if (oAuth2Options?.keepBearer === false && typeof newRequestHeaders.Authorization === 'string') { newRequestHeaders.Authorization = newRequestHeaders.Authorization.split(' ')[1]; } if (oAuth2Options?.keyToIncludeInAccessTokenHeader) { Object.assign(newRequestHeaders, { [oAuth2Options.keyToIncludeInAccessTokenHeader]: token.accessToken, }); } if (isN8nRequest) { return this.helpers.httpRequest(newRequestOptions).catch(async (error: AxiosError) => { if (error.response?.status === 401) { Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" expired. Should revalidate.`, ); const tokenRefreshOptions: IDataObject = {}; if (oAuth2Options?.includeCredentialsOnRefreshOnBody) { const body: IDataObject = { client_id: credentials.clientId as string, client_secret: credentials.clientSecret as string, }; tokenRefreshOptions.body = body; tokenRefreshOptions.headers = { Authorization: '', }; } let newToken; Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" has been renewed.`, ); // if it's OAuth2 with client credentials grant type, get a new token // instead of refreshing it. if (OAuth2GrantType.clientCredentials === credentials.grantType) { newToken = await getClientCredentialsToken(token.client, credentials); } else { newToken = await token.refresh(tokenRefreshOptions); } Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" has been renewed.`, ); credentials.oauthTokenData = newToken.data; // Find the credentials if (!node.credentials?.[credentialsType]) { throw new Error( `The node "${node.name}" does not have credentials of type "${credentialsType}"!`, ); } const nodeCredentials = node.credentials[credentialsType]; await additionalData.credentialsHelper.updateCredentials( nodeCredentials, credentialsType, credentials, ); const refreshedRequestOption = newToken.sign(requestOptions as clientOAuth2.RequestObject); if (oAuth2Options?.keyToIncludeInAccessTokenHeader) { Object.assign(newRequestHeaders, { [oAuth2Options.keyToIncludeInAccessTokenHeader]: token.accessToken, }); } return this.helpers.httpRequest(refreshedRequestOption); } throw error; }); } return this.helpers .request(newRequestOptions) .then((response) => { const requestOptions = newRequestOptions as any; if ( requestOptions.resolveWithFullResponse === true && requestOptions.simple === false && response.statusCode === (oAuth2Options?.tokenExpiredStatusCode === undefined ? 401 : oAuth2Options?.tokenExpiredStatusCode) ) { throw response; } return response; }) .catch(async (error: IResponseError) => { const statusCodeReturned = oAuth2Options?.tokenExpiredStatusCode === undefined ? 401 : oAuth2Options?.tokenExpiredStatusCode; if (error.statusCode === statusCodeReturned) { // Token is probably not valid anymore. So try refresh it. const tokenRefreshOptions: IDataObject = {}; if (oAuth2Options?.includeCredentialsOnRefreshOnBody) { const body: IDataObject = { client_id: credentials.clientId, client_secret: credentials.clientSecret, }; tokenRefreshOptions.body = body; // Override authorization property so the credentials are not included in it tokenRefreshOptions.headers = { Authorization: '', }; } Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" expired. Should revalidate.`, ); let newToken; // if it's OAuth2 with client credentials grant type, get a new token // instead of refreshing it. if (OAuth2GrantType.clientCredentials === credentials.grantType) { newToken = await getClientCredentialsToken(token.client, credentials); } else { newToken = await token.refresh(tokenRefreshOptions); } Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" has been renewed.`, ); credentials.oauthTokenData = newToken.data; // Find the credentials if (!node.credentials?.[credentialsType]) { throw new Error( `The node "${node.name}" does not have credentials of type "${credentialsType}"!`, ); } const nodeCredentials = node.credentials[credentialsType]; // Save the refreshed token await additionalData.credentialsHelper.updateCredentials( nodeCredentials, credentialsType, credentials as unknown as ICredentialDataDecryptedObject, ); Logger.debug( `OAuth2 token for "${credentialsType}" used by node "${node.name}" has been saved to database successfully.`, ); // Make the request again with the new token const newRequestOptions = newToken.sign(requestOptions as clientOAuth2.RequestObject); newRequestOptions.headers = newRequestOptions.headers ?? {}; if (oAuth2Options?.keyToIncludeInAccessTokenHeader) { Object.assign(newRequestOptions.headers, { [oAuth2Options.keyToIncludeInAccessTokenHeader]: token.accessToken, }); } return this.helpers.request(newRequestOptions); } // Unknown error so simply throw it throw error; }); } /** * Makes a request using OAuth1 data for authentication */ export async function requestOAuth1( this: IAllExecuteFunctions, credentialsType: string, requestOptions: | OptionsWithUrl | OptionsWithUri | requestPromise.RequestPromiseOptions | IHttpRequestOptions, isN8nRequest = false, ) { const credentials = await this.getCredentials(credentialsType); if (credentials === undefined) { throw new Error('No credentials were returned!'); } if (credentials.oauthTokenData === undefined) { throw new Error('OAuth credentials not connected!'); } const oauth = new clientOAuth1({ consumer: { key: credentials.consumerKey as string, secret: credentials.consumerSecret as string, }, signature_method: credentials.signatureMethod as string, hash_function(base, key) { const algorithm = credentials.signatureMethod === 'HMAC-SHA1' ? 'sha1' : 'sha256'; return createHmac(algorithm, key).update(base).digest('base64'); }, }); const oauthTokenData = credentials.oauthTokenData as IDataObject; const token: Token = { key: oauthTokenData.oauth_token as string, secret: oauthTokenData.oauth_token_secret as string, }; // @ts-expect-error @TECH_DEBT: Remove request library requestOptions.data = { ...requestOptions.qs, ...requestOptions.form }; // Fixes issue that OAuth1 library only works with "url" property and not with "uri" // @ts-expect-error @TECH_DEBT: Remove request library if (requestOptions.uri && !requestOptions.url) { // @ts-expect-error @TECH_DEBT: Remove request library requestOptions.url = requestOptions.uri; // @ts-expect-error @TECH_DEBT: Remove request library delete requestOptions.uri; } requestOptions.headers = oauth.toHeader( oauth.authorize(requestOptions as unknown as clientOAuth1.RequestOptions, token), ); if (isN8nRequest) { return this.helpers.httpRequest(requestOptions as IHttpRequestOptions); } return this.helpers.request(requestOptions).catch(async (error: IResponseError) => { // Unknown error so simply throw it throw error; }); } export async function httpRequestWithAuthentication( this: IAllExecuteFunctions, credentialsType: string, requestOptions: IHttpRequestOptions, workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, additionalCredentialOptions?: IAdditionalCredentialOptions, ) { let credentialsDecrypted: ICredentialDataDecryptedObject | undefined; try { const parentTypes = additionalData.credentialsHelper.getParentTypes(credentialsType); if (parentTypes.includes('oAuth1Api')) { return await requestOAuth1.call(this, credentialsType, requestOptions, true); } if (parentTypes.includes('oAuth2Api')) { return await requestOAuth2.call( this, credentialsType, requestOptions, node, additionalData, additionalCredentialOptions?.oauth2, true, ); } if (additionalCredentialOptions?.credentialsDecrypted) { credentialsDecrypted = additionalCredentialOptions.credentialsDecrypted.data; } else { credentialsDecrypted = await this.getCredentials(credentialsType); } if (credentialsDecrypted === undefined) { throw new NodeOperationError( node, `Node "${node.name}" does not have any credentials of type "${credentialsType}" set!`, ); } const data = await additionalData.credentialsHelper.preAuthentication( { helpers: this.helpers }, credentialsDecrypted, credentialsType, node, false, ); if (data) { // make the updated property in the credentials // available to the authenticate method Object.assign(credentialsDecrypted, data); } requestOptions = await additionalData.credentialsHelper.authenticate( credentialsDecrypted, credentialsType, requestOptions, workflow, node, additionalData.timezone, ); return await httpRequest(requestOptions); } catch (error) { // if there is a pre authorization method defined and // the method failed due to unauthorized request if ( error.response?.status === 401 && additionalData.credentialsHelper.preAuthentication !== undefined ) { try { if (credentialsDecrypted !== undefined) { // try to refresh the credentials const data = await additionalData.credentialsHelper.preAuthentication( { helpers: this.helpers }, credentialsDecrypted, credentialsType, node, true, ); if (data) { // make the updated property in the credentials // available to the authenticate method Object.assign(credentialsDecrypted, data); } requestOptions = await additionalData.credentialsHelper.authenticate( credentialsDecrypted, credentialsType, requestOptions, workflow, node, additionalData.timezone, ); } // retry the request return await httpRequest(requestOptions); } catch (error) { throw new NodeApiError(this.getNode(), error); } } throw new NodeApiError(this.getNode(), error); } } /** * Takes generic input data and brings it into the json format n8n uses. * * @param {(IDataObject | IDataObject[])} jsonData */ export function returnJsonArray(jsonData: IDataObject | IDataObject[]): INodeExecutionData[] { const returnData: INodeExecutionData[] = []; if (!Array.isArray(jsonData)) { jsonData = [jsonData]; } jsonData.forEach((data: IDataObject & { json?: IDataObject }) => { if (data?.json) { // We already have the JSON key so avoid double wrapping returnData.push({ ...data, json: data.json }); } else { returnData.push({ json: data }); } }); return returnData; } /** * Takes generic input data and brings it into the new json, pairedItem format n8n uses. * @param {(IPairedItemData)} itemData * @param {(INodeExecutionData[])} inputData */ export function constructExecutionMetaData( inputData: INodeExecutionData[], options: { itemData: IPairedItemData | IPairedItemData[] }, ): NodeExecutionWithMetadata[] { const { itemData } = options; return inputData.map((data: INodeExecutionData) => { const { json, ...rest } = data; return { json, pairedItem: itemData, ...rest } as NodeExecutionWithMetadata; }); } /** * Automatically put the objects under a 'json' key and don't error, * if some objects contain json/binary keys and others don't, throws error 'Inconsistent item format' * * @param {INodeExecutionData | INodeExecutionData[]} executionData */ export function normalizeItems( executionData: INodeExecutionData | INodeExecutionData[], ): INodeExecutionData[] { if (typeof executionData === 'object' && !Array.isArray(executionData)) { executionData = executionData.json ? [executionData] : [{ json: executionData as IDataObject }]; } if (executionData.every((item) => typeof item === 'object' && 'json' in item)) return executionData; if (executionData.some((item) => typeof item === 'object' && 'json' in item)) { throw new Error('Inconsistent item format'); } if (executionData.every((item) => typeof item === 'object' && 'binary' in item)) { const normalizedItems: INodeExecutionData[] = []; executionData.forEach((item) => { const json = Object.keys(item).reduce((acc, key) => { if (key === 'binary') return acc; return { ...acc, [key]: item[key] }; }, {}); normalizedItems.push({ json, binary: item.binary, }); }); return normalizedItems; } if (executionData.some((item) => typeof item === 'object' && 'binary' in item)) { throw new Error('Inconsistent item format'); } return executionData.map((item) => { return { json: item }; }); } // TODO: Move up later export async function requestWithAuthentication( this: IAllExecuteFunctions, credentialsType: string, requestOptions: OptionsWithUri | requestPromise.RequestPromiseOptions, workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, additionalCredentialOptions?: IAdditionalCredentialOptions, ) { let credentialsDecrypted: ICredentialDataDecryptedObject | undefined; try { const parentTypes = additionalData.credentialsHelper.getParentTypes(credentialsType); if (parentTypes.includes('oAuth1Api')) { return await requestOAuth1.call(this, credentialsType, requestOptions, false); } if (parentTypes.includes('oAuth2Api')) { return await requestOAuth2.call( this, credentialsType, requestOptions, node, additionalData, additionalCredentialOptions?.oauth2, false, ); } if (additionalCredentialOptions?.credentialsDecrypted) { credentialsDecrypted = additionalCredentialOptions.credentialsDecrypted.data; } else { credentialsDecrypted = await this.getCredentials(credentialsType); } if (credentialsDecrypted === undefined) { throw new NodeOperationError( node, `Node "${node.name}" does not have any credentials of type "${credentialsType}" set!`, ); } const data = await additionalData.credentialsHelper.preAuthentication( { helpers: this.helpers }, credentialsDecrypted, credentialsType, node, false, ); if (data) { // make the updated property in the credentials // available to the authenticate method Object.assign(credentialsDecrypted, data); } requestOptions = await additionalData.credentialsHelper.authenticate( credentialsDecrypted, credentialsType, requestOptions as IHttpRequestOptions, workflow, node, additionalData.timezone, ); return await proxyRequestToAxios(workflow, additionalData, node, requestOptions as IDataObject); } catch (error) { try { if (credentialsDecrypted !== undefined) { // try to refresh the credentials const data = await additionalData.credentialsHelper.preAuthentication( { helpers: this.helpers }, credentialsDecrypted, credentialsType, node, true, ); if (data) { // make the updated property in the credentials // available to the authenticate method Object.assign(credentialsDecrypted, data); requestOptions = await additionalData.credentialsHelper.authenticate( credentialsDecrypted, credentialsType, requestOptions as IHttpRequestOptions, workflow, node, additionalData.timezone, ); // retry the request return await proxyRequestToAxios( workflow, additionalData, node, requestOptions as IDataObject, ); } } throw error; } catch (error) { throw new NodeApiError(this.getNode(), error); } } } /** * Returns the additional keys for Expressions and Function-Nodes * */ export function getAdditionalKeys( additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, runExecutionData: IRunExecutionData | null, ): IWorkflowDataProxyAdditionalKeys { const executionId = additionalData.executionId || PLACEHOLDER_EMPTY_EXECUTION_ID; const resumeUrl = `${additionalData.webhookWaitingBaseUrl}/${executionId}`; return { $execution: { id: executionId, mode: mode === 'manual' ? 'test' : 'production', resumeUrl, customData: runExecutionData ? { set(key: string, value: string): void { setWorkflowExecutionMetadata(runExecutionData, key, value); }, setAll(obj: Record): void { setAllWorkflowExecutionMetadata(runExecutionData, obj); }, get(key: string): string { return getWorkflowExecutionMetadata(runExecutionData, key); }, getAll(): Record { return getAllWorkflowExecutionMetadata(runExecutionData); }, } : undefined, }, // deprecated $executionId: executionId, $resumeWebhookUrl: resumeUrl, }; } /** * Returns the requested decrypted credentials if the node has access to them. * * @param {Workflow} workflow Workflow which requests the data * @param {INode} node Node which request the data * @param {string} type The credential type to return */ export async function getCredentials( workflow: Workflow, node: INode, type: string, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, runExecutionData?: IRunExecutionData | null, runIndex?: number, connectionInputData?: INodeExecutionData[], itemIndex?: number, ): Promise { // Get the NodeType as it has the information if the credentials are required const nodeType = workflow.nodeTypes.getByNameAndVersion(node.type, node.typeVersion); if (nodeType === undefined) { throw new NodeOperationError( node, `Node type "${node.type}" is not known so can not get credentials!`, ); } // Hardcode for now for security reasons that only a single node can access // all credentials const fullAccess = ['n8n-nodes-base.httpRequest'].includes(node.type); let nodeCredentialDescription: INodeCredentialDescription | undefined; if (!fullAccess) { if (nodeType.description.credentials === undefined) { throw new NodeOperationError( node, `Node type "${node.type}" does not have any credentials defined!`, ); } nodeCredentialDescription = nodeType.description.credentials.find( (credentialTypeDescription) => credentialTypeDescription.name === type, ); if (nodeCredentialDescription === undefined) { throw new NodeOperationError( node, `Node type "${node.type}" does not have any credentials of type "${type}" defined!`, ); } if ( !NodeHelpers.displayParameter( additionalData.currentNodeParameters || node.parameters, nodeCredentialDescription, node, node.parameters, ) ) { // Credentials should not be displayed even if they would be defined throw new NodeOperationError(node, 'Credentials not found'); } } // Check if node has any credentials defined if (!fullAccess && !node.credentials?.[type]) { // If none are defined check if the credentials are required or not if (nodeCredentialDescription?.required === true) { // Credentials are required so error if (!node.credentials) { throw new NodeOperationError(node, 'Node does not have any credentials set!'); } if (!node.credentials[type]) { throw new NodeOperationError(node, `Node does not have any credentials set for "${type}"!`); } } else { // Credentials are not required throw new NodeOperationError(node, 'Node does not require credentials'); } } if (fullAccess && !node.credentials?.[type]) { // Make sure that fullAccess nodes still behave like before that if they // request access to credentials that are currently not set it returns undefined throw new NodeOperationError(node, 'Credentials not found'); } let expressionResolveValues: ICredentialsExpressionResolveValues | undefined; if (connectionInputData && runExecutionData && runIndex !== undefined) { expressionResolveValues = { connectionInputData, itemIndex: itemIndex || 0, node, runExecutionData, runIndex, workflow, } as ICredentialsExpressionResolveValues; } const nodeCredentials = node.credentials ? node.credentials[type] : ({} as INodeCredentialsDetails); // TODO: solve using credentials via expression // if (name.charAt(0) === '=') { // // If the credential name is an expression resolve it // const additionalKeys = getAdditionalKeys(additionalData, mode); // name = workflow.expression.getParameterValue( // name, // runExecutionData || null, // runIndex || 0, // itemIndex || 0, // node.name, // connectionInputData || [], // mode, // additionalKeys, // ) as string; // } const decryptedDataObject = await additionalData.credentialsHelper.getDecrypted( nodeCredentials, type, mode, additionalData.timezone, false, expressionResolveValues, ); return decryptedDataObject; } /** * Clean up parameter data to make sure that only valid data gets returned * INFO: Currently only converts Luxon Dates as we know for sure it will not be breaking */ function cleanupParameterData(inputData: NodeParameterValueType): void { if (typeof inputData !== 'object' || inputData === null) { return; } if (Array.isArray(inputData)) { inputData.forEach((value) => cleanupParameterData(value)); return; } if (typeof inputData === 'object') { Object.keys(inputData).forEach((key) => { if (typeof inputData[key as keyof typeof inputData] === 'object') { if (inputData[key as keyof typeof inputData]?.constructor.name === 'DateTime') { // Is a special luxon date so convert to string inputData[key as keyof typeof inputData] = inputData[key as keyof typeof inputData]?.toString(); } else { cleanupParameterData(inputData[key as keyof typeof inputData]); } } }); } } /** * Returns the requested resolved (all expressions replaced) node parameters. * * @param {(IRunExecutionData | null)} runExecutionData */ export function getNodeParameter( workflow: Workflow, runExecutionData: IRunExecutionData | null, runIndex: number, connectionInputData: INodeExecutionData[], node: INode, parameterName: string, itemIndex: number, mode: WorkflowExecuteMode, timezone: string, additionalKeys: IWorkflowDataProxyAdditionalKeys, executeData?: IExecuteData, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object { const nodeType = workflow.nodeTypes.getByNameAndVersion(node.type, node.typeVersion); if (nodeType === undefined) { throw new Error(`Node type "${node.type}" is not known so can not return parameter value!`); } const value = get(node.parameters, parameterName, fallbackValue); if (value === undefined) { throw new Error(`Could not get parameter "${parameterName}"!`); } let returnData; try { returnData = workflow.expression.getParameterValue( value, runExecutionData, runIndex, itemIndex, node.name, connectionInputData, mode, timezone, additionalKeys, executeData, ); cleanupParameterData(returnData); } catch (e) { if (e instanceof ExpressionError && node.continueOnFail && node.name === 'Set') { returnData = [{ name: undefined, value: undefined }]; } else { if (e.context) e.context.parameter = parameterName; e.cause = value; throw e; } } // This is outside the try/catch because it throws errors with proper messages if (options?.extractValue) { returnData = extractValue(returnData, parameterName, node, nodeType); } return returnData; } /** * Returns if execution should be continued even if there was an error. * */ export function continueOnFail(node: INode): boolean { return get(node, 'continueOnFail', false); } /** * Returns the webhook URL of the webhook with the given name * */ export function getNodeWebhookUrl( name: string, workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, timezone: string, additionalKeys: IWorkflowDataProxyAdditionalKeys, isTest?: boolean, ): string | undefined { let baseUrl = additionalData.webhookBaseUrl; if (isTest === true) { baseUrl = additionalData.webhookTestBaseUrl; } // eslint-disable-next-line @typescript-eslint/no-use-before-define const webhookDescription = getWebhookDescription(name, workflow, node); if (webhookDescription === undefined) { return undefined; } const path = workflow.expression.getSimpleParameterValue( node, webhookDescription.path, mode, timezone, additionalKeys, ); if (path === undefined) { return undefined; } const isFullPath: boolean = workflow.expression.getSimpleParameterValue( node, webhookDescription.isFullPath, mode, timezone, additionalKeys, undefined, false, ) as boolean; return NodeHelpers.getNodeWebhookUrl(baseUrl, workflow.id!, node, path.toString(), isFullPath); } /** * Returns the timezone for the workflow * */ export function getTimezone( workflow: Workflow, additionalData: IWorkflowExecuteAdditionalData, ): string { // eslint-disable-next-line @typescript-eslint/prefer-optional-chain if (workflow.settings !== undefined && workflow.settings.timezone !== undefined) { return (workflow.settings as IWorkflowSettings).timezone as string; } return additionalData.timezone; } /** * Returns the full webhook description of the webhook with the given name * */ export function getWebhookDescription( name: string, workflow: Workflow, node: INode, ): IWebhookDescription | undefined { const nodeType = workflow.nodeTypes.getByNameAndVersion(node.type, node.typeVersion); if (nodeType.description.webhooks === undefined) { // Node does not have any webhooks so return return undefined; } // eslint-disable-next-line no-restricted-syntax for (const webhookDescription of nodeType.description.webhooks) { if (webhookDescription.name === name) { return webhookDescription; } } return undefined; } const getCommonWorkflowFunctions = ( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, ): Omit => ({ logger: Logger, getNode: () => deepCopy(node), getWorkflow: () => ({ id: workflow.id, name: workflow.name, active: workflow.active, }), getWorkflowStaticData: (type) => workflow.getStaticData(type, node), getRestApiUrl: () => additionalData.restApiUrl, getTimezone: () => getTimezone(workflow, additionalData), }); const getRequestHelperFunctions = ( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, ): RequestHelperFunctions => ({ httpRequest, async httpRequestWithAuthentication( this, credentialsType, requestOptions, additionalCredentialOptions, ): Promise { return httpRequestWithAuthentication.call( this, credentialsType, requestOptions, workflow, node, additionalData, additionalCredentialOptions, ); }, request: async (uriOrObject, options) => proxyRequestToAxios(workflow, additionalData, node, uriOrObject, options), async requestWithAuthentication( this, credentialsType, requestOptions, additionalCredentialOptions, ): Promise { return requestWithAuthentication.call( this, credentialsType, requestOptions, workflow, node, additionalData, additionalCredentialOptions, ); }, async requestOAuth1( this: IAllExecuteFunctions, credentialsType: string, requestOptions: OptionsWithUrl | requestPromise.RequestPromiseOptions, ): Promise { return requestOAuth1.call(this, credentialsType, requestOptions); }, async requestOAuth2( this: IAllExecuteFunctions, credentialsType: string, requestOptions: OptionsWithUri | requestPromise.RequestPromiseOptions, oAuth2Options?: IOAuth2Options, ): Promise { return requestOAuth2.call( this, credentialsType, requestOptions, node, additionalData, oAuth2Options, ); }, }); const getFileSystemHelperFunctions = (node: INode): FileSystemHelperFunctions => ({ async createReadStream(filePath) { try { await fsAccess(filePath); } catch (error) { throw error.code === 'ENOENT' ? new NodeOperationError(node, error, { message: `The file "${String(filePath)}" could not be accessed.`, }) : error; } return createReadStream(filePath); }, }); const getNodeHelperFunctions = ({ executionId, }: IWorkflowExecuteAdditionalData): NodeHelperFunctions => ({ copyBinaryFile: async (filePath, fileName, mimeType) => copyBinaryFile(executionId!, filePath, fileName, mimeType), }); const getBinaryHelperFunctions = ({ executionId, }: IWorkflowExecuteAdditionalData): BinaryHelperFunctions => ({ getBinaryStream, getBinaryMetadata, binaryToBuffer, prepareBinaryData: async (binaryData, filePath, mimeType) => prepareBinaryData(binaryData, executionId!, filePath, mimeType), setBinaryDataBuffer: async (data, binaryData) => setBinaryDataBuffer(data, binaryData, executionId!), copyBinaryFile: async () => { throw new Error('copyBinaryFile has been removed. Please upgrade this node'); }, }); /** * Returns the execute functions the poll nodes have access to. */ // TODO: Check if I can get rid of: additionalData, and so then maybe also at ActiveWorkflowRunner.add export function getExecutePollFunctions( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, activation: WorkflowActivateMode, ): IPollFunctions { return ((workflow: Workflow, node: INode) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), __emit: (data: INodeExecutionData[][]): void => { throw new Error('Overwrite NodeExecuteFunctions.getExecutePollFunctions.__emit function!'); }, __emitError(error: Error) { throw new Error( 'Overwrite NodeExecuteFunctions.getExecutePollFunctions.__emitError function!', ); }, getMode: () => mode, getActivationMode: () => activation, getCredentials: async (type) => getCredentials(workflow, node, type, additionalData, mode), getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { const runExecutionData: IRunExecutionData | null = null; const itemIndex = 0; const runIndex = 0; const connectionInputData: INodeExecutionData[] = []; return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), undefined, fallbackValue, options, ); }, helpers: { createDeferredPromise, ...getRequestHelperFunctions(workflow, node, additionalData), ...getBinaryHelperFunctions(additionalData), returnJsonArray, }, }; })(workflow, node); } /** * Returns the execute functions the trigger nodes have access to. */ // TODO: Check if I can get rid of: additionalData, and so then maybe also at ActiveWorkflowRunner.add export function getExecuteTriggerFunctions( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, activation: WorkflowActivateMode, ): ITriggerFunctions { return ((workflow: Workflow, node: INode) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), emit: (data: INodeExecutionData[][]): void => { throw new Error('Overwrite NodeExecuteFunctions.getExecuteTriggerFunctions.emit function!'); }, emitError: (error: Error): void => { throw new Error('Overwrite NodeExecuteFunctions.getExecuteTriggerFunctions.emit function!'); }, getMode: () => mode, getActivationMode: () => activation, getCredentials: async (type) => getCredentials(workflow, node, type, additionalData, mode), getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { const runExecutionData: IRunExecutionData | null = null; const itemIndex = 0; const runIndex = 0; const connectionInputData: INodeExecutionData[] = []; return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), undefined, fallbackValue, options, ); }, helpers: { createDeferredPromise, ...getRequestHelperFunctions(workflow, node, additionalData), ...getBinaryHelperFunctions(additionalData), returnJsonArray, }, }; })(workflow, node); } /** * Returns the execute functions regular nodes have access to. */ export function getExecuteFunctions( workflow: Workflow, runExecutionData: IRunExecutionData, runIndex: number, connectionInputData: INodeExecutionData[], inputData: ITaskDataConnections, node: INode, additionalData: IWorkflowExecuteAdditionalData, executeData: IExecuteData, mode: WorkflowExecuteMode, ): IExecuteFunctions { return ((workflow, runExecutionData, connectionInputData, inputData, node) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), getMode: () => mode, getCredentials: async (type, itemIndex) => getCredentials( workflow, node, type, additionalData, mode, runExecutionData, runIndex, connectionInputData, itemIndex, ), getExecuteData: () => executeData, continueOnFail: () => continueOnFail(node), evaluateExpression: (expression: string, itemIndex: number) => { return workflow.expression.resolveSimpleParameterValue( `=${expression}`, {}, runExecutionData, runIndex, itemIndex, node.name, connectionInputData, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, ); }, async executeWorkflow( workflowInfo: IExecuteWorkflowInfo, inputData?: INodeExecutionData[], ): Promise { return additionalData .executeWorkflow(workflowInfo, additionalData, { parentWorkflowId: workflow.id?.toString(), inputData, parentWorkflowSettings: workflow.settings, }) .then(async (result) => BinaryDataManager.getInstance().duplicateBinaryData( result, additionalData.executionId!, ), ); }, getContext(type: string): IContextObject { return NodeHelpers.getContext(runExecutionData, type, node); }, getExecutionId: () => additionalData.executionId!, getInputData: (inputIndex = 0, inputName = 'main') => { if (!inputData.hasOwnProperty(inputName)) { // Return empty array because else it would throw error when nothing is connected to input return []; } // TODO: Check if nodeType has input with that index defined if (inputData[inputName].length < inputIndex) { throw new Error(`Could not get input index "${inputIndex}" of input "${inputName}"!`); } if (inputData[inputName][inputIndex] === null) { // return []; throw new Error(`Value "${inputIndex}" of input "${inputName}" did not get set!`); } return inputData[inputName][inputIndex] as INodeExecutionData[]; }, getInputSourceData: (inputIndex = 0, inputName = 'main') => { if (executeData?.source === null) { // Should never happen as n8n sets it automatically throw new Error('Source data is missing!'); } return executeData.source[inputName][inputIndex]; }, getNodeParameter: ( parameterName: string, itemIndex: number, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, fallbackValue, options, ); }, getWorkflowDataProxy: (itemIndex: number): IWorkflowDataProxyData => { const dataProxy = new WorkflowDataProxy( workflow, runExecutionData, runIndex, itemIndex, node.name, connectionInputData, {}, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, ); return dataProxy.getDataProxy(); }, prepareOutputData: NodeHelpers.prepareOutputData, binaryToBuffer, async putExecutionToWait(waitTill: Date): Promise { runExecutionData.waitTill = waitTill; if (additionalData.setExecutionStatus) { additionalData.setExecutionStatus('waiting'); } }, sendMessageToUI(...args: any[]): void { if (mode !== 'manual') { return; } try { if (additionalData.sendMessageToUI) { args = args.map((arg) => { // prevent invalid dates from being logged as null if (arg.isLuxonDateTime && arg.invalidReason) return { ...arg }; // log valid dates in human readable format, as in browser if (arg.isLuxonDateTime) return new Date(arg.ts).toString(); if (arg instanceof Date) return arg.toString(); return arg; }); additionalData.sendMessageToUI(node.name, args); } } catch (error) { // eslint-disable-next-line @typescript-eslint/restrict-template-expressions Logger.warn(`There was a problem sending message to UI: ${error.message}`); } }, async sendResponse(response: IExecuteResponsePromiseData): Promise { await additionalData.hooks?.executeHookFunctions('sendResponse', [response]); }, helpers: { createDeferredPromise, ...getRequestHelperFunctions(workflow, node, additionalData), ...getFileSystemHelperFunctions(node), ...getBinaryHelperFunctions(additionalData), assertBinaryData: (itemIndex, propertyName) => assertBinaryData(inputData, node, itemIndex, propertyName, 0), getBinaryDataBuffer: async (itemIndex, propertyName) => getBinaryDataBuffer(inputData, itemIndex, propertyName, 0), returnJsonArray, normalizeItems, constructExecutionMetaData, }, nodeHelpers: getNodeHelperFunctions(additionalData), }; })(workflow, runExecutionData, connectionInputData, inputData, node) as IExecuteFunctions; } /** * Returns the execute functions regular nodes have access to when single-function is defined. */ export function getExecuteSingleFunctions( workflow: Workflow, runExecutionData: IRunExecutionData, runIndex: number, connectionInputData: INodeExecutionData[], inputData: ITaskDataConnections, node: INode, itemIndex: number, additionalData: IWorkflowExecuteAdditionalData, executeData: IExecuteData, mode: WorkflowExecuteMode, ): IExecuteSingleFunctions { return ((workflow, runExecutionData, connectionInputData, inputData, node, itemIndex) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), continueOnFail: () => continueOnFail(node), evaluateExpression: (expression: string, evaluateItemIndex: number | undefined) => { evaluateItemIndex = evaluateItemIndex === undefined ? itemIndex : evaluateItemIndex; return workflow.expression.resolveSimpleParameterValue( `=${expression}`, {}, runExecutionData, runIndex, evaluateItemIndex, node.name, connectionInputData, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, ); }, getContext(type: string): IContextObject { return NodeHelpers.getContext(runExecutionData, type, node); }, getCredentials: async (type) => getCredentials( workflow, node, type, additionalData, mode, runExecutionData, runIndex, connectionInputData, itemIndex, ), getInputData: (inputIndex = 0, inputName = 'main') => { if (!inputData.hasOwnProperty(inputName)) { // Return empty array because else it would throw error when nothing is connected to input return { json: {} }; } // TODO: Check if nodeType has input with that index defined if (inputData[inputName].length < inputIndex) { throw new Error(`Could not get input index "${inputIndex}" of input "${inputName}"!`); } const allItems = inputData[inputName][inputIndex]; if (allItems === null) { // return []; throw new Error(`Value "${inputIndex}" of input "${inputName}" did not get set!`); } if (allItems[itemIndex] === null) { // return []; throw new Error( `Value "${inputIndex}" of input "${inputName}" with itemIndex "${itemIndex}" did not get set!`, ); } return allItems[itemIndex]; }, getInputSourceData: (inputIndex = 0, inputName = 'main') => { if (executeData?.source === null) { // Should never happen as n8n sets it automatically throw new Error('Source data is missing!'); } return executeData.source[inputName][inputIndex] as ISourceData; }, getItemIndex: () => itemIndex, getMode: () => mode, getExecuteData: () => executeData, getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, fallbackValue, options, ); }, getWorkflowDataProxy: (): IWorkflowDataProxyData => { const dataProxy = new WorkflowDataProxy( workflow, runExecutionData, runIndex, itemIndex, node.name, connectionInputData, {}, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), executeData, ); return dataProxy.getDataProxy(); }, helpers: { createDeferredPromise, ...getRequestHelperFunctions(workflow, node, additionalData), ...getBinaryHelperFunctions(additionalData), assertBinaryData: (propertyName, inputIndex = 0) => assertBinaryData(inputData, node, itemIndex, propertyName, inputIndex), getBinaryDataBuffer: async (propertyName, inputIndex = 0) => getBinaryDataBuffer(inputData, itemIndex, propertyName, inputIndex), }, }; })(workflow, runExecutionData, connectionInputData, inputData, node, itemIndex); } export function getCredentialTestFunctions(): ICredentialTestFunctions { return { helpers: { request: requestPromiseWithDefaults, }, }; } /** * Returns the execute functions regular nodes have access to in load-options-function. */ export function getLoadOptionsFunctions( workflow: Workflow, node: INode, path: string, additionalData: IWorkflowExecuteAdditionalData, ): ILoadOptionsFunctions { return ((workflow: Workflow, node: INode, path: string) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), getCredentials: async (type) => getCredentials(workflow, node, type, additionalData, 'internal'), getCurrentNodeParameter: ( parameterPath: string, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object | undefined => { const nodeParameters = additionalData.currentNodeParameters; if (parameterPath.charAt(0) === '&') { parameterPath = `${path.split('.').slice(1, -1).join('.')}.${parameterPath.slice(1)}`; } let returnData = get(nodeParameters, parameterPath); // This is outside the try/catch because it throws errors with proper messages if (options?.extractValue) { const nodeType = workflow.nodeTypes.getByNameAndVersion(node.type, node.typeVersion); if (nodeType === undefined) { throw new Error( `Node type "${node.type}" is not known so can not return parameter value!`, ); } returnData = extractValue( returnData, parameterPath, node, nodeType, ) as NodeParameterValueType; } return returnData; }, getCurrentNodeParameters: () => additionalData.currentNodeParameters, getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { const runExecutionData: IRunExecutionData | null = null; const itemIndex = 0; const runIndex = 0; const mode = 'internal' as WorkflowExecuteMode; const connectionInputData: INodeExecutionData[] = []; return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), undefined, fallbackValue, options, ); }, helpers: getRequestHelperFunctions(workflow, node, additionalData), }; })(workflow, node, path); } /** * Returns the execute functions regular nodes have access to in hook-function. */ export function getExecuteHookFunctions( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, activation: WorkflowActivateMode, isTest?: boolean, webhookData?: IWebhookData, ): IHookFunctions { return ((workflow: Workflow, node: INode) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), getCredentials: async (type) => getCredentials(workflow, node, type, additionalData, mode), getMode: () => mode, getActivationMode: () => activation, getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { const runExecutionData: IRunExecutionData | null = null; const itemIndex = 0; const runIndex = 0; const connectionInputData: INodeExecutionData[] = []; return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, runExecutionData), undefined, fallbackValue, options, ); }, getNodeWebhookUrl: (name: string): string | undefined => { return getNodeWebhookUrl( name, workflow, node, additionalData, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, null), isTest, ); }, getWebhookName(): string { if (webhookData === undefined) { throw new Error('Is only supported in webhook functions!'); } return webhookData.webhookDescription.name; }, getWebhookDescription: (name) => getWebhookDescription(name, workflow, node), helpers: getRequestHelperFunctions(workflow, node, additionalData), }; })(workflow, node); } /** * Returns the execute functions regular nodes have access to when webhook-function is defined. */ export function getExecuteWebhookFunctions( workflow: Workflow, node: INode, additionalData: IWorkflowExecuteAdditionalData, mode: WorkflowExecuteMode, webhookData: IWebhookData, ): IWebhookFunctions { return ((workflow: Workflow, node: INode) => { return { ...getCommonWorkflowFunctions(workflow, node, additionalData), getBodyData(): IDataObject { if (additionalData.httpRequest === undefined) { throw new Error('Request is missing!'); } return additionalData.httpRequest.body; }, getCredentials: async (type) => getCredentials(workflow, node, type, additionalData, mode), getHeaderData(): IncomingHttpHeaders { if (additionalData.httpRequest === undefined) { throw new Error('Request is missing!'); } return additionalData.httpRequest.headers; }, getMode: () => mode, getNodeParameter: ( parameterName: string, fallbackValue?: any, options?: IGetNodeParameterOptions, ): NodeParameterValueType | object => { const runExecutionData: IRunExecutionData | null = null; const itemIndex = 0; const runIndex = 0; const connectionInputData: INodeExecutionData[] = []; return getNodeParameter( workflow, runExecutionData, runIndex, connectionInputData, node, parameterName, itemIndex, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, null), undefined, fallbackValue, options, ); }, getParamsData(): object { if (additionalData.httpRequest === undefined) { throw new Error('Request is missing!'); } return additionalData.httpRequest.params; }, getQueryData(): object { if (additionalData.httpRequest === undefined) { throw new Error('Request is missing!'); } return additionalData.httpRequest.query; }, getRequestObject(): Request { if (additionalData.httpRequest === undefined) { throw new Error('Request is missing!'); } return additionalData.httpRequest; }, getResponseObject(): Response { if (additionalData.httpResponse === undefined) { throw new Error('Response is missing!'); } return additionalData.httpResponse; }, getNodeWebhookUrl: (name: string): string | undefined => getNodeWebhookUrl( name, workflow, node, additionalData, mode, additionalData.timezone, getAdditionalKeys(additionalData, mode, null), ), getWebhookName: () => webhookData.webhookDescription.name, prepareOutputData: NodeHelpers.prepareOutputData, helpers: { createDeferredPromise, ...getRequestHelperFunctions(workflow, node, additionalData), ...getBinaryHelperFunctions(additionalData), returnJsonArray, }, nodeHelpers: getNodeHelperFunctions(additionalData), }; })(workflow, node); }