From 75e0a3e2b02ad59ce7e32b4431976d8dad46b09c Mon Sep 17 00:00:00 2001 From: Alek Date: Fri, 30 Aug 2024 12:21:02 -0600 Subject: [PATCH] fix(builtin): patched methods accept Buffer and URL types The node specification for `fs` methods lists possible types for `path` as string | Buffer | URL. However, because the `path.resolve` and `path.dirname` methods only accept strings, the patched `fs` methods errored if the path arg was a Buffer or URL. This would occur when calling `fs.rm`, for example. --- internal/node/node_patches.js | 51 +++++++++++++------- internal/node/test/BUILD.bazel | 9 ++++ internal/node/test/patches.spec.js | 76 ++++++++++++++++++++++++++++++ 3 files changed, 120 insertions(+), 16 deletions(-) create mode 100644 internal/node/test/patches.spec.js diff --git a/internal/node/node_patches.js b/internal/node/node_patches.js index 360bff24f9..dad5facbb1 100644 --- a/internal/node/node_patches.js +++ b/internal/node/node_patches.js @@ -57,6 +57,15 @@ var __asyncGenerator = (commonjsGlobal && commonjsGlobal.__asyncGenerator) || fu Object.defineProperty(exports, "__esModule", { value: true }); exports.escapeFunction = exports.isOutPath = exports.patcher = void 0; +function stringifyPath(path) { + if (path instanceof Buffer) { + return path.toString(); + } else if (path instanceof URL) { + return path.toString().replace('file://', ''); + } else { + return path; + } +} // using require here on purpose so we can override methods with any // also even though imports are mutable in typescript the cognitive dissonance is too high because @@ -95,7 +104,8 @@ const patcher = (fs = fs__default['default'], roots) => { args[args.length - 1] = (err, stats) => { if (err) return cb(err); - path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + path__default['default'].resolve(pathString); if (!stats.isSymbolicLink()) { return cb(null, stats); } @@ -116,7 +126,7 @@ const patcher = (fs = fs__default['default'], roots) => { return cb(err); } } - str = path__default['default'].resolve(path__default['default'].dirname(args[0]), str); + str = path__default['default'].resolve(path__default['default'].dirname(pathString), str); if (isEscape(str, args[0])) { // if it's an out link we have to return the original stat. return origStat(args[0], (err, plainStat) => { @@ -142,8 +152,9 @@ const patcher = (fs = fs__default['default'], roots) => { args[args.length - 1] = (err, str) => { if (err) return cb(err); - if (isEscape(str, args[0])) { - cb(null, path__default['default'].resolve(args[0])); + const pathString = stringifyPath(args[0]); + if (isEscape(str, pathString)) { + cb(null, path__default['default'].resolve(pathString)); } else { cb(null, str); @@ -160,8 +171,9 @@ const patcher = (fs = fs__default['default'], roots) => { args[args.length - 1] = (err, str) => { if (err) return cb(err); - if (isEscape(str, args[0])) { - cb(null, path__default['default'].resolve(args[0])); + const pathString = stringifyPath(args[0]); + if (isEscape(str, pathString)) { + cb(null, path__default['default'].resolve(pathString)); } else { cb(null, str); @@ -176,7 +188,8 @@ const patcher = (fs = fs__default['default'], roots) => { if (cb) { cb = once(cb); args[args.length - 1] = (err, str) => { - args[0] = path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + args[0] = path__default['default'].resolve(pathString); if (str) str = path__default['default'].resolve(path__default['default'].dirname(args[0]), str); if (err) @@ -196,13 +209,14 @@ const patcher = (fs = fs__default['default'], roots) => { // tslint:disable-next-line:no-any fs.lstatSync = (...args) => { const stats = origLstatSync(...args); - const linkPath = path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + const linkPath = path__default['default'].resolve(pathString); if (!stats.isSymbolicLink()) { return stats; } let linkTarget; try { - linkTarget = path__default['default'].resolve(path__default['default'].dirname(args[0]), origReadlinkSync(linkPath)); + linkTarget = path__default['default'].resolve(path__default['default'].dirname(pathString), origReadlinkSync(linkPath)); } catch (e) { if (e.code === 'ENOENT') { @@ -227,22 +241,25 @@ const patcher = (fs = fs__default['default'], roots) => { // tslint:disable-next-line:no-any fs.realpathSync = (...args) => { const str = origRealpathSync(...args); - if (isEscape(str, args[0])) { - return path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + if (isEscape(str, pathString)) { + return path__default['default'].resolve(pathString); } return str; }; // tslint:disable-next-line:no-any fs.realpathSync.native = (...args) => { const str = origRealpathSyncNative(...args); - if (isEscape(str, args[0])) { - return path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + if (isEscape(str, pathString)) { + return path__default['default'].resolve(pathString); } return str; }; // tslint:disable-next-line:no-any fs.readlinkSync = (...args) => { - args[0] = path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + args[0] = path__default['default'].resolve(pathString); const str = path__default['default'].resolve(path__default['default'].dirname(args[0]), origReadlinkSync(...args)); if (isEscape(str, args[0]) || str === args[0]) { const e = new Error('EINVAL: invalid argument, readlink \'' + args[0] + '\''); @@ -254,7 +271,8 @@ const patcher = (fs = fs__default['default'], roots) => { }; // tslint:disable-next-line:no-any fs.readdir = (...args) => { - const p = path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + const p = path__default['default'].resolve(pathString); let cb = args[args.length - 1]; if (typeof cb !== 'function') { // this will likely throw callback required error. @@ -284,7 +302,8 @@ const patcher = (fs = fs__default['default'], roots) => { // tslint:disable-next-line:no-any fs.readdirSync = (...args) => { const res = origReaddirSync(...args); - const p = path__default['default'].resolve(args[0]); + const pathString = stringifyPath(args[0]); + const p = path__default['default'].resolve(pathString); // tslint:disable-next-line:no-any res.forEach((v) => { handleDirentSync(p, v); diff --git a/internal/node/test/BUILD.bazel b/internal/node/test/BUILD.bazel index 0b0cb8e8cf..f1b924aeb1 100644 --- a/internal/node/test/BUILD.bazel +++ b/internal/node/test/BUILD.bazel @@ -341,6 +341,15 @@ jasmine_node_test( ], ) +jasmine_node_test( + name = "patches_test", + srcs = ["patches.spec.js"], + data = [ + "dir_output", + "minified.js", + ], +) + [nodejs_toolchain_test( name = "nodejs_toolchain_%s_test" % platform, platform = platform, diff --git a/internal/node/test/patches.spec.js b/internal/node/test/patches.spec.js new file mode 100644 index 0000000000..1860cd2d71 --- /dev/null +++ b/internal/node/test/patches.spec.js @@ -0,0 +1,76 @@ +const fs = require('fs'); +const path = require('path'); +const runfiles = require(process.env['BAZEL_NODE_RUNFILES_HELPER']); + +describe('node fs function does not throw:', () => { + const files = runfiles.resolvePackageRelative('dir_output') + const filesBuf = Buffer.from(files); + const filesUrl = new URL('file://' + files); + const doNothingCallback = () => {} + + // listed in order of appearance in the patch file + it('lstat', () => { + expect(() => fs.lstat(files, doNothingCallback)).not.toThrow(); + expect(() => fs.lstat(filesBuf, doNothingCallback)).not.toThrow(); + expect(() => fs.lstat(filesUrl, doNothingCallback)).not.toThrow(); + }); + + it('realpath', () => { + expect(() => fs.realpath(files, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.realpath(filesBuf, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.realpath(filesUrl, undefined, doNothingCallback)).not.toThrow(); + }); + + it('realpath.native', () => { + expect(() => fs.realpath.native(files, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.realpath.native(filesBuf, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.realpath.native(filesUrl, undefined, doNothingCallback)).not.toThrow(); + }); + + it('readlink', () => { + expect(() => fs.readlink(files, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.readlink(filesBuf, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.readlink(filesUrl, undefined, doNothingCallback)).not.toThrow(); + }); + + it('lstatSync', () => { + expect(() => fs.lstatSync(files)).not.toThrow(); + expect(() => fs.lstatSync(filesBuf)).not.toThrow(); + expect(() => fs.lstatSync(filesUrl)).not.toThrow(); + }); + + it('realpathSync', () => { + expect(() => fs.realpathSync(files)).not.toThrow(); + expect(() => fs.realpathSync(filesBuf)).not.toThrow(); + expect(() => fs.realpathSync(filesUrl)).not.toThrow(); + }); + + it('realpathSync.native', () => { + expect(() => fs.realpathSync.native(files)).not.toThrow(); + expect(() => fs.realpathSync.native(filesBuf)).not.toThrow(); + expect(() => fs.realpathSync.native(filesUrl)).not.toThrow(); + }); + + it('readlinkSync', () => { + const symlinkPath = path.join(files, 'symlink'); + fs.symlinkSync(files, symlinkPath); + const symlinkPathBuf = Buffer.from(symlinkPath); + const symlinkPathUrl = new URL(`file://${symlinkPath}`); + expect(() => fs.readlinkSync(symlinkPath)).not.toThrow(); + expect(() => fs.readlinkSync(symlinkPathBuf)).not.toThrow(); + expect(() => fs.readlinkSync(symlinkPathUrl)).not.toThrow(); + }); + + it('readdir', () => { + expect(() => fs.readdir(files, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.readdir(filesBuf, undefined, doNothingCallback)).not.toThrow(); + expect(() => fs.readdir(filesUrl, undefined, doNothingCallback)).not.toThrow(); + }); + + it('readdirSync', () => { + expect(() => fs.readdirSync(files)).not.toThrow(); + expect(() => fs.readdirSync(filesBuf)).not.toThrow(); + expect(() => fs.readdirSync(filesUrl)).not.toThrow(); + }); + +});