n8n/packages/nodes-base/nodes/Google/Drive/v2/helpers/utils.ts

160 lines
3.9 KiB
TypeScript

import type { Readable } from 'stream';
import type { IDataObject, IExecuteFunctions } from 'n8n-workflow';
import { BINARY_ENCODING, NodeOperationError } from 'n8n-workflow';
import { RLC_DRIVE_DEFAULT, RLC_FOLDER_DEFAULT, UPLOAD_CHUNK_SIZE } from './interfaces';
export function prepareQueryString(fields: string[] | undefined) {
let queryFields = 'id, name';
if (fields) {
if (fields.includes('*')) {
queryFields = '*';
} else {
queryFields = fields.join(', ');
}
}
return queryFields;
}
export async function getItemBinaryData(
this: IExecuteFunctions,
inputDataFieldName: string,
i: number,
chunkSize = UPLOAD_CHUNK_SIZE,
) {
let contentLength: number;
let fileContent: Buffer | Readable;
let originalFilename: string | undefined;
let mimeType;
if (!inputDataFieldName) {
throw new NodeOperationError(
this.getNode(),
'The name of the input field containing the binary file data must be set',
{
itemIndex: i,
},
);
}
const binaryData = this.helpers.assertBinaryData(i, inputDataFieldName);
if (binaryData.id) {
// Stream data in 256KB chunks, and upload the via the resumable upload api
fileContent = await this.helpers.getBinaryStream(binaryData.id, chunkSize);
const metadata = await this.helpers.getBinaryMetadata(binaryData.id);
contentLength = metadata.fileSize;
originalFilename = metadata.fileName;
if (metadata.mimeType) mimeType = binaryData.mimeType;
} else {
fileContent = Buffer.from(binaryData.data, BINARY_ENCODING);
contentLength = fileContent.length;
originalFilename = binaryData.fileName;
mimeType = binaryData.mimeType;
}
return {
contentLength,
fileContent,
originalFilename,
mimeType,
};
}
export function setFileProperties(body: IDataObject, options: IDataObject) {
if (options.propertiesUi) {
const values = ((options.propertiesUi as IDataObject).propertyValues as IDataObject[]) || [];
body.properties = values.reduce(
(acc, value) => Object.assign(acc, { [`${value.key}`]: value.value }),
{} as IDataObject,
);
}
if (options.appPropertiesUi) {
const values =
((options.appPropertiesUi as IDataObject).appPropertyValues as IDataObject[]) || [];
body.appProperties = values.reduce(
(acc, value) => Object.assign(acc, { [`${value.key}`]: value.value }),
{} as IDataObject,
);
}
return body;
}
export function setUpdateCommonParams(qs: IDataObject, options: IDataObject) {
if (options.keepRevisionForever) {
qs.keepRevisionForever = options.keepRevisionForever;
}
if (options.ocrLanguage) {
qs.ocrLanguage = options.ocrLanguage;
}
if (options.useContentAsIndexableText) {
qs.useContentAsIndexableText = options.useContentAsIndexableText;
}
return qs;
}
export function updateDriveScopes(
qs: IDataObject,
driveId: string,
defaultDrive = RLC_DRIVE_DEFAULT,
) {
if (driveId) {
if (driveId === defaultDrive) {
qs.includeItemsFromAllDrives = false;
qs.supportsAllDrives = false;
qs.spaces = 'appDataFolder, drive';
qs.corpora = 'user';
} else {
qs.driveId = driveId;
qs.corpora = 'drive';
}
}
}
export function setParentFolder(
folderId: string,
driveId: string,
folderIdDefault = RLC_FOLDER_DEFAULT,
driveIdDefault = RLC_DRIVE_DEFAULT,
) {
if (folderId !== folderIdDefault) {
return folderId;
} else if (driveId && driveId !== driveIdDefault) {
return driveId;
} else {
return 'root';
}
}
export async function processInChunks(
stream: Readable,
chunkSize: number,
process: (chunk: Buffer, offset: number) => void | Promise<void>,
) {
let buffer = Buffer.alloc(0);
let offset = 0;
for await (const chunk of stream) {
buffer = Buffer.concat([buffer, chunk]);
while (buffer.length >= chunkSize) {
const chunkToProcess = buffer.subarray(0, chunkSize);
await process(chunkToProcess, offset);
buffer = buffer.subarray(chunkSize);
offset += chunkSize;
}
}
// Process last chunk, could be smaller than chunkSize
if (buffer.length > 0) {
await process(buffer, offset);
}
}