From 744b8aa8073689335bbaa0f182255f63fb36655e Mon Sep 17 00:00:00 2001 From: Zijian Liu Date: Tue, 24 Nov 2020 14:30:26 +0800 Subject: [PATCH] fs: pass ERR_DIR_CLOSED asynchronously to dir.close Pass the error to the callback or returns a rejected Promise instead of throwing a synchonous error. Fixes: https://github.com/nodejs/node/issues/36237 PR-URL: https://github.com/nodejs/node/pull/36243 Reviewed-By: Antoine du Hamel Reviewed-By: Joyee Cheung Reviewed-By: Rich Trott Reviewed-By: Yongsheng Zhang --- lib/internal/fs/dir.js | 19 ++++++++++++++----- test/parallel/test-fs-opendir.js | 21 ++++++++++++++++++--- 2 files changed, 32 insertions(+), 8 deletions(-) diff --git a/lib/internal/fs/dir.js b/lib/internal/fs/dir.js index d3e84500936176..6eeeb84bda9d70 100644 --- a/lib/internal/fs/dir.js +++ b/lib/internal/fs/dir.js @@ -6,6 +6,7 @@ const { ArrayPrototypeSplice, FunctionPrototypeBind, ObjectDefineProperty, + PromiseReject, Symbol, SymbolAsyncIterator, } = primordials; @@ -164,16 +165,24 @@ class Dir { } close(callback) { - if (this[kDirClosed] === true) { - throw new ERR_DIR_CLOSED(); - } - + // Promise if (callback === undefined) { + if (this[kDirClosed] === true) { + return PromiseReject(new ERR_DIR_CLOSED()); + } return this[kDirClosePromisified](); - } else if (typeof callback !== 'function') { + } + + // callback + if (typeof callback !== 'function') { throw new ERR_INVALID_CALLBACK(callback); } + if (this[kDirClosed] === true) { + process.nextTick(callback, new ERR_DIR_CLOSED()); + return; + } + if (this[kDirOperationQueue] !== null) { this[kDirOperationQueue].push(() => { this.close(callback); diff --git a/test/parallel/test-fs-opendir.js b/test/parallel/test-fs-opendir.js index ab7b24b63e76b2..1064bd79f44409 100644 --- a/test/parallel/test-fs-opendir.js +++ b/test/parallel/test-fs-opendir.js @@ -223,12 +223,11 @@ async function doAsyncIterInvalidCallbackTest() { } doAsyncIterInvalidCallbackTest().then(common.mustCall()); -// Check if directory already closed - throw an exception +// Check first call to close() - should not report an error. async function doAsyncIterDirClosedTest() { const dir = await fs.promises.opendir(testDir); await dir.close(); - - assert.throws(() => dir.close(), dirclosedError); + await assert.rejects(() => dir.close(), dirclosedError); } doAsyncIterDirClosedTest().then(common.mustCall()); @@ -267,3 +266,19 @@ async function doConcurrentAsyncMixedOps() { await promise2; } doConcurrentAsyncMixedOps().then(common.mustCall()); + +// Check if directory already closed - the callback should pass an error. +{ + const dir = fs.opendirSync(testDir); + dir.closeSync(); + dir.close(common.mustCall((error) => { + assert.strictEqual(error.code, dirclosedError.code); + })); +} + +// Check if directory already closed - throw an promise exception. +{ + const dir = fs.opendirSync(testDir); + dir.closeSync(); + assert.rejects(dir.close(), dirclosedError).then(common.mustCall()); +}