Skip to content

Commit 00b2f07

Browse files
fs,win: fix bug in paths with trailing slashes
Fixes: #17801 Refs: #33831 PR-URL: #54160 Reviewed-By: Matteo Collina <[email protected]> Reviewed-By: Yagiz Nizipli <[email protected]>
1 parent 4a3fffa commit 00b2f07

File tree

5 files changed

+237
-20
lines changed

5 files changed

+237
-20
lines changed

lib/fs.js

+17-9
Original file line numberDiff line numberDiff line change
@@ -385,7 +385,11 @@ function readFile(path, options, callback) {
385385
const req = new FSReqCallback();
386386
req.context = context;
387387
req.oncomplete = readFileAfterOpen;
388-
binding.open(getValidatedPath(path), flagsNumber, 0o666, req);
388+
binding.open(
389+
getValidatedPath(path, 'path', { expectFile: true, syscall: 'read' }),
390+
flagsNumber,
391+
0o666,
392+
req);
389393
}
390394

391395
function tryStatSync(fd, isUserFd) {
@@ -437,7 +441,9 @@ function readFileSync(path, options) {
437441

438442
if (options.encoding === 'utf8' || options.encoding === 'utf-8') {
439443
if (!isInt32(path)) {
440-
path = getValidatedPath(path);
444+
path = getValidatedPath(path,
445+
'path',
446+
{ expectFile: true, syscall: 'read' });
441447
}
442448
return binding.readFileUtf8(path, stringToFlags(options.flag));
443449
}
@@ -531,7 +537,7 @@ function closeSync(fd) {
531537
* @returns {void}
532538
*/
533539
function open(path, flags, mode, callback) {
534-
path = getValidatedPath(path);
540+
path = getValidatedPath(path, 'path', { expectFile: true, syscall: 'open' });
535541
if (arguments.length < 3) {
536542
callback = flags;
537543
flags = 'r';
@@ -560,7 +566,7 @@ function open(path, flags, mode, callback) {
560566
*/
561567
function openSync(path, flags, mode) {
562568
return binding.open(
563-
getValidatedPath(path),
569+
getValidatedPath(path, 'path', { expectFile: true, syscall: 'open' }),
564570
stringToFlags(flags),
565571
parseFileMode(mode, 'mode', 0o666),
566572
);
@@ -2339,7 +2345,9 @@ function writeFileSync(path, data, options) {
23392345
// C++ fast path for string data and UTF8 encoding
23402346
if (typeof data === 'string' && (options.encoding === 'utf8' || options.encoding === 'utf-8')) {
23412347
if (!isInt32(path)) {
2342-
path = getValidatedPath(path);
2348+
path = getValidatedPath(path,
2349+
'path',
2350+
{ expectFile: true, syscall: 'write' });
23432351
}
23442352

23452353
return binding.writeFileUtf8(
@@ -2984,8 +2992,8 @@ function copyFile(src, dest, mode, callback) {
29842992
mode = 0;
29852993
}
29862994

2987-
src = getValidatedPath(src, 'src');
2988-
dest = getValidatedPath(dest, 'dest');
2995+
src = getValidatedPath(src, 'src', { expectFile: true, syscall: 'cp' });
2996+
dest = getValidatedPath(dest, 'dest', { expectFile: true, syscall: 'cp' });
29892997
callback = makeCallback(callback);
29902998

29912999
const req = new FSReqCallback();
@@ -3003,8 +3011,8 @@ function copyFile(src, dest, mode, callback) {
30033011
*/
30043012
function copyFileSync(src, dest, mode) {
30053013
binding.copyFile(
3006-
getValidatedPath(src, 'src'),
3007-
getValidatedPath(dest, 'dest'),
3014+
getValidatedPath(src, 'src', { expectFile: true, syscall: 'cp' }),
3015+
getValidatedPath(dest, 'dest', { expectFile: true, syscall: 'cp' }),
30083016
mode,
30093017
);
30103018
}

lib/internal/fs/promises.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -620,8 +620,8 @@ async function cp(src, dest, options) {
620620
async function copyFile(src, dest, mode) {
621621
return await PromisePrototypeThen(
622622
binding.copyFile(
623-
getValidatedPath(src, 'src'),
624-
getValidatedPath(dest, 'dest'),
623+
getValidatedPath(src, 'src', { expectFile: true, syscall: 'cp' }),
624+
getValidatedPath(dest, 'dest', { expectFile: true, syscall: 'cp' }),
625625
mode,
626626
kUsePromises,
627627
),
@@ -633,7 +633,7 @@ async function copyFile(src, dest, mode) {
633633
// Note that unlike fs.open() which uses numeric file descriptors,
634634
// fsPromises.open() uses the fs.FileHandle class.
635635
async function open(path, flags, mode) {
636-
path = getValidatedPath(path);
636+
path = getValidatedPath(path, 'path', { expectFile: true, syscall: 'open' });
637637
const flagsNumber = stringToFlags(flags);
638638
mode = parseFileMode(mode, 'mode', 0o666);
639639
return new FileHandle(await PromisePrototypeThen(

lib/internal/fs/streams.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -182,7 +182,7 @@ function ReadStream(path, options) {
182182
this.flags = options.flags === undefined ? 'r' : options.flags;
183183
this.mode = options.mode === undefined ? 0o666 : options.mode;
184184

185-
validatePath(this.path);
185+
validatePath(this.path, 'path', { expectFile: true, syscall: 'read' });
186186
} else {
187187
this.fd = getValidatedFd(importFd(this, options));
188188
}
@@ -337,7 +337,7 @@ function WriteStream(path, options) {
337337
this.flags = options.flags === undefined ? 'w' : options.flags;
338338
this.mode = options.mode === undefined ? 0o666 : options.mode;
339339

340-
validatePath(this.path);
340+
validatePath(this.path, 'path', { expectFile: true, syscall: 'write' });
341341
} else {
342342
this.fd = getValidatedFd(importFd(this, options));
343343
}

lib/internal/fs/utils.js

+41-6
Original file line numberDiff line numberDiff line change
@@ -706,13 +706,38 @@ const validateOffsetLengthWrite = hideStackFrames(
706706
},
707707
);
708708

709-
const validatePath = hideStackFrames((path, propName = 'path') => {
709+
/**
710+
* Validates the given path based on the expected type (file or directory).
711+
* @param {string} path - The path to validate.
712+
* @param {string} [propName='path'] - The name of the property being validated.
713+
* @param {object} options - Additional options for validation.
714+
* @param {boolean} [options.expectFile] - If true, expects the path to be a file.
715+
* @param {string} [options.syscall] - The name of the syscall.
716+
* @throws {TypeError} Throws if the path is not a string, Uint8Array, or URL without null bytes.
717+
* @throws {Error} Throws if the path does not meet the expected type (file).
718+
*/
719+
const validatePath = hideStackFrames((path, propName = 'path', options) => {
710720
if (typeof path !== 'string' && !isUint8Array(path)) {
711721
throw new ERR_INVALID_ARG_TYPE.HideStackFramesError(propName, ['string', 'Buffer', 'URL'], path);
712722
}
713723

714724
const pathIsString = typeof path === 'string';
715725
const pathIsUint8Array = isUint8Array(path);
726+
if (options && options.expectFile) {
727+
const lastCharacter = path[path.length - 1];
728+
if (
729+
lastCharacter === '/' || lastCharacter === 47 ||
730+
(isWindows && (lastCharacter === '\\' || lastCharacter === 92))
731+
) {
732+
throw new ERR_FS_EISDIR({
733+
code: 'ERR_FS_EISDIR',
734+
message: 'is a directory',
735+
path,
736+
syscall: options.syscall,
737+
errno: ERR_FS_EISDIR,
738+
});
739+
}
740+
}
716741

717742
// We can only perform meaningful checks on strings and Uint8Arrays.
718743
if ((!pathIsString && !pathIsUint8Array) ||
@@ -728,11 +753,21 @@ const validatePath = hideStackFrames((path, propName = 'path') => {
728753
);
729754
});
730755

731-
const getValidatedPath = hideStackFrames((fileURLOrPath, propName = 'path') => {
732-
const path = toPathIfFileURL(fileURLOrPath);
733-
validatePath(path, propName);
734-
return path;
735-
});
756+
/**
757+
* Validates and returns the given file URL or path based on the expected type (file or directory).
758+
* @param {string|URL} fileURLOrPath - The file URL or path to validate.
759+
* @param {string} [propName='path'] - The name of the property being validated.
760+
* @param {object} options - Additional options for validation.
761+
* @param {boolean} [options.expectFile] - If true, expects the path to be a file.
762+
* @param {string} [options.syscall] - The name of the syscall.
763+
* @returns {string} The validated path.
764+
*/
765+
const getValidatedPath =
766+
hideStackFrames((fileURLOrPath, propName = 'path', options) => {
767+
const path = toPathIfFileURL(fileURLOrPath);
768+
validatePath(path, propName, options);
769+
return path;
770+
});
736771

737772
const getValidatedFd = hideStackFrames((fd, propName = 'fd') => {
738773
if (ObjectIs(fd, -0)) {

test/sequential/test-fs-path-dir.js

+174
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,174 @@
1+
'use strict';
2+
3+
const common = require('../common');
4+
const tmpdir = require('../common/tmpdir');
5+
const assert = require('assert');
6+
const path = require('path');
7+
const fs = require('fs');
8+
const URL = require('url').URL;
9+
10+
tmpdir.refresh();
11+
12+
let fileCounter = 1;
13+
const nextFile = () => path.join(tmpdir.path, `file${fileCounter++}`);
14+
15+
const generateStringPath = (file, suffix = '') => file + suffix;
16+
17+
const generateURLPath = (file, suffix = '') =>
18+
new URL('file://' + path.resolve(file) + suffix);
19+
20+
const generateUint8ArrayPath = (file, suffix = '') =>
21+
new Uint8Array(Buffer.from(file + suffix));
22+
23+
const generatePaths = (file, suffix = '') => [
24+
generateStringPath(file, suffix),
25+
generateURLPath(file, suffix),
26+
generateUint8ArrayPath(file, suffix),
27+
];
28+
29+
const generateNewPaths = (suffix = '') => [
30+
generateStringPath(nextFile(), suffix),
31+
generateURLPath(nextFile(), suffix),
32+
generateUint8ArrayPath(nextFile(), suffix),
33+
];
34+
35+
function checkSyncFn(syncFn, p, args, fail) {
36+
const result = fail ? 'must fail' : 'must succeed';
37+
const failMsg = `failed testing sync ${p} ${syncFn.name} ${result}`;
38+
if (!fail) {
39+
try {
40+
syncFn(p, ...args);
41+
} catch (e) {
42+
console.log(failMsg, e);
43+
throw e;
44+
}
45+
} else {
46+
assert.throws(() => syncFn(p, ...args), {
47+
code: 'ERR_FS_EISDIR',
48+
}, failMsg);
49+
}
50+
}
51+
52+
function checkAsyncFn(asyncFn, p, args, fail) {
53+
const result = fail ? 'must fail' : 'must succeed';
54+
const failMsg = `failed testing async ${p} ${asyncFn.name} ${result}`;
55+
if (!fail) {
56+
try {
57+
asyncFn(p, ...args, common.mustCall());
58+
} catch (e) {
59+
console.log(failMsg, e);
60+
throw e;
61+
}
62+
} else {
63+
assert.throws(
64+
() => asyncFn(p, ...args, common.mustNotCall()), {
65+
code: 'ERR_FS_EISDIR',
66+
},
67+
failMsg
68+
);
69+
}
70+
}
71+
72+
function checkPromiseFn(promiseFn, p, args, fail) {
73+
const result = fail ? 'must fail' : 'must succeed';
74+
const failMsg = `failed testing promise ${p} ${promiseFn.name} ${result}`;
75+
if (!fail) {
76+
const r = promiseFn(p, ...args).catch((err) => {
77+
console.log(failMsg, err);
78+
throw err;
79+
});
80+
if (r && r.close) r.close();
81+
} else {
82+
assert.rejects(
83+
promiseFn(p, ...args), {
84+
code: 'ERR_FS_EISDIR',
85+
},
86+
failMsg
87+
).then(common.mustCall());
88+
}
89+
}
90+
91+
function check(asyncFn, syncFn, promiseFn,
92+
{ suffix, fail, args = [] }) {
93+
const file = nextFile();
94+
fs.writeFile(file, 'data', (e) => {
95+
assert.ifError(e);
96+
const paths = generatePaths(file, suffix);
97+
for (const p of paths) {
98+
if (asyncFn) checkAsyncFn(asyncFn, p, args, fail);
99+
if (promiseFn) checkPromiseFn(promiseFn, p, args, fail);
100+
if (syncFn) checkSyncFn(syncFn, p, args, fail);
101+
}
102+
});
103+
}
104+
105+
function checkManyToMany(asyncFn, syncFn, promiseFn,
106+
{ suffix, fail, args = [] }) {
107+
const source = nextFile();
108+
fs.writeFile(source, 'data', (err) => {
109+
assert.ifError(err);
110+
if (asyncFn) {
111+
generateNewPaths(suffix)
112+
.forEach((p) => checkAsyncFn(asyncFn, source, [p, ...args], fail));
113+
}
114+
if (promiseFn) {
115+
generateNewPaths(suffix)
116+
.forEach((p) => checkPromiseFn(promiseFn, source, [p, ...args], fail));
117+
}
118+
if (syncFn) {
119+
generateNewPaths(suffix)
120+
.forEach((p) => checkSyncFn(syncFn, source, [p, ...args], fail));
121+
}
122+
});
123+
}
124+
check(fs.readFile, fs.readFileSync, fs.promises.readFile,
125+
{ suffix: '', fail: false });
126+
check(fs.readFile, fs.readFileSync, fs.promises.readFile,
127+
{ suffix: '/', fail: true });
128+
check(fs.writeFile, fs.writeFileSync, fs.promises.writeFile,
129+
{ suffix: '', fail: false, args: ['data'] });
130+
check(fs.writeFile, fs.writeFileSync, fs.promises.writeFile,
131+
{ suffix: '/', fail: true, args: ['data'] });
132+
check(fs.appendFile, fs.appendFileSync, fs.promises.appendFile,
133+
{ suffix: '', fail: false, args: ['data'] });
134+
check(fs.appendFile, fs.appendFileSync, fs.promises.appendFile,
135+
{ suffix: '/', fail: true, args: ['data'] });
136+
check(undefined, fs.createReadStream, undefined,
137+
{ suffix: '', fail: false });
138+
check(undefined, fs.createReadStream, undefined,
139+
{ suffix: '/', fail: true });
140+
check(undefined, fs.createWriteStream, undefined,
141+
{ suffix: '', fail: false });
142+
check(undefined, fs.createWriteStream, undefined,
143+
{ suffix: '/', fail: true });
144+
check(fs.truncate, fs.truncateSync, fs.promises.truncate,
145+
{ suffix: '', fail: false, args: [42] });
146+
check(fs.truncate, fs.truncateSync, fs.promises.truncate,
147+
{ suffix: '/', fail: true, args: [42] });
148+
149+
check(fs.open, fs.openSync, fs.promises.open, { suffix: '', fail: false });
150+
check(fs.open, fs.openSync, fs.promises.open, { suffix: '/', fail: true });
151+
152+
checkManyToMany(fs.copyFile, fs.copyFileSync, fs.promises.copyFile,
153+
{ suffix: '', fail: false });
154+
155+
checkManyToMany(fs.copyFile, fs.copyFileSync, fs.promises.copyFile,
156+
{ suffix: '/', fail: true });
157+
158+
if (common.isWindows) {
159+
check(fs.readFile, fs.readFileSync, fs.promises.readFile,
160+
{ suffix: '\\', fail: true });
161+
check(fs.writeFile, fs.writeFileSync, fs.promises.writeFile,
162+
{ suffix: '\\', fail: true, args: ['data'] });
163+
check(fs.appendFile, fs.appendFileSync, fs.promises.appendFile,
164+
{ suffix: '\\', fail: true, args: ['data'] });
165+
check(undefined, fs.createReadStream, undefined,
166+
{ suffix: '\\', fail: true });
167+
check(undefined, fs.createWriteStream, undefined,
168+
{ suffix: '\\', fail: true });
169+
check(fs.truncate, fs.truncateSync, fs.promises.truncate,
170+
{ suffix: '\\', fail: true, args: [42] });
171+
check(fs.open, fs.openSync, fs.promises.open, { suffix: '\\', fail: true });
172+
checkManyToMany(fs.copyFile, fs.copyFileSync, fs.promises.copyFile,
173+
{ suffix: '\\', fail: true });
174+
}

0 commit comments

Comments
 (0)