From 454dd8ee1ff7d649b46cd8e8f988a3b268156e09 Mon Sep 17 00:00:00 2001 From: Mike Bostock Date: Tue, 16 Apr 2024 14:24:39 -0700 Subject: [PATCH] GH-40407: [JS] Fix string coercion in MapRowProxyHandler.ownKeys (#40408) --- js/src/row/map.ts | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/js/src/row/map.ts b/js/src/row/map.ts index d0ccb872ec33c..9f826f083b355 100644 --- a/js/src/row/map.ts +++ b/js/src/row/map.ts @@ -24,6 +24,8 @@ import { instance as setVisitor } from '../visitor/set.js'; /** @ignore */ export const kKeys = Symbol.for('keys'); /** @ignore */ export const kVals = Symbol.for('vals'); +/** @ignore */ export const kKeysAsStrings = Symbol.for('kKeysAsStrings'); +/** @ignore */ export const _kKeysAsStrings = Symbol.for('_kKeysAsStrings'); export class MapRow { @@ -31,6 +33,7 @@ export class MapRow { declare private [kKeys]: Vector; declare private [kVals]: Data; + declare private [_kKeysAsStrings]: string[]; constructor(slice: Data>) { this[kKeys] = new Vector([slice.children[0]]).memoize() as Vector; @@ -38,6 +41,11 @@ export class MapRow { return new Proxy(this, new MapRowProxyHandler()); } + /** @ignore */ + get [kKeysAsStrings]() { + return this[_kKeysAsStrings] || (this[_kKeysAsStrings] = Array.from(this[kKeys].toArray(), String)); + } + [Symbol.iterator]() { return new MapRowIterator(this[kKeys], this[kVals]); } @@ -107,13 +115,13 @@ class MapRowProxyHandler imp deleteProperty() { return false; } preventExtensions() { return true; } ownKeys(row: MapRow) { - return row[kKeys].toArray().map(String); + return row[kKeysAsStrings]; } has(row: MapRow, key: string | symbol) { - return row[kKeys].includes(key); + return row[kKeysAsStrings].includes(key as string); } getOwnPropertyDescriptor(row: MapRow, key: string | symbol) { - const idx = row[kKeys].indexOf(key); + const idx = row[kKeysAsStrings].indexOf(key as string); if (idx !== -1) { return { writable: true, enumerable: true, configurable: true }; } @@ -124,7 +132,7 @@ class MapRowProxyHandler imp if (Reflect.has(row, key)) { return (row as any)[key]; } - const idx = row[kKeys].indexOf(key); + const idx = row[kKeysAsStrings].indexOf(key as string); if (idx !== -1) { const val = getVisitor.visit(Reflect.get(row, kVals), idx); // Cache key/val lookups @@ -133,7 +141,7 @@ class MapRowProxyHandler imp } } set(row: MapRow, key: string | symbol, val: V) { - const idx = row[kKeys].indexOf(key); + const idx = row[kKeysAsStrings].indexOf(key as string); if (idx !== -1) { setVisitor.visit(Reflect.get(row, kVals), idx, val); // Cache key/val lookups @@ -149,4 +157,5 @@ Object.defineProperties(MapRow.prototype, { [Symbol.toStringTag]: { enumerable: false, configurable: false, value: 'Row' }, [kKeys]: { writable: true, enumerable: false, configurable: false, value: null }, [kVals]: { writable: true, enumerable: false, configurable: false, value: null }, + [_kKeysAsStrings]: { writable: true, enumerable: false, configurable: false, value: null }, });