Skip to content

Commit b9289a6

Browse files
committed
lib,permission: support fs.lstat
PR-URL: nodejs-private/node-private#486 Fixes: https://hackerone.com/bugs?subject=nodejs&report_id=2145862 Reviewed-By: Yagiz Nizipli <[email protected]> Reviewed-By: Moshe Atlow <[email protected]> CVE-ID: CVE-2024-22018
1 parent 39f2070 commit b9289a6

File tree

3 files changed

+37
-6
lines changed

3 files changed

+37
-6
lines changed

lib/fs.js

+8
Original file line numberDiff line numberDiff line change
@@ -1550,6 +1550,10 @@ function lstat(path, options = { bigint: false }, callback) {
15501550
}
15511551
callback = makeStatsCallback(callback);
15521552
path = getValidatedPath(path);
1553+
if (permission.isEnabled() && !permission.has('fs.read', path)) {
1554+
callback(new ERR_ACCESS_DENIED('Access to this API has been restricted', 'FileSystemRead', path));
1555+
return;
1556+
}
15531557

15541558
const req = new FSReqCallback(options.bigint);
15551559
req.oncomplete = callback;
@@ -1622,6 +1626,10 @@ function fstatSync(fd, options = { bigint: false }) {
16221626
* @returns {Stats | undefined}
16231627
*/
16241628
function lstatSync(path, options = { bigint: false, throwIfNoEntry: true }) {
1629+
path = getValidatedPath(path);
1630+
if (permission.isEnabled() && !permission.has('fs.read', path)) {
1631+
throw new ERR_ACCESS_DENIED('Access to this API has been restricted', 'FileSystemRead', path);
1632+
}
16251633
const stats = binding.lstat(
16261634
getValidatedPath(path),
16271635
options.bigint,

lib/internal/errors.js

+5-1
Original file line numberDiff line numberDiff line change
@@ -1112,7 +1112,11 @@ module.exports = {
11121112
// Note: Node.js specific errors must begin with the prefix ERR_
11131113

11141114
E('ERR_ACCESS_DENIED',
1115-
'Access to this API has been restricted. Permission: %s',
1115+
function(msg, permission = '', resource = '') {
1116+
this.permission = permission;
1117+
this.resource = resource;
1118+
return msg;
1119+
},
11161120
Error);
11171121
E('ERR_AMBIGUOUS_ARGUMENT', 'The "%s" argument is ambiguous. %s', TypeError);
11181122
E('ERR_ARG_NOT_ITERABLE', '%s must be iterable', TypeError);

test/fixtures/permission/fs-read.js

+24-5
Original file line numberDiff line numberDiff line change
@@ -161,23 +161,23 @@ const regularFile = __filename;
161161
}, common.expectsError({
162162
code: 'ERR_ACCESS_DENIED',
163163
permission: 'FileSystemRead',
164-
// cpSync calls statSync before reading blockedFile
165-
resource: path.toNamespacedPath(blockedFolder),
164+
// cpSync calls lstatSync before reading blockedFile
165+
resource: blockedFile,
166166
}));
167167
assert.throws(() => {
168168
fs.cpSync(blockedFileURL, path.join(blockedFolder, 'any-other-file'));
169169
}, common.expectsError({
170170
code: 'ERR_ACCESS_DENIED',
171171
permission: 'FileSystemRead',
172-
// cpSync calls statSync before reading blockedFile
173-
resource: path.toNamespacedPath(blockedFolder),
172+
// cpSync calls lstatSync before reading blockedFile
173+
resource: blockedFile,
174174
}));
175175
assert.throws(() => {
176176
fs.cpSync(blockedFile, path.join(__dirname, 'any-other-file'));
177177
}, common.expectsError({
178178
code: 'ERR_ACCESS_DENIED',
179179
permission: 'FileSystemRead',
180-
resource: path.toNamespacedPath(__dirname),
180+
resource: blockedFile,
181181
}));
182182
}
183183

@@ -396,4 +396,23 @@ const regularFile = __filename;
396396
permission: 'FileSystemRead',
397397
resource: blockedFolder,
398398
}));
399+
}
400+
401+
// fs.lstat
402+
{
403+
assert.throws(() => {
404+
fs.lstatSync(blockedFile);
405+
}, common.expectsError({
406+
code: 'ERR_ACCESS_DENIED',
407+
}));
408+
assert.throws(() => {
409+
fs.lstatSync(path.join(blockedFolder, 'anyfile'));
410+
}, common.expectsError({
411+
code: 'ERR_ACCESS_DENIED',
412+
}));
413+
414+
// doesNotThrow
415+
fs.lstat(regularFile, (err) => {
416+
assert.ifError(err);
417+
});
399418
}

0 commit comments

Comments
 (0)