mirror of
https://github.com/Abdulazizzn/n8n-enterprise-unlocked.git
synced 2025-12-16 17:46:45 +00:00
219 lines
5.0 KiB
TypeScript
219 lines
5.0 KiB
TypeScript
import FormData from 'form-data';
|
|
|
|
import type {
|
|
IDataObject,
|
|
IExecuteFunctions,
|
|
INodeExecutionData,
|
|
INodeProperties,
|
|
} from 'n8n-workflow';
|
|
|
|
import { updateDisplayOptions } from '@utils/utilities';
|
|
|
|
import {
|
|
getItemBinaryData,
|
|
setFileProperties,
|
|
setUpdateCommonParams,
|
|
setParentFolder,
|
|
processInChunks,
|
|
} from '../../helpers/utils';
|
|
import { googleApiRequest } from '../../transport';
|
|
import { driveRLC, folderRLC, updateCommonOptions } from '../common.descriptions';
|
|
|
|
const properties: INodeProperties[] = [
|
|
{
|
|
displayName: 'Input Data Field Name',
|
|
name: 'inputDataFieldName',
|
|
type: 'string',
|
|
placeholder: '“e.g. data',
|
|
default: 'data',
|
|
required: true,
|
|
hint: 'The name of the input field containing the binary file data to update the file',
|
|
description:
|
|
'Find the name of input field containing the binary data to update the file in the Input panel on the left, in the Binary tab',
|
|
},
|
|
{
|
|
displayName: 'File Name',
|
|
name: 'name',
|
|
type: 'string',
|
|
default: '',
|
|
placeholder: 'e.g. My New File',
|
|
description: 'If not specified, the original file name will be used',
|
|
},
|
|
{
|
|
...driveRLC,
|
|
displayName: 'Parent Drive',
|
|
description: 'The drive where to upload the file',
|
|
},
|
|
{
|
|
...folderRLC,
|
|
displayName: 'Parent Folder',
|
|
description: 'The folder where to upload the file',
|
|
},
|
|
{
|
|
displayName: 'Options',
|
|
name: 'options',
|
|
type: 'collection',
|
|
placeholder: 'Add option',
|
|
default: {},
|
|
options: [
|
|
...updateCommonOptions,
|
|
{
|
|
displayName: 'Simplify Output',
|
|
name: 'simplifyOutput',
|
|
type: 'boolean',
|
|
default: true,
|
|
description: 'Whether to return a simplified version of the response instead of all fields',
|
|
},
|
|
],
|
|
},
|
|
];
|
|
|
|
const displayOptions = {
|
|
show: {
|
|
resource: ['file'],
|
|
operation: ['upload'],
|
|
},
|
|
};
|
|
|
|
export const description = updateDisplayOptions(displayOptions, properties);
|
|
|
|
export async function execute(this: IExecuteFunctions, i: number): Promise<INodeExecutionData[]> {
|
|
const returnData: INodeExecutionData[] = [];
|
|
|
|
const inputDataFieldName = this.getNodeParameter('inputDataFieldName', i) as string;
|
|
|
|
const { contentLength, fileContent, originalFilename, mimeType } = await getItemBinaryData.call(
|
|
this,
|
|
inputDataFieldName,
|
|
i,
|
|
);
|
|
|
|
const name = (this.getNodeParameter('name', i) as string) || originalFilename;
|
|
|
|
const driveId = this.getNodeParameter('driveId', i, undefined, {
|
|
extractValue: true,
|
|
}) as string;
|
|
|
|
const folderId = this.getNodeParameter('folderId', i, undefined, {
|
|
extractValue: true,
|
|
}) as string;
|
|
|
|
let uploadId;
|
|
const metadata = {
|
|
name,
|
|
parents: [setParentFolder(folderId, driveId)],
|
|
};
|
|
if (Buffer.isBuffer(fileContent)) {
|
|
const multiPartBody = new FormData();
|
|
multiPartBody.append('metadata', JSON.stringify(metadata), {
|
|
contentType: 'application/json',
|
|
});
|
|
multiPartBody.append('data', fileContent, {
|
|
contentType: mimeType,
|
|
knownLength: contentLength,
|
|
});
|
|
|
|
const response = await googleApiRequest.call(
|
|
this,
|
|
'POST',
|
|
'/upload/drive/v3/files',
|
|
multiPartBody.getBuffer(),
|
|
{
|
|
uploadType: 'multipart',
|
|
supportsAllDrives: true,
|
|
},
|
|
undefined,
|
|
{
|
|
headers: {
|
|
'Content-Type': `multipart/related; boundary=${multiPartBody.getBoundary()}`,
|
|
'Content-Length': multiPartBody.getLengthSync(),
|
|
},
|
|
},
|
|
);
|
|
|
|
uploadId = response.id;
|
|
} else {
|
|
const resumableUpload = await googleApiRequest.call(
|
|
this,
|
|
'POST',
|
|
'/upload/drive/v3/files',
|
|
metadata,
|
|
{
|
|
uploadType: 'resumable',
|
|
supportsAllDrives: true,
|
|
},
|
|
undefined,
|
|
{
|
|
returnFullResponse: true,
|
|
headers: {
|
|
'X-Upload-Content-Type': mimeType,
|
|
},
|
|
},
|
|
);
|
|
|
|
const uploadUrl = resumableUpload.headers.location;
|
|
|
|
// 2MB chunks, needs to be a multiple of 256kB for Google Drive API
|
|
const chunkSizeBytes = 2048 * 1024;
|
|
|
|
await processInChunks(fileContent, chunkSizeBytes, async (chunk, offset) => {
|
|
try {
|
|
const response = await this.helpers.httpRequest({
|
|
method: 'PUT',
|
|
url: uploadUrl,
|
|
headers: {
|
|
'Content-Length': chunk.length,
|
|
'Content-Range': `bytes ${offset}-${offset + chunk.byteLength - 1}/${contentLength}`,
|
|
},
|
|
body: chunk,
|
|
});
|
|
uploadId = response?.id;
|
|
} catch (error) {
|
|
if (error.response?.status !== 308) throw error;
|
|
}
|
|
});
|
|
}
|
|
|
|
const options = this.getNodeParameter('options', i, {});
|
|
|
|
const qs = setUpdateCommonParams(
|
|
{
|
|
addParents: setParentFolder(folderId, driveId),
|
|
includeItemsFromAllDrives: true,
|
|
supportsAllDrives: true,
|
|
spaces: 'appDataFolder, drive',
|
|
corpora: 'allDrives',
|
|
},
|
|
options,
|
|
);
|
|
|
|
if (!options.simplifyOutput) {
|
|
qs.fields = '*';
|
|
}
|
|
|
|
const body = setFileProperties(
|
|
{
|
|
mimeType,
|
|
name,
|
|
originalFilename,
|
|
},
|
|
options,
|
|
);
|
|
|
|
const response = await googleApiRequest.call(
|
|
this,
|
|
'PATCH',
|
|
`/drive/v3/files/${uploadId}`,
|
|
body,
|
|
qs,
|
|
);
|
|
|
|
const executionData = this.helpers.constructExecutionMetaData(
|
|
this.helpers.returnJsonArray(response as IDataObject[]),
|
|
{ itemData: { item: i } },
|
|
);
|
|
returnData.push(...executionData);
|
|
|
|
return returnData;
|
|
}
|