Skip to content

Commit 057bd44

Browse files
lirantalRafaelGSS
authored andcommitted
child_process: fix incomplete prototype pollution hardening
Prior pull request (#48726) hardened against prototype pollution vulnerabilities but effectively missed some use-cases which opened a window for prototype pollution for some child_process functions such as spawn(), spawnSync(), and execFileSync(). PR-URL: #53781 Reviewed-By: Vinícius Lourenço Claro Cardoso <contact@viniciusl.com.br> Reviewed-By: Matteo Collina <matteo.collina@gmail.com> Reviewed-By: Rafael Gonzaga <rafael.nunu@hotmail.com>
1 parent 04bb677 commit 057bd44

File tree

2 files changed

+34
-1
lines changed

2 files changed

+34
-1
lines changed

lib/child_process.js

+1
Original file line numberDiff line numberDiff line change
@@ -568,6 +568,7 @@ function normalizeSpawnArguments(file, args, options) {
568568
else
569569
validateObject(options, 'options');
570570

571+
options = { __proto__: null, ...options };
571572
let cwd = options.cwd;
572573

573574
// Validate the cwd, if present.

test/parallel/test-child-process-prototype-tampering.mjs

+33-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
import * as common from '../common/index.mjs';
22
import * as fixtures from '../common/fixtures.mjs';
33
import { EOL } from 'node:os';
4-
import { strictEqual } from 'node:assert';
4+
import { strictEqual, notStrictEqual, throws } from 'node:assert';
55
import cp from 'node:child_process';
66

77
// TODO(LiviaMedeiros): test on different platforms
@@ -57,3 +57,35 @@ for (const tamperedUID of [0, 1, 999, 1000, 0n, 'gwak']) {
5757

5858
delete Object.prototype.execPath;
5959
}
60+
61+
for (const shellCommandArgument of ['-L && echo "tampered"']) {
62+
Object.prototype.shell = true;
63+
const cmd = 'pwd';
64+
let cmdExitCode = '';
65+
66+
const program = cp.spawn(cmd, [shellCommandArgument], { cwd: expectedCWD });
67+
program.stderr.on('data', common.mustCall());
68+
program.stdout.on('data', common.mustNotCall());
69+
70+
program.on('exit', common.mustCall((code) => {
71+
notStrictEqual(code, 0);
72+
}));
73+
74+
cp.execFile(cmd, [shellCommandArgument], { cwd: expectedCWD },
75+
common.mustCall((err) => {
76+
notStrictEqual(err.code, 0);
77+
})
78+
);
79+
80+
throws(() => {
81+
cp.execFileSync(cmd, [shellCommandArgument], { cwd: expectedCWD });
82+
}, (e) => {
83+
notStrictEqual(e.status, 0);
84+
return true;
85+
});
86+
87+
cmdExitCode = cp.spawnSync(cmd, [shellCommandArgument], { cwd: expectedCWD }).status;
88+
notStrictEqual(cmdExitCode, 0);
89+
90+
delete Object.prototype.shell;
91+
}

0 commit comments

Comments
 (0)