mirror of https://github.com/nodejs/node.git
url: improving URLSearchParams
- add some benchmarks for URLSearchParams - change URLSearchParams backing store to an array - add custom inspection for URLSearchParams and its iterators PR-URL: https://github.com/nodejs/node/pull/10399 Reviewed-By: James M Snell <jasnell@gmail.com>
This commit is contained in:
parent
44c6b6b22a
commit
98bb65f641
|
@ -0,0 +1,60 @@
|
||||||
|
'use strict';
|
||||||
|
const common = require('../common.js');
|
||||||
|
const assert = require('assert');
|
||||||
|
const URLSearchParams = new (require('url').URL)('a:').searchParams.constructor;
|
||||||
|
|
||||||
|
const bench = common.createBenchmark(main, {
|
||||||
|
method: ['forEach', 'iterator'],
|
||||||
|
n: [1e6]
|
||||||
|
});
|
||||||
|
|
||||||
|
const str = 'one=single&two=first&three=first&two=2nd&three=2nd&three=3rd';
|
||||||
|
|
||||||
|
function forEach(n) {
|
||||||
|
const params = new URLSearchParams(str);
|
||||||
|
const noDead = [];
|
||||||
|
const cb = (val, key) => {
|
||||||
|
noDead[0] = key;
|
||||||
|
noDead[1] = val;
|
||||||
|
};
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
params.forEach(cb);
|
||||||
|
bench.end(n);
|
||||||
|
|
||||||
|
assert.strictEqual(noDead[0], 'three');
|
||||||
|
assert.strictEqual(noDead[1], '3rd');
|
||||||
|
}
|
||||||
|
|
||||||
|
function iterator(n) {
|
||||||
|
const params = new URLSearchParams(str);
|
||||||
|
const noDead = [];
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
for (var pair of params) {
|
||||||
|
noDead[0] = pair[0];
|
||||||
|
noDead[1] = pair[1];
|
||||||
|
}
|
||||||
|
bench.end(n);
|
||||||
|
|
||||||
|
assert.strictEqual(noDead[0], 'three');
|
||||||
|
assert.strictEqual(noDead[1], '3rd');
|
||||||
|
}
|
||||||
|
|
||||||
|
function main(conf) {
|
||||||
|
const method = conf.method;
|
||||||
|
const n = conf.n | 0;
|
||||||
|
|
||||||
|
switch (method) {
|
||||||
|
case 'forEach':
|
||||||
|
forEach(n);
|
||||||
|
break;
|
||||||
|
case 'iterator':
|
||||||
|
iterator(n);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown method');
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,31 @@
|
||||||
|
'use strict';
|
||||||
|
const common = require('../common.js');
|
||||||
|
const URLSearchParams = new (require('url').URL)('a:').searchParams.constructor;
|
||||||
|
|
||||||
|
const inputs = {
|
||||||
|
noencode: 'foo=bar&baz=quux&xyzzy=thud',
|
||||||
|
// multicharsep: 'foo=bar&&&&&&&&&&baz=quux&&&&&&&&&&xyzzy=thud',
|
||||||
|
multicharsep: '&&&&&&&&&&&&&&&&&&&&&&&&&&&&',
|
||||||
|
encodemany: '%66%6F%6F=bar&%62%61%7A=quux&xyzzy=%74h%75d',
|
||||||
|
encodelast: 'foo=bar&baz=quux&xyzzy=thu%64',
|
||||||
|
multivalue: 'foo=bar&foo=baz&foo=quux&quuy=quuz',
|
||||||
|
multivaluemany: 'foo=bar&foo=baz&foo=quux&quuy=quuz&foo=abc&foo=def&' +
|
||||||
|
'foo=ghi&foo=jkl&foo=mno&foo=pqr&foo=stu&foo=vwxyz',
|
||||||
|
manypairs: 'a&b&c&d&e&f&g&h&i&j&k&l&m&n&o&p&q&r&s&t&u&v&w&x&y&z'
|
||||||
|
};
|
||||||
|
|
||||||
|
const bench = common.createBenchmark(main, {
|
||||||
|
type: Object.keys(inputs),
|
||||||
|
n: [1e5]
|
||||||
|
});
|
||||||
|
|
||||||
|
function main(conf) {
|
||||||
|
const input = inputs[conf.type];
|
||||||
|
const n = conf.n | 0;
|
||||||
|
|
||||||
|
var i;
|
||||||
|
bench.start();
|
||||||
|
for (i = 0; i < n; i++)
|
||||||
|
new URLSearchParams(input);
|
||||||
|
bench.end(n);
|
||||||
|
}
|
|
@ -0,0 +1,58 @@
|
||||||
|
'use strict';
|
||||||
|
const common = require('../common.js');
|
||||||
|
const URLSearchParams = new (require('url').URL)('a:').searchParams.constructor;
|
||||||
|
|
||||||
|
const bench = common.createBenchmark(main, {
|
||||||
|
method: ['get', 'getAll', 'has'],
|
||||||
|
param: ['one', 'two', 'three', 'nonexistent'],
|
||||||
|
n: [1e6]
|
||||||
|
});
|
||||||
|
|
||||||
|
const str = 'one=single&two=first&three=first&two=2nd&three=2nd&three=3rd';
|
||||||
|
|
||||||
|
function get(n, param) {
|
||||||
|
const params = new URLSearchParams(str);
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
params.get(param);
|
||||||
|
bench.end(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getAll(n, param) {
|
||||||
|
const params = new URLSearchParams(str);
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
params.getAll(param);
|
||||||
|
bench.end(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
function has(n, param) {
|
||||||
|
const params = new URLSearchParams(str);
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
params.has(param);
|
||||||
|
bench.end(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
function main(conf) {
|
||||||
|
const method = conf.method;
|
||||||
|
const param = conf.param;
|
||||||
|
const n = conf.n | 0;
|
||||||
|
|
||||||
|
switch (method) {
|
||||||
|
case 'get':
|
||||||
|
get(n, param);
|
||||||
|
break;
|
||||||
|
case 'getAll':
|
||||||
|
getAll(n, param);
|
||||||
|
break;
|
||||||
|
case 'has':
|
||||||
|
has(n, param);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
throw new Error('Unknown method');
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
'use strict';
|
||||||
|
const common = require('../common.js');
|
||||||
|
const Buffer = require('buffer').Buffer;
|
||||||
|
const URLSearchParams = new (require('url').URL)('a:').searchParams.constructor;
|
||||||
|
|
||||||
|
const inputs = {
|
||||||
|
noencode: 'foo=bar&baz=quux&xyzzy=thud',
|
||||||
|
// multicharsep: 'foo=bar&&&&&&&&&&baz=quux&&&&&&&&&&xyzzy=thud',
|
||||||
|
multicharsep: '&&&&&&&&&&&&&&&&&&&&&&&&&&&&',
|
||||||
|
encodemany: '%66%6F%6F=bar&%62%61%7A=quux&xyzzy=%74h%75d',
|
||||||
|
encodelast: 'foo=bar&baz=quux&xyzzy=thu%64',
|
||||||
|
multivalue: 'foo=bar&foo=baz&foo=quux&quuy=quuz',
|
||||||
|
multivaluemany: 'foo=bar&foo=baz&foo=quux&quuy=quuz&foo=abc&foo=def&' +
|
||||||
|
'foo=ghi&foo=jkl&foo=mno&foo=pqr&foo=stu&foo=vwxyz',
|
||||||
|
manypairs: 'a&b&c&d&e&f&g&h&i&j&k&l&m&n&o&p&q&r&s&t&u&v&w&x&y&z'
|
||||||
|
};
|
||||||
|
|
||||||
|
const bench = common.createBenchmark(main, {
|
||||||
|
type: Object.keys(inputs),
|
||||||
|
n: [1e5]
|
||||||
|
});
|
||||||
|
|
||||||
|
function main(conf) {
|
||||||
|
const input = inputs[conf.type];
|
||||||
|
const n = conf.n | 0;
|
||||||
|
|
||||||
|
const params = new URLSearchParams(input);
|
||||||
|
|
||||||
|
bench.start();
|
||||||
|
// Using Buffer.from to prevent JS version from cheating with ropes instead
|
||||||
|
// of strings
|
||||||
|
for (var i = 0; i < n; i += 1)
|
||||||
|
Buffer.from(params.toString());
|
||||||
|
bench.end(n);
|
||||||
|
}
|
|
@ -8,6 +8,7 @@ function getPunycode() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const punycode = getPunycode();
|
const punycode = getPunycode();
|
||||||
|
const util = require('util');
|
||||||
const binding = process.binding('url');
|
const binding = process.binding('url');
|
||||||
const context = Symbol('context');
|
const context = Symbol('context');
|
||||||
const cannotBeBase = Symbol('cannot-be-base');
|
const cannotBeBase = Symbol('cannot-be-base');
|
||||||
|
@ -493,7 +494,7 @@ Object.defineProperties(URL.prototype, {
|
||||||
ctx.query = '';
|
ctx.query = '';
|
||||||
binding.parse(search, binding.kQuery, null, ctx,
|
binding.parse(search, binding.kQuery, null, ctx,
|
||||||
onParseSearchComplete.bind(this));
|
onParseSearchComplete.bind(this));
|
||||||
this[searchParams][searchParams] = querystring.parse(search);
|
initSearchParams(this[searchParams], search);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
searchParams: { // readonly
|
searchParams: { // readonly
|
||||||
|
@ -610,8 +611,13 @@ function update(url, params) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function getSearchParamPairs(target) {
|
// Reused by the URL parse function invoked by
|
||||||
const obj = target[searchParams];
|
// the href setter, and the URLSearchParams constructor
|
||||||
|
function initSearchParams(url, init) {
|
||||||
|
url[searchParams] = getParamsFromObject(querystring.parse(init));
|
||||||
|
}
|
||||||
|
|
||||||
|
function getParamsFromObject(obj) {
|
||||||
const keys = Object.keys(obj);
|
const keys = Object.keys(obj);
|
||||||
const values = [];
|
const values = [];
|
||||||
for (var i = 0; i < keys.length; i++) {
|
for (var i = 0; i < keys.length; i++) {
|
||||||
|
@ -619,25 +625,33 @@ function getSearchParamPairs(target) {
|
||||||
const value = obj[name];
|
const value = obj[name];
|
||||||
if (Array.isArray(value)) {
|
if (Array.isArray(value)) {
|
||||||
for (const item of value)
|
for (const item of value)
|
||||||
values.push([name, item]);
|
values.push(name, item);
|
||||||
} else {
|
} else {
|
||||||
values.push([name, value]);
|
values.push(name, value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return values;
|
return values;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reused by the URL parse function invoked by
|
function getObjectFromParams(array) {
|
||||||
// the href setter, and the URLSearchParams constructor
|
const obj = new StorageObject();
|
||||||
function initSearchParams(url, init) {
|
for (var i = 0; i < array.length; i += 2) {
|
||||||
url[searchParams] = querystring.parse(init);
|
const name = array[i];
|
||||||
|
const value = array[i + 1];
|
||||||
|
if (obj[name]) {
|
||||||
|
obj[name].push(value);
|
||||||
|
} else {
|
||||||
|
obj[name] = [value];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return obj;
|
||||||
}
|
}
|
||||||
|
|
||||||
class URLSearchParams {
|
class URLSearchParams {
|
||||||
constructor(init = '') {
|
constructor(init = '') {
|
||||||
if (init instanceof URLSearchParams) {
|
if (init instanceof URLSearchParams) {
|
||||||
const childParams = init[searchParams];
|
const childParams = init[searchParams];
|
||||||
this[searchParams] = Object.assign(Object.create(null), childParams);
|
this[searchParams] = childParams.slice();
|
||||||
} else {
|
} else {
|
||||||
init = String(init);
|
init = String(init);
|
||||||
if (init[0] === '?') init = init.slice(1);
|
if (init[0] === '?') init = init.slice(1);
|
||||||
|
@ -662,17 +676,9 @@ class URLSearchParams {
|
||||||
'Both `name` and `value` arguments need to be specified');
|
'Both `name` and `value` arguments need to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
|
||||||
name = String(name);
|
name = String(name);
|
||||||
value = String(value);
|
value = String(value);
|
||||||
var existing = obj[name];
|
this[searchParams].push(name, value);
|
||||||
if (existing === undefined) {
|
|
||||||
obj[name] = value;
|
|
||||||
} else if (Array.isArray(existing)) {
|
|
||||||
existing.push(value);
|
|
||||||
} else {
|
|
||||||
obj[name] = [existing, value];
|
|
||||||
}
|
|
||||||
update(this[context], this);
|
update(this[context], this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -684,9 +690,16 @@ class URLSearchParams {
|
||||||
throw new TypeError('The `name` argument needs to be specified');
|
throw new TypeError('The `name` argument needs to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
const list = this[searchParams];
|
||||||
name = String(name);
|
name = String(name);
|
||||||
delete obj[name];
|
for (var i = 0; i < list.length;) {
|
||||||
|
const cur = list[i];
|
||||||
|
if (cur === name) {
|
||||||
|
list.splice(i, 2);
|
||||||
|
} else {
|
||||||
|
i += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
update(this[context], this);
|
update(this[context], this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -699,10 +712,35 @@ class URLSearchParams {
|
||||||
'Both `name` and `value` arguments need to be specified');
|
'Both `name` and `value` arguments need to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
const list = this[searchParams];
|
||||||
name = String(name);
|
name = String(name);
|
||||||
value = String(value);
|
value = String(value);
|
||||||
obj[name] = value;
|
|
||||||
|
// If there are any name-value pairs whose name is `name`, in `list`, set
|
||||||
|
// the value of the first such name-value pair to `value` and remove the
|
||||||
|
// others.
|
||||||
|
var found = false;
|
||||||
|
for (var i = 0; i < list.length;) {
|
||||||
|
const cur = list[i];
|
||||||
|
if (cur === name) {
|
||||||
|
if (!found) {
|
||||||
|
list[i + 1] = value;
|
||||||
|
found = true;
|
||||||
|
i += 2;
|
||||||
|
} else {
|
||||||
|
list.splice(i, 2);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
i += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Otherwise, append a new name-value pair whose name is `name` and value
|
||||||
|
// is `value`, to `list`.
|
||||||
|
if (!found) {
|
||||||
|
list.push(name, value);
|
||||||
|
}
|
||||||
|
|
||||||
update(this[context], this);
|
update(this[context], this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -714,10 +752,14 @@ class URLSearchParams {
|
||||||
throw new TypeError('The `name` argument needs to be specified');
|
throw new TypeError('The `name` argument needs to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
const list = this[searchParams];
|
||||||
name = String(name);
|
name = String(name);
|
||||||
var value = obj[name];
|
for (var i = 0; i < list.length; i += 2) {
|
||||||
return value === undefined ? null : Array.isArray(value) ? value[0] : value;
|
if (list[i] === name) {
|
||||||
|
return list[i + 1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
getAll(name) {
|
getAll(name) {
|
||||||
|
@ -728,10 +770,15 @@ class URLSearchParams {
|
||||||
throw new TypeError('The `name` argument needs to be specified');
|
throw new TypeError('The `name` argument needs to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
const list = this[searchParams];
|
||||||
|
const values = [];
|
||||||
name = String(name);
|
name = String(name);
|
||||||
var value = obj[name];
|
for (var i = 0; i < list.length; i += 2) {
|
||||||
return value === undefined ? [] : Array.isArray(value) ? value : [value];
|
if (list[i] === name) {
|
||||||
|
values.push(list[i + 1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return values;
|
||||||
}
|
}
|
||||||
|
|
||||||
has(name) {
|
has(name) {
|
||||||
|
@ -742,9 +789,14 @@ class URLSearchParams {
|
||||||
throw new TypeError('The `name` argument needs to be specified');
|
throw new TypeError('The `name` argument needs to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
const obj = this[searchParams];
|
const list = this[searchParams];
|
||||||
name = String(name);
|
name = String(name);
|
||||||
return name in obj;
|
for (var i = 0; i < list.length; i += 2) {
|
||||||
|
if (list[i] === name) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// https://heycam.github.io/webidl/#es-iterators
|
// https://heycam.github.io/webidl/#es-iterators
|
||||||
|
@ -766,14 +818,16 @@ class URLSearchParams {
|
||||||
throw new TypeError('The `callback` argument needs to be specified');
|
throw new TypeError('The `callback` argument needs to be specified');
|
||||||
}
|
}
|
||||||
|
|
||||||
let pairs = getSearchParamPairs(this);
|
let list = this[searchParams];
|
||||||
|
|
||||||
var i = 0;
|
var i = 0;
|
||||||
while (i < pairs.length) {
|
while (i < list.length) {
|
||||||
const [key, value] = pairs[i];
|
const key = list[i];
|
||||||
|
const value = list[i + 1];
|
||||||
callback.call(thisArg, value, key, this);
|
callback.call(thisArg, value, key, this);
|
||||||
pairs = getSearchParamPairs(this);
|
// in case the URL object's `search` is updated
|
||||||
i++;
|
list = this[searchParams];
|
||||||
|
i += 2;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -800,12 +854,38 @@ class URLSearchParams {
|
||||||
throw new TypeError('Value of `this` is not a URLSearchParams');
|
throw new TypeError('Value of `this` is not a URLSearchParams');
|
||||||
}
|
}
|
||||||
|
|
||||||
return querystring.stringify(this[searchParams]);
|
return querystring.stringify(getObjectFromParams(this[searchParams]));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// https://heycam.github.io/webidl/#es-iterable-entries
|
// https://heycam.github.io/webidl/#es-iterable-entries
|
||||||
URLSearchParams.prototype[Symbol.iterator] = URLSearchParams.prototype.entries;
|
URLSearchParams.prototype[Symbol.iterator] = URLSearchParams.prototype.entries;
|
||||||
|
|
||||||
|
URLSearchParams.prototype[util.inspect.custom] =
|
||||||
|
function inspect(recurseTimes, ctx) {
|
||||||
|
const innerOpts = Object.assign({}, ctx);
|
||||||
|
if (recurseTimes !== null) {
|
||||||
|
innerOpts.depth = recurseTimes - 1;
|
||||||
|
}
|
||||||
|
const innerInspect = (v) => util.inspect(v, innerOpts);
|
||||||
|
|
||||||
|
const list = this[searchParams];
|
||||||
|
const output = [];
|
||||||
|
for (var i = 0; i < list.length; i += 2)
|
||||||
|
output.push(`${innerInspect(list[i])} => ${innerInspect(list[i + 1])}`);
|
||||||
|
|
||||||
|
const colorRe = /\u001b\[\d\d?m/g;
|
||||||
|
const length = output.reduce((prev, cur) => {
|
||||||
|
return prev + cur.replace(colorRe, '').length + ', '.length;
|
||||||
|
}, -', '.length);
|
||||||
|
if (length > ctx.breakLength) {
|
||||||
|
return `${this.constructor.name} {\n ${output.join(',\n ')} }`;
|
||||||
|
} else if (output.length) {
|
||||||
|
return `${this.constructor.name} { ${output.join(', ')} }`;
|
||||||
|
} else {
|
||||||
|
return `${this.constructor.name} {}`;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// https://heycam.github.io/webidl/#dfn-default-iterator-object
|
// https://heycam.github.io/webidl/#dfn-default-iterator-object
|
||||||
function createSearchParamsIterator(target, kind) {
|
function createSearchParamsIterator(target, kind) {
|
||||||
const iterator = Object.create(URLSearchParamsIteratorPrototype);
|
const iterator = Object.create(URLSearchParamsIteratorPrototype);
|
||||||
|
@ -830,7 +910,7 @@ const URLSearchParamsIteratorPrototype = Object.setPrototypeOf({
|
||||||
kind,
|
kind,
|
||||||
index
|
index
|
||||||
} = this[context];
|
} = this[context];
|
||||||
const values = getSearchParamPairs(target);
|
const values = target[searchParams];
|
||||||
const len = values.length;
|
const len = values.length;
|
||||||
if (index >= len) {
|
if (index >= len) {
|
||||||
return {
|
return {
|
||||||
|
@ -839,22 +919,54 @@ const URLSearchParamsIteratorPrototype = Object.setPrototypeOf({
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const pair = values[index];
|
const name = values[index];
|
||||||
this[context].index = index + 1;
|
const value = values[index + 1];
|
||||||
|
this[context].index = index + 2;
|
||||||
|
|
||||||
let result;
|
let result;
|
||||||
if (kind === 'key') {
|
if (kind === 'key') {
|
||||||
result = pair[0];
|
result = name;
|
||||||
} else if (kind === 'value') {
|
} else if (kind === 'value') {
|
||||||
result = pair[1];
|
result = value;
|
||||||
} else {
|
} else {
|
||||||
result = pair;
|
result = [name, value];
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
value: result,
|
value: result,
|
||||||
done: false
|
done: false
|
||||||
};
|
};
|
||||||
|
},
|
||||||
|
[util.inspect.custom]: function inspect(recurseTimes, ctx) {
|
||||||
|
const innerOpts = Object.assign({}, ctx);
|
||||||
|
if (recurseTimes !== null) {
|
||||||
|
innerOpts.depth = recurseTimes - 1;
|
||||||
|
}
|
||||||
|
const {
|
||||||
|
target,
|
||||||
|
kind,
|
||||||
|
index
|
||||||
|
} = this[context];
|
||||||
|
const output = target[searchParams].slice(index).reduce((prev, cur, i) => {
|
||||||
|
const key = i % 2 === 0;
|
||||||
|
if (kind === 'key' && key) {
|
||||||
|
prev.push(cur);
|
||||||
|
} else if (kind === 'value' && !key) {
|
||||||
|
prev.push(cur);
|
||||||
|
} else if (kind === 'key+value' && !key) {
|
||||||
|
prev.push([target[searchParams][index + i - 1], cur]);
|
||||||
|
}
|
||||||
|
return prev;
|
||||||
|
}, []);
|
||||||
|
const breakLn = util.inspect(output, innerOpts).includes('\n');
|
||||||
|
const outputStrs = output.map((p) => util.inspect(p, innerOpts));
|
||||||
|
let outputStr;
|
||||||
|
if (breakLn) {
|
||||||
|
outputStr = `\n ${outputStrs.join(',\n ')}`;
|
||||||
|
} else {
|
||||||
|
outputStr = ` ${outputStrs.join(', ')}`;
|
||||||
|
}
|
||||||
|
return `${this[Symbol.toStringTag]} {${outputStr} }`;
|
||||||
}
|
}
|
||||||
}, IteratorPrototype);
|
}, IteratorPrototype);
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,30 @@
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
require('../common');
|
||||||
|
const assert = require('assert');
|
||||||
|
const util = require('util');
|
||||||
|
const URL = require('url').URL;
|
||||||
|
|
||||||
|
// Until we export URLSearchParams
|
||||||
|
const m = new URL('http://example.org');
|
||||||
|
const URLSearchParams = m.searchParams.constructor;
|
||||||
|
|
||||||
|
const sp = new URLSearchParams('?a=a&b=b&b=c');
|
||||||
|
assert.strictEqual(util.inspect(sp),
|
||||||
|
"URLSearchParams { 'a' => 'a', 'b' => 'b', 'b' => 'c' }");
|
||||||
|
assert.strictEqual(util.inspect(sp.keys()),
|
||||||
|
"URLSearchParamsIterator { 'a', 'b', 'b' }");
|
||||||
|
assert.strictEqual(util.inspect(sp.values()),
|
||||||
|
"URLSearchParamsIterator { 'a', 'b', 'c' }");
|
||||||
|
|
||||||
|
const iterator = sp.entries();
|
||||||
|
assert.strictEqual(util.inspect(iterator),
|
||||||
|
"URLSearchParamsIterator { [ 'a', 'a' ], [ 'b', 'b' ], " +
|
||||||
|
"[ 'b', 'c' ] }");
|
||||||
|
iterator.next();
|
||||||
|
assert.strictEqual(util.inspect(iterator),
|
||||||
|
"URLSearchParamsIterator { [ 'b', 'b' ], [ 'b', 'c' ] }");
|
||||||
|
iterator.next();
|
||||||
|
iterator.next();
|
||||||
|
assert.strictEqual(util.inspect(iterator),
|
||||||
|
'URLSearchParamsIterator { }');
|
Loading…
Reference in New Issue