Skip to content

Commit b22b8b0

Browse files
committed
fixup! reveiwer suggestions
1 parent 6445345 commit b22b8b0

16 files changed

+81
-59
lines changed

packages/no-trapping-shim/README.md

-1
This file was deleted.

packages/no-trapping-shim/index.js

-1
This file was deleted.
File renamed without changes.
File renamed without changes.
File renamed without changes.

packages/non-trapping-shim/README.md

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
# Ponyfill and Shim for Non-trapping Integrity Trait
2+
3+
Emulates support for the non-trapping integrity trait from the
4+
[Stabilize proposal](https://github.com/tc39/proposal-stabilize).
5+
6+
A *shim* attempts to be as full fidelity as practical an emulation of the proposal itself. This is sometimes called a "polyfill". A *ponyfill* provides the functionality of the shim, but sacrifices fidelity of the API in order to avoid monkey patching the primordials. Confusingly, this is also sometimes called a "polyfill", which is why we avoid that ambiguous term.
7+
8+
A shim typically "exports" its functionality by adding properties to primordial objects. A ponyfill typically exports its functionality by explicit module exports, to be explicitly imported by code wishing to use it.
9+
10+
See https://github.com/endojs/endo/blob/master/packages/ses/docs/preparing-for-stabilize.md for guidance on how to prepare for the changes that will be introduced by this proposal.
11+
12+
TODO: More explanation.
File renamed without changes.

packages/non-trapping-shim/index.js

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
export * from './src/non-trapping-pony.js';

packages/no-trapping-shim/package.json packages/non-trapping-shim/package.json

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,16 @@
11
{
2-
"name": "@endo/no-trapping-shim",
2+
"name": "@endo/non-trapping-shim",
33
"version": "0.1.0",
44
"private": true,
5-
"description": "shim and ponyfill for no-trapping integrity level",
5+
"description": "shim and ponyfill for non-trapping integrity trait",
66
"keywords": [],
77
"author": "Endo contributors",
88
"license": "Apache-2.0",
99
"homepage": "https://github.com/endojs/endo/tree/master/packages/skel#readme",
1010
"repository": {
1111
"type": "git",
1212
"url": "git+https://github.com/endojs/endo.git",
13-
"directory": "packages/no-trapping-shim"
13+
"directory": "packages/non-trapping-shim"
1414
},
1515
"bugs": {
1616
"url": "https://github.com/endojs/endo/issues"

packages/no-trapping-shim/shim.js packages/non-trapping-shim/shim.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/* global globalThis */
2-
import { ReflectPlus, ObjectPlus, ProxyPlus } from './src/no-trapping-pony.js';
2+
import { ReflectPlus, ObjectPlus, ProxyPlus } from './src/non-trapping-pony.js';
33

44
globalThis.Reflect = ReflectPlus;
55

packages/no-trapping-shim/src/no-trapping-pony.js packages/non-trapping-shim/src/non-trapping-pony.js

+55-44
Original file line numberDiff line numberDiff line change
@@ -4,48 +4,48 @@ const OriginalProxy = Proxy;
44
const { freeze, defineProperty, hasOwn } = OriginalObject;
55
const { apply, construct, ownKeys } = OriginalReflect;
66

7-
const noTrappingSet = new WeakSet();
7+
const nonTrappingSet = new WeakSet();
88

99
const proxyHandlerMap = new WeakMap();
1010

1111
const isPrimitive = specimen => OriginalObject(specimen) !== specimen;
1212

1313
/**
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`.
1616
*
1717
* @param {any} specimen
1818
* @param {boolean} shouldThrow
1919
* @returns {boolean}
2020
*/
21-
const isNoTrappingInternal = (specimen, shouldThrow) => {
22-
if (noTrappingSet.has(specimen)) {
21+
const isNonTrappingInternal = (specimen, shouldThrow) => {
22+
if (nonTrappingSet.has(specimen)) {
2323
return true;
2424
}
2525
if (!proxyHandlerMap.has(specimen)) {
2626
return false;
2727
}
2828
const [target, handler] = proxyHandlerMap.get(specimen);
29-
if (isNoTrappingInternal(target, shouldThrow)) {
30-
noTrappingSet.add(specimen);
29+
if (isNonTrappingInternal(target, shouldThrow)) {
30+
nonTrappingSet.add(specimen);
3131
return true;
3232
}
33-
const trap = handler.isNoTrapping;
33+
const trap = handler.isNonTrapping;
3434
if (trap === undefined) {
3535
return false;
3636
}
3737
const result = apply(trap, handler, [target]);
38-
const ofTarget = isNoTrappingInternal(target, shouldThrow);
38+
const ofTarget = isNonTrappingInternal(target, shouldThrow);
3939
if (result !== ofTarget) {
4040
if (shouldThrow) {
4141
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}')`,
4343
);
4444
}
4545
return false;
4646
}
4747
if (result) {
48-
noTrappingSet.add(specimen);
48+
nonTrappingSet.add(specimen);
4949
}
5050
return result;
5151
};
@@ -59,49 +59,49 @@ const isNoTrappingInternal = (specimen, shouldThrow) => {
5959
* @returns {boolean}
6060
*/
6161
const suppressTrappingInternal = (specimen, shouldThrow) => {
62-
if (noTrappingSet.has(specimen)) {
62+
if (nonTrappingSet.has(specimen)) {
6363
return true;
6464
}
6565
freeze(specimen);
6666
if (!proxyHandlerMap.has(specimen)) {
67-
noTrappingSet.add(specimen);
67+
nonTrappingSet.add(specimen);
6868
return true;
6969
}
7070
const [target, handler] = proxyHandlerMap.get(specimen);
71-
if (isNoTrappingInternal(target, shouldThrow)) {
72-
noTrappingSet.add(specimen);
71+
if (isNonTrappingInternal(target, shouldThrow)) {
72+
nonTrappingSet.add(specimen);
7373
return true;
7474
}
7575
const trap = handler.suppressTrapping;
7676
if (trap === undefined) {
7777
const result = suppressTrappingInternal(target, shouldThrow);
7878
if (result) {
79-
noTrappingSet.add(specimen);
79+
nonTrappingSet.add(specimen);
8080
}
8181
return result;
8282
}
8383
const result = apply(trap, handler, [target]);
84-
const ofTarget = isNoTrappingInternal(target, shouldThrow);
84+
const ofTarget = isNonTrappingInternal(target, shouldThrow);
8585
if (result !== ofTarget) {
8686
if (shouldThrow) {
8787
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}')`,
8989
);
9090
}
9191
return false;
9292
}
9393
if (result) {
94-
noTrappingSet.add(specimen);
94+
nonTrappingSet.add(specimen);
9595
}
9696
return result;
9797
};
9898

9999
export const extraReflectMethods = freeze({
100-
isNoTrapping(target) {
100+
isNonTrapping(target) {
101101
if (isPrimitive(target)) {
102-
throw TypeError('Reflect.isNoTrapping called on non-object');
102+
throw TypeError('Reflect.isNonTrapping called on non-object');
103103
}
104-
return isNoTrappingInternal(target, false);
104+
return isNonTrappingInternal(target, false);
105105
},
106106
suppressTrapping(target) {
107107
if (isPrimitive(target)) {
@@ -112,11 +112,11 @@ export const extraReflectMethods = freeze({
112112
});
113113

114114
export const extraObjectMethods = freeze({
115-
isNoTrapping(target) {
115+
isNonTrapping(target) {
116116
if (isPrimitive(target)) {
117117
return true;
118118
}
119-
return isNoTrappingInternal(target, true);
119+
return isNonTrappingInternal(target, true);
120120
},
121121
suppressTrapping(target) {
122122
if (isPrimitive(target)) {
@@ -125,7 +125,7 @@ export const extraObjectMethods = freeze({
125125
if (suppressTrappingInternal(target, true)) {
126126
return target;
127127
}
128-
throw TypeError('preventExtensions trap returned falsy');
128+
throw TypeError('suppressTrapping trap returned falsy');
129129
},
130130
});
131131

@@ -167,6 +167,17 @@ ObjectPlus.prototype = OriginalObject.prototype;
167167
addExtras(ObjectPlus, OriginalObject, extraObjectMethods);
168168
export { ObjectPlus };
169169

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+
170181
const metaHandler = freeze({
171182
get(_, trapName, handlerPlus) {
172183
/**
@@ -182,7 +193,7 @@ const metaHandler = freeze({
182193
* @param {any[]} rest
183194
*/
184195
const trapPlus = freeze((target, ...rest) => {
185-
if (isNoTrappingInternal(target, true)) {
196+
if (isNonTrappingInternal(target, true)) {
186197
defineProperty(handlerPlus, trapName, {
187198
value: undefined,
188199
writable: false,
@@ -198,7 +209,7 @@ const metaHandler = freeze({
198209
configurable: true,
199210
});
200211
}
201-
const { originalHandler } = handlerPlus;
212+
const { [ORIGINAL_HANDLER]: originalHandler } = handlerPlus;
202213
const trap = originalHandler[trapName];
203214
if (trap !== undefined) {
204215
// Note that whether `trap === undefined` can change dynamically,
@@ -225,27 +236,19 @@ const metaHandler = freeze({
225236
*
226237
* @param {ProxyHandler<any>} originalHandler
227238
* @returns {ProxyHandler<any> & {
228-
* isNoTrapping: (target: any) => boolean,
239+
* isNonTrapping: (target: any) => boolean,
229240
* suppressTrapping: (target: any) => boolean,
230241
* originalHandler: ProxyHandler<any>
231242
* }}
232243
*/
233244
const makeHandlerPlus = originalHandler => ({
234245
// @ts-expect-error TS does not know what this __proto__ is doing
235246
__proto__: new OriginalProxy({}, metaHandler),
236-
// relies on there never being a trap named `originalHandler`.
237-
originalHandler,
247+
[ORIGINAL_HANDLER]: originalHandler,
238248
});
239249

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) {
249252
if (new.target === undefined) {
250253
throw TypeError('Proxy constructor requires "new"');
251254
}
@@ -256,18 +259,26 @@ const ProxyPlus = function Proxy(target, handler) {
256259
proxyHandlerMap.set(proxy, [target, handler]);
257260
return proxy;
258261
};
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+
263274
ProxyPlus.revocable = (target, handler) => {
264275
const handlerPlus = makeHandlerPlus(handler);
265276
const { proxy, revoke } = OriginalProxy.revocable(target, handlerPlus);
266277
proxyHandlerMap.set(proxy, [target, handler]);
267278
return {
268279
proxy,
269280
revoke() {
270-
if (isNoTrappingInternal(target, true)) {
281+
if (isNonTrappingInternal(target, true)) {
271282
throw TypeError('Cannot revoke non-trapping proxy');
272283
}
273284
revoke();

packages/no-trapping-shim/test/no-trapping-pony.test.js packages/non-trapping-shim/test/non-trapping-pony.test.js

+4-4
Original file line numberDiff line numberDiff line change
@@ -2,11 +2,11 @@
22
// dependencies. We will need similar tests is higher level packages, in order
33
// to test compat with ses and ses-ava.
44
import test from 'ava';
5-
import { ReflectPlus, ProxyPlus } from '../src/no-trapping-pony.js';
5+
import { ReflectPlus, ProxyPlus } from '../src/non-trapping-pony.js';
66

77
const { freeze, isFrozen } = Object;
88

9-
test('no-trapping-pony', async t => {
9+
test('non-trapping-pony', async t => {
1010
const specimen = { foo: 8 };
1111

1212
const sillyHandler = freeze({
@@ -17,13 +17,13 @@ test('no-trapping-pony', async t => {
1717

1818
const safeProxy = new ProxyPlus(specimen, sillyHandler);
1919

20-
t.false(ReflectPlus.isNoTrapping(specimen));
20+
t.false(ReflectPlus.isNonTrapping(specimen));
2121
t.false(isFrozen(specimen));
2222
t.deepEqual(safeProxy.foo, [specimen, 'foo', safeProxy]);
2323

2424
t.true(ReflectPlus.suppressTrapping(specimen));
2525

26-
t.true(ReflectPlus.isNoTrapping(specimen));
26+
t.true(ReflectPlus.isNonTrapping(specimen));
2727
t.true(isFrozen(specimen));
2828
t.deepEqual(safeProxy.foo, 8);
2929
});

packages/no-trapping-shim/test/no-trapping-shim.test.js packages/non-trapping-shim/test/non-trapping-shim.test.js

+3-3
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import '../shim.js';
66

77
const { freeze, isFrozen } = Object;
88

9-
test('no-trapping-pony', async t => {
9+
test('non-trapping-pony', async t => {
1010
const specimen = { foo: 8 };
1111

1212
const sillyHandler = freeze({
@@ -17,13 +17,13 @@ test('no-trapping-pony', async t => {
1717

1818
const safeProxy = new Proxy(specimen, sillyHandler);
1919

20-
t.false(Reflect.isNoTrapping(specimen));
20+
t.false(Reflect.isNonTrapping(specimen));
2121
t.false(isFrozen(specimen));
2222
t.deepEqual(safeProxy.foo, [specimen, 'foo', safeProxy]);
2323

2424
t.true(Reflect.suppressTrapping(specimen));
2525

26-
t.true(Reflect.isNoTrapping(specimen));
26+
t.true(Reflect.isNonTrapping(specimen));
2727
t.true(isFrozen(specimen));
2828
t.deepEqual(safeProxy.foo, 8);
2929
});

yarn.lock

+2-2
Original file line numberDiff line numberDiff line change
@@ -702,9 +702,9 @@ __metadata:
702702
languageName: unknown
703703
linkType: soft
704704

705-
"@endo/no-trapping-shim@workspace:packages/no-trapping-shim":
705+
"@endo/non-trapping-shim@workspace:packages/non-trapping-shim":
706706
version: 0.0.0-use.local
707-
resolution: "@endo/no-trapping-shim@workspace:packages/no-trapping-shim"
707+
resolution: "@endo/non-trapping-shim@workspace:packages/non-trapping-shim"
708708
dependencies:
709709
ava: "npm:^6.1.3"
710710
c8: "npm:^7.14.0"

0 commit comments

Comments
 (0)