@@ -4,48 +4,48 @@ const OriginalProxy = Proxy;
4
4
const { freeze, defineProperty, hasOwn } = OriginalObject ;
5
5
const { apply, construct, ownKeys } = OriginalReflect ;
6
6
7
- const noTrappingSet = new WeakSet ( ) ;
7
+ const nonTrappingSet = new WeakSet ( ) ;
8
8
9
9
const proxyHandlerMap = new WeakMap ( ) ;
10
10
11
11
const isPrimitive = specimen => OriginalObject ( specimen ) !== specimen ;
12
12
13
13
/**
14
- * Corresponds to the internal function shared by `Object.isNoTrapping ` and
15
- * `Reflect.isNoTrapping `.
14
+ * Corresponds to the internal function shared by `Object.isNonTrapping ` and
15
+ * `Reflect.isNonTrapping `.
16
16
*
17
17
* @param {any } specimen
18
18
* @param {boolean } shouldThrow
19
19
* @returns {boolean }
20
20
*/
21
- const isNoTrappingInternal = ( specimen , shouldThrow ) => {
22
- if ( noTrappingSet . has ( specimen ) ) {
21
+ const isNonTrappingInternal = ( specimen , shouldThrow ) => {
22
+ if ( nonTrappingSet . has ( specimen ) ) {
23
23
return true ;
24
24
}
25
25
if ( ! proxyHandlerMap . has ( specimen ) ) {
26
26
return false ;
27
27
}
28
28
const [ target , handler ] = proxyHandlerMap . get ( specimen ) ;
29
- if ( isNoTrappingInternal ( target , shouldThrow ) ) {
30
- noTrappingSet . add ( specimen ) ;
29
+ if ( isNonTrappingInternal ( target , shouldThrow ) ) {
30
+ nonTrappingSet . add ( specimen ) ;
31
31
return true ;
32
32
}
33
- const trap = handler . isNoTrapping ;
33
+ const trap = handler . isNonTrapping ;
34
34
if ( trap === undefined ) {
35
35
return false ;
36
36
}
37
37
const result = apply ( trap , handler , [ target ] ) ;
38
- const ofTarget = isNoTrappingInternal ( target , shouldThrow ) ;
38
+ const ofTarget = isNonTrappingInternal ( target , shouldThrow ) ;
39
39
if ( result !== ofTarget ) {
40
40
if ( shouldThrow ) {
41
41
throw TypeError (
42
- `'isNoTrapping ' proxy trap does not reflect 'isNoTrapping ' of proxy target (which is '${ ofTarget } ')` ,
42
+ `'isNonTrapping ' proxy trap does not reflect 'isNonTrapping ' of proxy target (which is '${ ofTarget } ')` ,
43
43
) ;
44
44
}
45
45
return false ;
46
46
}
47
47
if ( result ) {
48
- noTrappingSet . add ( specimen ) ;
48
+ nonTrappingSet . add ( specimen ) ;
49
49
}
50
50
return result ;
51
51
} ;
@@ -59,49 +59,49 @@ const isNoTrappingInternal = (specimen, shouldThrow) => {
59
59
* @returns {boolean }
60
60
*/
61
61
const suppressTrappingInternal = ( specimen , shouldThrow ) => {
62
- if ( noTrappingSet . has ( specimen ) ) {
62
+ if ( nonTrappingSet . has ( specimen ) ) {
63
63
return true ;
64
64
}
65
65
freeze ( specimen ) ;
66
66
if ( ! proxyHandlerMap . has ( specimen ) ) {
67
- noTrappingSet . add ( specimen ) ;
67
+ nonTrappingSet . add ( specimen ) ;
68
68
return true ;
69
69
}
70
70
const [ target , handler ] = proxyHandlerMap . get ( specimen ) ;
71
- if ( isNoTrappingInternal ( target , shouldThrow ) ) {
72
- noTrappingSet . add ( specimen ) ;
71
+ if ( isNonTrappingInternal ( target , shouldThrow ) ) {
72
+ nonTrappingSet . add ( specimen ) ;
73
73
return true ;
74
74
}
75
75
const trap = handler . suppressTrapping ;
76
76
if ( trap === undefined ) {
77
77
const result = suppressTrappingInternal ( target , shouldThrow ) ;
78
78
if ( result ) {
79
- noTrappingSet . add ( specimen ) ;
79
+ nonTrappingSet . add ( specimen ) ;
80
80
}
81
81
return result ;
82
82
}
83
83
const result = apply ( trap , handler , [ target ] ) ;
84
- const ofTarget = isNoTrappingInternal ( target , shouldThrow ) ;
84
+ const ofTarget = isNonTrappingInternal ( target , shouldThrow ) ;
85
85
if ( result !== ofTarget ) {
86
86
if ( shouldThrow ) {
87
87
throw TypeError (
88
- `'suppressTrapping' proxy trap does not reflect 'isNoTrapping ' of proxy target (which is '${ ofTarget } ')` ,
88
+ `'suppressTrapping' proxy trap does not reflect 'isNonTrapping ' of proxy target (which is '${ ofTarget } ')` ,
89
89
) ;
90
90
}
91
91
return false ;
92
92
}
93
93
if ( result ) {
94
- noTrappingSet . add ( specimen ) ;
94
+ nonTrappingSet . add ( specimen ) ;
95
95
}
96
96
return result ;
97
97
} ;
98
98
99
99
export const extraReflectMethods = freeze ( {
100
- isNoTrapping ( target ) {
100
+ isNonTrapping ( target ) {
101
101
if ( isPrimitive ( target ) ) {
102
- throw TypeError ( 'Reflect.isNoTrapping called on non-object' ) ;
102
+ throw TypeError ( 'Reflect.isNonTrapping called on non-object' ) ;
103
103
}
104
- return isNoTrappingInternal ( target , false ) ;
104
+ return isNonTrappingInternal ( target , false ) ;
105
105
} ,
106
106
suppressTrapping ( target ) {
107
107
if ( isPrimitive ( target ) ) {
@@ -112,11 +112,11 @@ export const extraReflectMethods = freeze({
112
112
} ) ;
113
113
114
114
export const extraObjectMethods = freeze ( {
115
- isNoTrapping ( target ) {
115
+ isNonTrapping ( target ) {
116
116
if ( isPrimitive ( target ) ) {
117
117
return true ;
118
118
}
119
- return isNoTrappingInternal ( target , true ) ;
119
+ return isNonTrappingInternal ( target , true ) ;
120
120
} ,
121
121
suppressTrapping ( target ) {
122
122
if ( isPrimitive ( target ) ) {
@@ -125,7 +125,7 @@ export const extraObjectMethods = freeze({
125
125
if ( suppressTrappingInternal ( target , true ) ) {
126
126
return target ;
127
127
}
128
- throw TypeError ( 'preventExtensions trap returned falsy' ) ;
128
+ throw TypeError ( 'suppressTrapping trap returned falsy' ) ;
129
129
} ,
130
130
} ) ;
131
131
@@ -167,6 +167,17 @@ ObjectPlus.prototype = OriginalObject.prototype;
167
167
addExtras ( ObjectPlus , OriginalObject , extraObjectMethods ) ;
168
168
export { ObjectPlus } ;
169
169
170
+ /**
171
+ * A way to store the `originalHandler` on the `handlerPlus` without
172
+ * possible conflict with an future trap name.
173
+ *
174
+ * Normally, we'd use a WeakMap for this, so the property is also
175
+ * undiscoverable. But in this case, the `handlerPlus` objects are
176
+ * safely encapsulated within this module, so no one is in a position to
177
+ * discovery this property by inspection.
178
+ */
179
+ const ORIGINAL_HANDLER = Symbol ( 'OriginalHandler' ) ;
180
+
170
181
const metaHandler = freeze ( {
171
182
get ( _ , trapName , handlerPlus ) {
172
183
/**
@@ -182,7 +193,7 @@ const metaHandler = freeze({
182
193
* @param {any[] } rest
183
194
*/
184
195
const trapPlus = freeze ( ( target , ...rest ) => {
185
- if ( isNoTrappingInternal ( target , true ) ) {
196
+ if ( isNonTrappingInternal ( target , true ) ) {
186
197
defineProperty ( handlerPlus , trapName , {
187
198
value : undefined ,
188
199
writable : false ,
@@ -198,7 +209,7 @@ const metaHandler = freeze({
198
209
configurable : true ,
199
210
} ) ;
200
211
}
201
- const { originalHandler } = handlerPlus ;
212
+ const { [ ORIGINAL_HANDLER ] : originalHandler } = handlerPlus ;
202
213
const trap = originalHandler [ trapName ] ;
203
214
if ( trap !== undefined ) {
204
215
// Note that whether `trap === undefined` can change dynamically,
@@ -225,27 +236,19 @@ const metaHandler = freeze({
225
236
*
226
237
* @param {ProxyHandler<any> } originalHandler
227
238
* @returns {ProxyHandler<any> & {
228
- * isNoTrapping : (target: any) => boolean,
239
+ * isNonTrapping : (target: any) => boolean,
229
240
* suppressTrapping: (target: any) => boolean,
230
241
* originalHandler: ProxyHandler<any>
231
242
* }}
232
243
*/
233
244
const makeHandlerPlus = originalHandler => ( {
234
245
// @ts -expect-error TS does not know what this __proto__ is doing
235
246
__proto__ : new OriginalProxy ( { } , metaHandler ) ,
236
- // relies on there never being a trap named `originalHandler`.
237
- originalHandler,
247
+ [ ORIGINAL_HANDLER ] : originalHandler ,
238
248
} ) ;
239
249
240
- /**
241
- * In the shim, `ProxyPlus` replaces the global `Proxy`.
242
- *
243
- * @type {ProxyConstructor }
244
- */
245
- // @ts -expect-error We reject non-new calls in the body
246
- const ProxyPlus = function Proxy ( target , handler ) {
247
- // @ts -expect-error Yes, we mean to compare these.
248
- if ( new . target !== ProxyPlus ) {
250
+ const ProxyInternal = function Proxy ( target , handler ) {
251
+ if ( new . target !== ProxyInternal ) {
249
252
if ( new . target === undefined ) {
250
253
throw TypeError ( 'Proxy constructor requires "new"' ) ;
251
254
}
@@ -256,18 +259,26 @@ const ProxyPlus = function Proxy(target, handler) {
256
259
proxyHandlerMap . set ( proxy , [ target , handler ] ) ;
257
260
return proxy ;
258
261
} ;
259
- // The `OriginalProxy` is both constructible (i.e., responsive to `new`) and
260
- // lacks a `prototype` property. The closest we can come to this is to set
261
- // `ProxyPlus.prototype` to `undefined`
262
- ProxyPlus . prototype = undefined ;
262
+
263
+ /**
264
+ * In the shim, `ProxyPlus` replaces the global `Proxy`.
265
+ *
266
+ * We use `bind` as the only way for user code to produce a
267
+ * constructible function (i.e., one that responds to `new`) without a
268
+ * `.prototype` property.
269
+ *
270
+ * @type {ProxyConstructor }
271
+ */
272
+ const ProxyPlus = ProxyInternal . bind ( undefined ) ;
273
+
263
274
ProxyPlus . revocable = ( target , handler ) => {
264
275
const handlerPlus = makeHandlerPlus ( handler ) ;
265
276
const { proxy, revoke } = OriginalProxy . revocable ( target , handlerPlus ) ;
266
277
proxyHandlerMap . set ( proxy , [ target , handler ] ) ;
267
278
return {
268
279
proxy,
269
280
revoke ( ) {
270
- if ( isNoTrappingInternal ( target , true ) ) {
281
+ if ( isNonTrappingInternal ( target , true ) ) {
271
282
throw TypeError ( 'Cannot revoke non-trapping proxy' ) ;
272
283
}
273
284
revoke ( ) ;
0 commit comments