diff --git a/src/file.ts b/src/file.ts index d949106a0..6eed2cabd 100644 --- a/src/file.ts +++ b/src/file.ts @@ -29,7 +29,14 @@ import * as crypto from 'crypto'; import * as fs from 'fs'; import * as mime from 'mime'; import * as resumableUpload from './resumable-upload'; -import {Writable, Readable, pipeline, Transform, PassThrough} from 'stream'; +import { + Writable, + Readable, + pipeline, + Transform, + PassThrough, + PipelineSource, +} from 'stream'; import * as zlib from 'zlib'; import * as http from 'http'; @@ -93,6 +100,8 @@ export interface PolicyDocument { signature: string; } +export type SaveData = string | Buffer | PipelineSource; + export type GenerateSignedPostPolicyV2Response = [PolicyDocument]; export interface GenerateSignedPostPolicyV2Callback { @@ -454,6 +463,7 @@ export enum FileExceptionMessages { UPLOAD_MISMATCH = `The uploaded data did not match the data from the server. As a precaution, the file has been deleted. To be sure the content is the same, you should try uploading the file again.`, + STREAM_NOT_READABLE = 'Stream must be readable.', } /** @@ -3557,13 +3567,9 @@ class File extends ServiceObject { this.copy(newFile, copyOptions, callback!); } - save(data: string | Buffer, options?: SaveOptions): Promise; - save(data: string | Buffer, callback: SaveCallback): void; - save( - data: string | Buffer, - options: SaveOptions, - callback: SaveCallback - ): void; + save(data: SaveData, options?: SaveOptions): Promise; + save(data: SaveData, callback: SaveCallback): void; + save(data: SaveData, options: SaveOptions, callback: SaveCallback): void; /** * @typedef {object} SaveOptions * @extends CreateWriteStreamOptions @@ -3590,7 +3596,7 @@ class File extends ServiceObject { * resumable feature is disabled. *

* - * @param {string | Buffer} data The data to write to a file. + * @param {SaveData} data The data to write to a file. * @param {SaveOptions} [options] See {@link File#createWriteStream}'s `options` * parameter. * @param {SaveCallback} [callback] Callback function. @@ -3618,7 +3624,7 @@ class File extends ServiceObject { * ``` */ save( - data: string | Buffer, + data: SaveData, optionsOrCallback?: SaveOptions | SaveCallback, callback?: SaveCallback ): Promise | void { @@ -3638,28 +3644,68 @@ class File extends ServiceObject { } const returnValue = retry( async (bail: (err: Error) => void) => { - await new Promise((resolve, reject) => { + if (data instanceof Readable) { + // Make sure any pending async readable operations are finished before + // attempting to check if the stream is readable. + await new Promise(resolve => setImmediate(resolve)); + + if (!data.readable || data.destroyed) { + // Calling pipeline() with a non-readable stream will result in the + // callback being called without an error, and no piping taking + // place. In that case, file.save() would appear to succeed, but + // nothing would be uploaded. + return bail(new Error(FileExceptionMessages.STREAM_NOT_READABLE)); + } + } + + return new Promise((resolve, reject) => { if (maxRetries === 0) { this.storage.retryOptions.autoRetry = false; } - const writable = this.createWriteStream(options) - .on('error', err => { - if ( - this.storage.retryOptions.autoRetry && - this.storage.retryOptions.retryableErrorFn!(err) - ) { - return reject(err); + const writable = this.createWriteStream(options); + + if (options.onUploadProgress) { + writable.on('progress', options.onUploadProgress); + } + + const handleError = (err: Error) => { + if ( + !this.storage.retryOptions.autoRetry || + !this.storage.retryOptions.retryableErrorFn!(err) + ) { + bail(err); + } + + reject(err); + }; + + if (typeof data === 'string' || Buffer.isBuffer(data)) { + writable + .on('error', handleError) + .on('finish', () => resolve()) + .end(data); + } else { + pipeline(data, writable, err => { + if (err) { + // If data is not a valid PipelineSource, then pipeline will + // fail without destroying the writable stream. If data is a + // PipelineSource that yields invalid chunks (e.g. a stream in + // object mode or an iterable that does not yield Buffers or + // strings), then pipeline will destroy the writable stream. + if (!writable.destroyed) writable.destroy(); + + if (typeof data !== 'function') { + // Only PipelineSourceFunction can be retried. Async-iterables + // and Readable streams can only be consumed once. + bail(err); + } + + handleError(err); } else { - return bail(err); + resolve(); } - }) - .on('finish', () => { - return resolve(); }); - if (options.onUploadProgress) { - writable.on('progress', options.onUploadProgress); } - writable.end(data); }); }, { diff --git a/test/file.ts b/test/file.ts index aa050254e..567cf77d4 100644 --- a/test/file.ts +++ b/test/file.ts @@ -4275,6 +4275,197 @@ describe('File', () => { await file.save(DATA, options, assert.ifError); }); + it('should save a Readable with no errors', done => { + const options = {resumable: false}; + file.createWriteStream = () => { + const writeStream = new PassThrough(); + writeStream.on('data', data => { + assert.strictEqual(data.toString(), DATA); + }); + writeStream.once('finish', done); + return writeStream; + }; + + const readable = new Readable({ + read() { + this.push(DATA); + this.push(null); + }, + }); + + void file.save(readable, options); + }); + + it('should propagate Readable errors', done => { + const options = {resumable: false}; + file.createWriteStream = () => { + const writeStream = new PassThrough(); + let errorCalled = false; + writeStream.on('data', data => { + assert.strictEqual(data.toString(), DATA); + }); + writeStream.on('error', err => { + errorCalled = true; + assert.strictEqual(err.message, 'Error!'); + }); + writeStream.on('finish', () => { + assert.ok(errorCalled); + }); + return writeStream; + }; + + const readable = new Readable({ + read() { + setTimeout(() => { + this.push(DATA); + this.destroy(new Error('Error!')); + }, 50); + }, + }); + + file.save(readable, options, (err: Error) => { + assert.strictEqual(err.message, 'Error!'); + done(); + }); + }); + + it('Readable upload should not retry', async () => { + const options = {resumable: false}; + + let retryCount = 0; + + file.createWriteStream = () => { + retryCount++; + return new Transform({ + transform( + chunk: string | Buffer, + _encoding: string, + done: Function + ) { + this.push(chunk); + setTimeout(() => { + done(new HTTPError('retryable error', 408)); + }, 5); + }, + }); + }; + try { + const readable = new Readable({ + read() { + this.push(DATA); + this.push(null); + }, + }); + + await file.save(readable, options); + throw Error('unreachable'); + } catch (e) { + assert.strictEqual((e as Error).message, 'retryable error'); + assert.ok(retryCount === 1); + } + }); + + it('Destroyed Readable upload should throw', async () => { + const options = {resumable: false}; + + file.createWriteStream = () => { + throw new Error('unreachable'); + }; + try { + const readable = new Readable({ + read() { + this.push(DATA); + this.push(null); + }, + }); + + readable.destroy(); + + await file.save(readable, options); + } catch (e) { + assert.strictEqual( + (e as Error).message, + FileExceptionMessages.STREAM_NOT_READABLE + ); + } + }); + + it('should save a generator with no error', done => { + const options = {resumable: false}; + file.createWriteStream = () => { + const writeStream = new PassThrough(); + writeStream.on('data', data => { + assert.strictEqual(data.toString(), DATA); + done(); + }); + return writeStream; + }; + + const generator = async function* (arg?: {signal?: AbortSignal}) { + await new Promise(resolve => setTimeout(resolve, 5)); + if (arg?.signal?.aborted) return; + yield DATA; + }; + + void file.save(generator, options); + }); + + it('should propagate async iterable errors', done => { + const options = {resumable: false}; + file.createWriteStream = () => { + const writeStream = new PassThrough(); + let errorCalled = false; + writeStream.on('data', data => { + assert.strictEqual(data.toString(), DATA); + }); + writeStream.on('error', err => { + errorCalled = true; + assert.strictEqual(err.message, 'Error!'); + }); + writeStream.on('finish', () => { + assert.ok(errorCalled); + }); + return writeStream; + }; + + const generator = async function* () { + yield DATA; + throw new Error('Error!'); + }; + + file.save(generator(), options, (err: Error) => { + assert.strictEqual(err.message, 'Error!'); + done(); + }); + }); + + it('should error on invalid async iterator data', done => { + const options = {resumable: false}; + file.createWriteStream = () => { + const writeStream = new PassThrough(); + let errorCalled = false; + writeStream.on('error', () => { + errorCalled = true; + }); + writeStream.on('finish', () => { + assert.ok(errorCalled); + }); + return writeStream; + }; + + const generator = async function* () { + yield {thisIsNot: 'a buffer or a string'}; + }; + + file.save(generator(), options, (err: Error) => { + assert.strictEqual( + err.message, + 'The "chunk" argument must be of type string or an instance of Buffer or Uint8Array. Received an instance of Object' + ); + done(); + }); + }); + it('buffer upload should retry on first failure', async () => { const options = { resumable: false,