Skip to content

Commit ae40cb1

Browse files
LiviaMedeirosUlisesGascon
authored andcommitted
url: validate pathToFileURL(path) argument as string
PR-URL: #49161 Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com> Reviewed-By: Antoine du Hamel <duhamelantoine1995@gmail.com>
1 parent 8d0913b commit ae40cb1

File tree

3 files changed

+47
-5
lines changed

3 files changed

+47
-5
lines changed

lib/internal/url.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -1441,14 +1441,14 @@ function pathToFileURL(filepath) {
14411441
const hostnameEndIndex = StringPrototypeIndexOf(filepath, '\\', 2);
14421442
if (hostnameEndIndex === -1) {
14431443
throw new ERR_INVALID_ARG_VALUE(
1444-
'filepath',
1444+
'path',
14451445
filepath,
14461446
'Missing UNC resource path',
14471447
);
14481448
}
14491449
if (hostnameEndIndex === 2) {
14501450
throw new ERR_INVALID_ARG_VALUE(
1451-
'filepath',
1451+
'path',
14521452
filepath,
14531453
'Empty UNC servername',
14541454
);

lib/url.js

+10-1
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ const {
5353
domainToASCII,
5454
domainToUnicode,
5555
fileURLToPath,
56-
pathToFileURL,
56+
pathToFileURL: _pathToFileURL,
5757
urlToHttpOptions,
5858
unsafeProtocol,
5959
hostlessProtocol,
@@ -1017,6 +1017,15 @@ Url.prototype.parseHost = function parseHost() {
10171017
if (host) this.hostname = host;
10181018
};
10191019

1020+
// When used internally, we are not obligated to associate TypeError with
1021+
// this function, so non-strings can be rejected by underlying implementation.
1022+
// Public API has to validate input and throw appropriate error.
1023+
function pathToFileURL(path) {
1024+
validateString(path, 'path');
1025+
1026+
return _pathToFileURL(path);
1027+
}
1028+
10201029
module.exports = {
10211030
// Original API
10221031
Url,

test/parallel/test-url-pathtofileurl.js

+35-2
Original file line numberDiff line numberDiff line change
@@ -29,13 +29,30 @@ const url = require('url');
2929

3030
// Missing server:
3131
assert.throws(() => url.pathToFileURL('\\\\\\no-server'), {
32-
code: 'ERR_INVALID_ARG_VALUE'
32+
code: 'ERR_INVALID_ARG_VALUE',
3333
});
3434

3535
// Missing share or resource:
3636
assert.throws(() => url.pathToFileURL('\\\\host'), {
37-
code: 'ERR_INVALID_ARG_VALUE'
37+
code: 'ERR_INVALID_ARG_VALUE',
3838
});
39+
40+
// Regression test for direct String.prototype.startsWith call
41+
assert.throws(() => url.pathToFileURL([
42+
'\\\\',
43+
{ [Symbol.toPrimitive]: () => 'blep\\blop' },
44+
]), {
45+
code: 'ERR_INVALID_ARG_TYPE',
46+
});
47+
assert.throws(() => url.pathToFileURL(['\\\\', 'blep\\blop']), {
48+
code: 'ERR_INVALID_ARG_TYPE',
49+
});
50+
assert.throws(() => url.pathToFileURL({
51+
[Symbol.toPrimitive]: () => '\\\\blep\\blop',
52+
}), {
53+
code: 'ERR_INVALID_ARG_TYPE',
54+
});
55+
3956
} else {
4057
// UNC paths on posix are considered a single path that has backslashes:
4158
const fileURL = url.pathToFileURL('\\\\nas\\share\\path.txt').href;
@@ -144,3 +161,19 @@ const url = require('url');
144161
assert.strictEqual(actual, expected);
145162
}
146163
}
164+
165+
// Test for non-string parameter
166+
{
167+
for (const badPath of [
168+
undefined, null, true, 42, 42n, Symbol('42'), NaN, {}, [], () => {},
169+
Promise.resolve('foo'),
170+
new Date(),
171+
new String('notPrimitive'),
172+
{ toString() { return 'amObject'; } },
173+
{ [Symbol.toPrimitive]: (hint) => 'amObject' },
174+
]) {
175+
assert.throws(() => url.pathToFileURL(badPath), {
176+
code: 'ERR_INVALID_ARG_TYPE',
177+
});
178+
}
179+
}

0 commit comments

Comments
 (0)