Skip to content

Commit 3bed5f1

Browse files
committed
url: runtime-deprecate url.parse() with invalid ports
These URLs throw with WHATWG URL. They are permitted with url.parse() but that allows potential host spoofing by sending a domain name as the port. PR-URL: #45526 Reviewed-By: Yagiz Nizipli <[email protected]> Reviewed-By: Matteo Collina <[email protected]> Reviewed-By: Robert Nagy <[email protected]> Reviewed-By: Antoine du Hamel <[email protected]> Reviewed-By: James M Snell <[email protected]>
1 parent 2589fb1 commit 3bed5f1

File tree

3 files changed

+45
-3
lines changed

3 files changed

+45
-3
lines changed

doc/api/deprecations.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -3302,13 +3302,17 @@ issued for `url.parse()` vulnerabilities.
33023302

33033303
<!-- YAML
33043304
changes:
3305+
- version:
3306+
- REPLACEME
3307+
pr-url: https://github.com/nodejs/node/pull/45526
3308+
description: Runtime deprecation.
33053309
- version:
33063310
- v19.2.0
33073311
pr-url: https://github.com/nodejs/node/pull/45576
33083312
description: Documentation-only deprecation.
33093313
-->
33103314

3311-
Type: Documentation-only
3315+
Type: Runtime
33123316

33133317
[`url.parse()`][] accepts URLs with ports that are not numbers. This behavior
33143318
might result in host name spoofing with unexpected input. These URLs will throw

lib/url.js

+11-2
Original file line numberDiff line numberDiff line change
@@ -387,7 +387,7 @@ Url.prototype.parse = function parse(url, parseQueryString, slashesDenoteHost) {
387387

388388
// validate a little.
389389
if (!ipv6Hostname) {
390-
rest = getHostname(this, rest, hostname);
390+
rest = getHostname(this, rest, hostname, url);
391391
}
392392

393393
if (this.hostname.length > hostnameMaxLen) {
@@ -506,7 +506,8 @@ Url.prototype.parse = function parse(url, parseQueryString, slashesDenoteHost) {
506506
return this;
507507
};
508508

509-
function getHostname(self, rest, hostname) {
509+
let warnInvalidPort = true;
510+
function getHostname(self, rest, hostname, url) {
510511
for (let i = 0; i < hostname.length; ++i) {
511512
const code = hostname.charCodeAt(i);
512513
const isValid = (code !== CHAR_FORWARD_SLASH &&
@@ -516,6 +517,14 @@ function getHostname(self, rest, hostname) {
516517
code !== CHAR_COLON);
517518

518519
if (!isValid) {
520+
// If leftover starts with :, then it represents an invalid port.
521+
// But url.parse() is lenient about it for now.
522+
// Issue a warning and continue.
523+
if (warnInvalidPort && code === CHAR_COLON) {
524+
const detail = `The URL ${url} is invalid. Future versions of Node.js will throw an error.`;
525+
process.emitWarning(detail, 'DeprecationWarning', 'DEP0170');
526+
warnInvalidPort = false;
527+
}
519528
self.hostname = hostname.slice(0, i);
520529
return `/${hostname.slice(i)}${rest}`;
521530
}

test/parallel/test-url-parse-invalid-input.js

+29
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
'use strict';
22
const common = require('../common');
33
const assert = require('assert');
4+
const childProcess = require('child_process');
45
const url = require('url');
56

67
// https://github.com/joyent/node/issues/568
@@ -74,3 +75,31 @@ if (common.hasIntl) {
7475
(e) => e.code === 'ERR_INVALID_URL',
7576
'parsing http://\u00AD/bad.com/');
7677
}
78+
79+
{
80+
const badURLs = [
81+
'https://evil.com:.example.com',
82+
'git+ssh://[email protected]:npm/npm',
83+
];
84+
badURLs.forEach((badURL) => {
85+
childProcess.exec(`${process.execPath} -e "url.parse('${badURL}')"`,
86+
common.mustCall((err, stdout, stderr) => {
87+
assert.strictEqual(err, null);
88+
assert.strictEqual(stdout, '');
89+
assert.match(stderr, /\[DEP0170\] DeprecationWarning:/);
90+
})
91+
);
92+
});
93+
94+
// Warning should only happen once per process.
95+
const expectedWarning = [
96+
`The URL ${badURLs[0]} is invalid. Future versions of Node.js will throw an error.`,
97+
'DEP0170',
98+
];
99+
common.expectWarning({
100+
DeprecationWarning: expectedWarning,
101+
});
102+
badURLs.forEach((badURL) => {
103+
url.parse(badURL);
104+
});
105+
}

0 commit comments

Comments
 (0)