From e727eb066fd78c40d95db7ea97b2b3166e062088 Mon Sep 17 00:00:00 2001 From: Timothy Gu Date: Thu, 6 Apr 2023 20:32:26 -0700 Subject: [PATCH] url: do not use object as hashmap Fixes cases like new URLSearchParams({ hasOwnProperty: 1 }). PR-URL: https://github.com/nodejs/node/pull/47415 Reviewed-By: Yagiz Nizipli Reviewed-By: Luigi Pinca Reviewed-By: Benjamin Gruenbaum --- lib/internal/url.js | 18 ++++++++++-------- ...atwg-url-custom-searchparams-constructor.js | 5 +++++ 2 files changed, 15 insertions(+), 8 deletions(-) diff --git a/lib/internal/url.js b/lib/internal/url.js index 7f043aca793422..0c60757b1efd89 100644 --- a/lib/internal/url.js +++ b/lib/internal/url.js @@ -20,6 +20,7 @@ const { ReflectGetOwnPropertyDescriptor, ReflectOwnKeys, RegExpPrototypeSymbolReplace, + SafeMap, StringPrototypeCharAt, StringPrototypeCharCodeAt, StringPrototypeCodePointAt, @@ -243,7 +244,7 @@ class URLSearchParams { } else { // Record // Need to use reflection APIs for full spec compliance. - const visited = {}; + const visited = new SafeMap(); const keys = ReflectOwnKeys(init); for (let i = 0; i < keys.length; i++) { const key = keys[i]; @@ -252,14 +253,15 @@ class URLSearchParams { const typedKey = toUSVString(key); const typedValue = toUSVString(init[key]); - // Two different key may result same after `toUSVString()`, we only - // leave the later one. Refers to WPT. - if (visited[typedKey] !== undefined) { - this[searchParams][visited[typedKey]] = typedValue; + // Two different keys may become the same USVString after normalization. + // In that case, we retain the later one. Refer to WPT. + const keyIdx = visited.get(typedKey); + if (keyIdx !== undefined) { + this[searchParams][keyIdx] = typedValue; } else { - visited[typedKey] = ArrayPrototypePush(this[searchParams], - typedKey, - typedValue) - 1; + visited.set(typedKey, ArrayPrototypePush(this[searchParams], + typedKey, + typedValue) - 1); } } } diff --git a/test/parallel/test-whatwg-url-custom-searchparams-constructor.js b/test/parallel/test-whatwg-url-custom-searchparams-constructor.js index 1b7409680b2a2a..75888f9270d25b 100644 --- a/test/parallel/test-whatwg-url-custom-searchparams-constructor.js +++ b/test/parallel/test-whatwg-url-custom-searchparams-constructor.js @@ -38,8 +38,13 @@ function makeIterableFunc(array) { makeIterableFunc([['key', 'val'], ['key2', 'val2']].map(makeIterableFunc)) ); assert.strictEqual(params.toString(), 'key=val&key2=val2'); + params = new URLSearchParams({ hasOwnProperty: 1 }); + assert.strictEqual(params.get('hasOwnProperty'), '1'); + assert.strictEqual(params.toString(), 'hasOwnProperty=1'); assert.throws(() => new URLSearchParams([[1]]), tupleError); assert.throws(() => new URLSearchParams([[1, 2, 3]]), tupleError); + assert.throws(() => new URLSearchParams({ [Symbol('test')]: 42 }), + TypeError); assert.throws(() => new URLSearchParams({ [Symbol.iterator]: 42 }), iterableError); assert.throws(() => new URLSearchParams([{}]), tupleError);