mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-18 02:21:13 +00:00
Co-authored-by: Eugene <eugene@n8n.io> Co-authored-by: Daria <daria.staferova@n8n.io> Co-authored-by: Alex Grozav <alex@grozav.com> Co-authored-by: Tomi Turtiainen <10324676+tomi@users.noreply.github.com> Co-authored-by: Guillaume Jacquart <jacquart.guillaume@gmail.com> Co-authored-by: Charlie Kolb <charlie@n8n.io> Co-authored-by: Elias Meire <elias@meire.dev> Co-authored-by: Michael Kret <88898367+michael-radency@users.noreply.github.com> Co-authored-by: shortstacked <declan@n8n.io> Co-authored-by: oleg <me@olegivaniv.com> Co-authored-by: Csaba Tuncsik <csaba@n8n.io> Co-authored-by: Jaakko Husso <jaakko@n8n.io> Co-authored-by: Raúl Gómez Morales <raul00gm@gmail.com> Co-authored-by: Suguru Inoue <suguru@n8n.io> Co-authored-by: Milorad FIlipović <milorad@n8n.io> Co-authored-by: Danny Martini <danny@n8n.io> Co-authored-by: Iván Ovejero <ivov.src@gmail.com> Co-authored-by: RomanDavydchuk <roman.davydchuk@n8n.io> Co-authored-by: Mutasem Aldmour <4711238+mutdmour@users.noreply.github.com> Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: cubic-dev-ai[bot] <191113872+cubic-dev-ai[bot]@users.noreply.github.com> Co-authored-by: Ricardo Espinoza <ricardo@n8n.io> Co-authored-by: Dana <152518854+dana-gill@users.noreply.github.com> Co-authored-by: Michael Kret <michael.k@radency.com> Co-authored-by: Mutasem Aldmour <mutasem@n8n.io>
249 lines
8.4 KiB
TypeScript
249 lines
8.4 KiB
TypeScript
import pick from 'lodash/pick';
|
|
import type { IExecuteFunctions } from 'n8n-workflow';
|
|
import { NodeApiError } from 'n8n-workflow';
|
|
|
|
import { BASE_URL, ERROR_MESSAGES, OPERATION_TIMEOUT } from '../../constants';
|
|
import { apiRequest } from '../../transport';
|
|
import type { IAirtopResponseWithFiles, IAirtopFileInputRequest } from '../../transport/types';
|
|
|
|
/**
|
|
* Fetches all files from the Airtop API using pagination
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param sessionIds - Comma-separated string of session IDs to filter files by
|
|
* @returns Promise resolving to a response object containing the complete array of files
|
|
*/
|
|
export async function requestAllFiles(
|
|
this: IExecuteFunctions,
|
|
sessionIds: string,
|
|
): Promise<IAirtopResponseWithFiles> {
|
|
const endpoint = '/files';
|
|
let hasMore = true;
|
|
let currentOffset = 0;
|
|
const limit = 100;
|
|
const files: IAirtopResponseWithFiles['data']['files'] = [];
|
|
let responseData: IAirtopResponseWithFiles;
|
|
|
|
while (hasMore) {
|
|
// request files
|
|
responseData = (await apiRequest.call(
|
|
this,
|
|
'GET',
|
|
endpoint,
|
|
{},
|
|
{ offset: currentOffset, limit, sessionIds },
|
|
)) as IAirtopResponseWithFiles;
|
|
// add files to the array
|
|
if (responseData.data?.files && Array.isArray(responseData.data?.files)) {
|
|
files.push(...responseData.data.files);
|
|
}
|
|
// check if there are more files
|
|
hasMore = Boolean(responseData.data?.pagination?.hasMore);
|
|
currentOffset += limit;
|
|
}
|
|
|
|
return {
|
|
data: {
|
|
files,
|
|
pagination: {
|
|
hasMore,
|
|
},
|
|
},
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Polls the Airtop API until a file reaches "available" status or times out
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param fileId - The unique identifier of the file to poll
|
|
* @param timeout - Maximum time in milliseconds to wait before failing (defaults to OPERATION_TIMEOUT)
|
|
* @param intervalSeconds - Time in seconds to wait between polling attempts (defaults to 1)
|
|
* @returns Promise resolving to the file ID when the file is available
|
|
* @throws NodeApiError if the operation times out or API request fails
|
|
*/
|
|
export async function pollFileUntilAvailable(
|
|
this: IExecuteFunctions,
|
|
fileId: string,
|
|
timeout = OPERATION_TIMEOUT,
|
|
intervalSeconds = 1,
|
|
): Promise<string> {
|
|
let fileStatus = '';
|
|
const startTime = Date.now();
|
|
|
|
while (fileStatus !== 'available') {
|
|
const elapsedTime = Date.now() - startTime;
|
|
if (elapsedTime >= timeout) {
|
|
throw new NodeApiError(this.getNode(), {
|
|
message: ERROR_MESSAGES.TIMEOUT_REACHED,
|
|
code: 500,
|
|
});
|
|
}
|
|
|
|
const response = await apiRequest.call(this, 'GET', `/files/${fileId}`);
|
|
fileStatus = response.data?.status as string;
|
|
|
|
// Wait before the next polling attempt
|
|
await new Promise((resolve) => setTimeout(resolve, intervalSeconds * 1000));
|
|
}
|
|
|
|
return fileId;
|
|
}
|
|
|
|
/**
|
|
* Creates a file entry in Airtop, uploads the file content, and waits until processing completes
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param fileName - Name to assign to the uploaded file
|
|
* @param fileBuffer - Buffer containing the binary file data to upload
|
|
* @param fileType - Classification of the file in Airtop (e.g., 'customer_upload')
|
|
* @param pollingFunction - Function to use for checking file availability (defaults to pollFileUntilAvailable)
|
|
* @returns Promise resolving to the file ID once upload is complete and file is available
|
|
* @throws NodeApiError if file creation, upload, or polling fails
|
|
*/
|
|
export async function createAndUploadFile(
|
|
this: IExecuteFunctions,
|
|
fileName: string,
|
|
fileBuffer: Buffer,
|
|
fileType: string,
|
|
pollingFunction = pollFileUntilAvailable,
|
|
): Promise<string> {
|
|
// Create file entry
|
|
const createResponse = await apiRequest.call(this, 'POST', '/files', { fileName, fileType });
|
|
|
|
const fileId = createResponse.data?.id;
|
|
const uploadUrl = createResponse.data?.uploadUrl as string;
|
|
|
|
if (!fileId || !uploadUrl) {
|
|
throw new NodeApiError(this.getNode(), {
|
|
message: 'Failed to create file entry: missing file ID or upload URL',
|
|
code: 500,
|
|
});
|
|
}
|
|
|
|
// Upload the file
|
|
await this.helpers.httpRequest({
|
|
method: 'PUT',
|
|
url: uploadUrl,
|
|
body: fileBuffer,
|
|
headers: {
|
|
'Content-Type': 'application/octet-stream',
|
|
},
|
|
});
|
|
|
|
// Poll until the file is available
|
|
return await pollingFunction.call(this, fileId as string);
|
|
}
|
|
|
|
/**
|
|
* Waits for a file to be ready in a session by polling file's information
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param sessionId - ID of the session to check for file availability
|
|
* @param fileId - ID of the file
|
|
* @param timeout - Maximum time in milliseconds to wait before failing (defaults to OPERATION_TIMEOUT)
|
|
* @returns Promise that resolves when a file in the session becomes available
|
|
* @throws NodeApiError if the timeout is reached before a file becomes available
|
|
*/
|
|
export async function waitForFileInSession(
|
|
this: IExecuteFunctions,
|
|
sessionId: string,
|
|
fileId: string,
|
|
timeout = OPERATION_TIMEOUT,
|
|
): Promise<void> {
|
|
const url = `${BASE_URL}/files/${fileId}`;
|
|
const isFileInSession = async (): Promise<boolean> => {
|
|
const fileInfo = (await apiRequest.call(this, 'GET', url)) as IAirtopResponseWithFiles;
|
|
return Boolean(fileInfo.data?.sessionIds?.includes(sessionId));
|
|
};
|
|
|
|
const startTime = Date.now();
|
|
while (!(await isFileInSession())) {
|
|
const elapsedTime = Date.now() - startTime;
|
|
// throw error if timeout is reached
|
|
if (elapsedTime >= timeout) {
|
|
throw new NodeApiError(this.getNode(), {
|
|
message: ERROR_MESSAGES.TIMEOUT_REACHED,
|
|
code: 500,
|
|
});
|
|
}
|
|
// wait 1 second before checking again
|
|
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Associates a file with a session and waits until the file is ready for use
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param fileId - ID of the file to associate with the session
|
|
* @param sessionId - ID of the session to add the file to
|
|
* @param pollingFunction - Function to use for checking file availability in session (defaults to waitForFileInSession)
|
|
* @returns Promise that resolves when the file is ready for use in the session
|
|
*/
|
|
export async function pushFileToSession(
|
|
this: IExecuteFunctions,
|
|
fileId: string,
|
|
sessionId: string,
|
|
pollingFunction = waitForFileInSession,
|
|
): Promise<void> {
|
|
// Push file into session
|
|
await apiRequest.call(this, 'POST', `/files/${fileId}/push`, { sessionIds: [sessionId] });
|
|
await pollingFunction.call(this, sessionId, fileId);
|
|
}
|
|
|
|
/**
|
|
* Activates a file upload input in a specific window within a session
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param fileId - ID of the file to use for the input
|
|
* @param windowId - ID of the window where the file input will be triggered
|
|
* @param sessionId - ID of the session containing the window
|
|
* @returns Promise that resolves when the file input has been triggered
|
|
*/
|
|
export async function triggerFileInput(
|
|
this: IExecuteFunctions,
|
|
request: IAirtopFileInputRequest,
|
|
): Promise<void> {
|
|
await apiRequest.call(
|
|
this,
|
|
'POST',
|
|
`/sessions/${request.sessionId}/windows/${request.windowId}/file-input`,
|
|
pick(request, ['fileId', 'elementDescription', 'includeHiddenElements']),
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Creates a file Buffer from either a URL or binary data
|
|
* This function supports two source types:
|
|
* - URL: Downloads the file from the specified URL and returns it as a Buffer
|
|
* - Binary: Retrieves binary data from the workflow's binary data storage
|
|
*
|
|
* @param this - The execution context providing access to n8n functionality
|
|
* @param source - Source type, either 'url' or 'binary'
|
|
* @param value - Either a URL string or binary data property name depending on source type
|
|
* @param itemIndex - Index of the workflow item to get binary data from (when source is 'binary')
|
|
* @returns Promise resolving to a Buffer containing the file data
|
|
* @throws NodeApiError if the source type is unsupported or retrieval fails
|
|
*/
|
|
export async function createFileBuffer(
|
|
this: IExecuteFunctions,
|
|
source: string,
|
|
value: string,
|
|
itemIndex: number,
|
|
): Promise<Buffer> {
|
|
if (source === 'url') {
|
|
const buffer = (await this.helpers.httpRequest({
|
|
url: value,
|
|
json: false,
|
|
encoding: 'arraybuffer',
|
|
})) as Buffer;
|
|
|
|
return buffer;
|
|
}
|
|
|
|
if (source === 'binary') {
|
|
const binaryData = await this.helpers.getBinaryDataBuffer(itemIndex, value);
|
|
return binaryData;
|
|
}
|
|
|
|
throw new NodeApiError(this.getNode(), {
|
|
message: `Unsupported source type: ${source}. Please use 'url' or 'binary'`,
|
|
code: 500,
|
|
});
|
|
}
|