-
Notifications
You must be signed in to change notification settings - Fork 30.3k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
url: extend url.format to support WHATWG URL #10857
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,13 +1,5 @@ | ||
'use strict'; | ||
|
||
function getPunycode() { | ||
try { | ||
return process.binding('icu'); | ||
} catch (err) { | ||
return require('punycode'); | ||
} | ||
} | ||
const punycode = getPunycode(); | ||
const util = require('util'); | ||
const binding = process.binding('url'); | ||
const context = Symbol('context'); | ||
|
@@ -20,6 +12,7 @@ const kScheme = Symbol('scheme'); | |
const kHost = Symbol('host'); | ||
const kPort = Symbol('port'); | ||
const kDomain = Symbol('domain'); | ||
const kFormat = Symbol('format'); | ||
|
||
// https://tc39.github.io/ecma262/#sec-%iteratorprototype%-object | ||
const IteratorPrototype = Object.getPrototypeOf( | ||
|
@@ -263,18 +256,19 @@ class URL { | |
} | ||
|
||
Object.defineProperties(URL.prototype, { | ||
toString: { | ||
// https://heycam.github.io/webidl/#es-stringifier | ||
writable: true, | ||
enumerable: true, | ||
configurable: true, | ||
[kFormat]: { | ||
enumerable: false, | ||
configurable: false, | ||
// eslint-disable-next-line func-name-matching | ||
value: function toString(options) { | ||
options = options || {}; | ||
const fragment = | ||
options.fragment !== undefined ? | ||
!!options.fragment : true; | ||
const unicode = !!options.unicode; | ||
value: function format(options) { | ||
if (options && typeof options !== 'object') | ||
throw new TypeError('options must be an object'); | ||
options = Object.assign({ | ||
fragment: true, | ||
unicode: false, | ||
search: true, | ||
auth: true | ||
}, options); | ||
const ctx = this[context]; | ||
var ret; | ||
if (this.protocol) | ||
|
@@ -284,28 +278,23 @@ Object.defineProperties(URL.prototype, { | |
const has_username = typeof ctx.username === 'string'; | ||
const has_password = typeof ctx.password === 'string' && | ||
ctx.password !== ''; | ||
if (has_username || has_password) { | ||
if (options.auth && (has_username || has_password)) { | ||
if (has_username) | ||
ret += ctx.username; | ||
if (has_password) | ||
ret += `:${ctx.password}`; | ||
ret += '@'; | ||
} | ||
if (unicode) { | ||
ret += punycode.toUnicode(this.hostname); | ||
if (this.port !== undefined) | ||
ret += `:${this.port}`; | ||
} else { | ||
ret += this.host; | ||
} | ||
ret += options.unicode ? | ||
domainToUnicode(this.host) : this.host; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Is the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. (Strange, my response ended up both in my own review and here, but when I deleted that one in my review this one disappeared too).. I think There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Eh, if it’s in the standard, even just on the parse side, it’s probably best to leave it like that. Thanks for pointing it out.
Well, if it’s working and people know what the option describes, that’s probably just fine then. (At least in my head, There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I will make sure that the relevant documentation appropriately covers the option and makes its function clear |
||
} else if (ctx.scheme === 'file:') { | ||
ret += '//'; | ||
} | ||
if (this.pathname) | ||
ret += this.pathname; | ||
if (typeof ctx.query === 'string') | ||
if (options.search && typeof ctx.query === 'string') | ||
ret += `?${ctx.query}`; | ||
if (fragment & typeof ctx.fragment === 'string') | ||
if (options.fragment && typeof ctx.fragment === 'string') | ||
ret += `#${ctx.fragment}`; | ||
return ret; | ||
} | ||
|
@@ -314,11 +303,21 @@ Object.defineProperties(URL.prototype, { | |
configurable: true, | ||
value: 'URL' | ||
}, | ||
toString: { | ||
// https://heycam.github.io/webidl/#es-stringifier | ||
writable: true, | ||
enumerable: true, | ||
configurable: true, | ||
// eslint-disable-next-line func-name-matching | ||
value: function toString() { | ||
return this[kFormat]({}); | ||
} | ||
}, | ||
href: { | ||
enumerable: true, | ||
configurable: true, | ||
get() { | ||
return this.toString(); | ||
return this[kFormat]({}); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think keep it as There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The result is the same. Using |
||
}, | ||
set(input) { | ||
parse(this, input); | ||
|
@@ -1120,3 +1119,4 @@ exports.domainToASCII = domainToASCII; | |
exports.domainToUnicode = domainToUnicode; | ||
exports.encodeAuth = encodeAuth; | ||
exports.urlToOptions = urlToOptions; | ||
exports.formatSymbol = kFormat; |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,102 @@ | ||
'use strict'; | ||
|
||
require('../common'); | ||
const assert = require('assert'); | ||
const url = require('url'); | ||
const URL = url.URL; | ||
|
||
const myURL = new URL('http://xn--lck1c3crb1723bpq4a.com/a?a=b#c'); | ||
|
||
assert.strictEqual( | ||
url.format(myURL), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
const errreg = /^TypeError: options must be an object$/; | ||
assert.throws(() => url.format(myURL, true), errreg); | ||
assert.throws(() => url.format(myURL, 1), errreg); | ||
assert.throws(() => url.format(myURL, 'test'), errreg); | ||
assert.throws(() => url.format(myURL, Infinity), errreg); | ||
|
||
// Any falsy value other than undefined will be treated as false. | ||
// Any truthy value will be treated as true. | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {fragment: false}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {fragment: ''}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {fragment: 0}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {fragment: 1}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {fragment: {}}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {search: false}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {search: ''}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {search: 0}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {search: 1}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {search: {}}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {unicode: true}), | ||
'http://理容ナカムラ.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {unicode: 1}), | ||
'http://理容ナカムラ.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {unicode: {}}), | ||
'http://理容ナカムラ.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {unicode: false}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); | ||
|
||
assert.strictEqual( | ||
url.format(myURL, {unicode: 0}), | ||
'http://xn--lck1c3crb1723bpq4a.com/a?a=b#c' | ||
); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
These two lines are not needed as they are the default. However, the trend in Web APIs seems to be make symbol properties configurable and writable, so I'd recommend sticking to that
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, I know they are the default. I prefer to be explicit. Also, because this is an internal only method, I prefer to keep these values