From 4c67fe2cbcfd2737cb389b3fb0e358f4eed58af0 Mon Sep 17 00:00:00 2001 From: "Mark S. Miller" Date: Wed, 3 Jan 2024 12:08:43 -0800 Subject: [PATCH] fix(exo): reform exo amplifier API (#1924) --- packages/exo/src/exo-makers.js | 104 ++++++++++++++---- .../exo/test/test-amplify-heap-class-kits.js | 23 ++-- 2 files changed, 89 insertions(+), 38 deletions(-) diff --git a/packages/exo/src/exo-makers.js b/packages/exo/src/exo-makers.js index 14b307e0be..934b7346cc 100644 --- a/packages/exo/src/exo-makers.js +++ b/packages/exo/src/exo-makers.js @@ -70,37 +70,93 @@ export const initEmpty = () => emptyRecord; */ /** - * @callback Revoker - * @param {any} exo - * @returns {boolean} - */ - -/** - * @callback ReceiveRevoker - * @param {Revoker} revoke - * @returns {void} + * Template for function-valued options for exo class or exo class kit + * definitions, for receiving powers back at definition time. For example, + * ```js + * let amplify; + * const makeFoo = defineExoClassKit( + * tag, + * interfaceGuardKit, + * initFn, + * behaviorKit, + * { + * receiveAmplifier(a) { amplify = a; }, + * }, + * ); + * ``` + * uses the `receiveAmplifier` option to receive, during the + * definition of this exo class kit, the power to amplify a facet of the kit. + * + * @template {any} P + * @typedef {(power: P) => void} ReceivePower */ /** - * @callback Amplifier + * The power to revoke a live instance of the associated exo class, or the + * power to revoke a live facet instance of the associated exo class kit. + * If called with such a live instance, it revokes it and returns true. Once + * revoked, it is no longer live, and calling any of its methods throw + * an informative diagnostic with no further effects. + * + * @callback Revoke * @param {any} exo - * @param {string} facetName - * @returns {any} + * @returns {boolean} */ /** - * @callback ReceiveAmplifier - * @param {Amplifier} amplifier - * @returns {void} + * The power to amplify a live facet instance of the associated exo class kit + * into the record of all facets of this facet instance's cohort. + * + * @template {any} [F=any] + * @callback Amplify + * @param {any} exoFacet + * @returns {F} */ +// TODO Should we split FarClassOptions into distinct types for +// class options vs class kit options? After all, `receiveAmplifier` +// makes no sense for normal exo classes. /** - * @template C + * Currently, this one options type is used both for regular exo classes + * as well as exo class kits. However, we may split these into distinct types + * in the future, as not all options make sense for both uses. + * + * @template {any} C + * @template {any} [F=any] * @typedef {object} FarClassOptions * @property {(context: C) => void} [finish] + * If provided, the `finish` function is called after the instance has been + * initialized and registered, but before it is returned. Try to avoid using + * `finish` if you can, as we think we'd like to deprecate and retire it. + * OTOH, if you encounter a compelling need, please let us know so we can + * revise our plans. + * * @property {StateShape} [stateShape] - * @property {ReceiveRevoker} [receiveRevoker] - * @property {ReceiveAmplifier} [receiveAmplifier] + * If provided, it must be a RecordPattern, i.e., a CopyRecord which is also + * a Pattern. It thus has an exactly defined set of property names and + * a Pattern as the value of each property. This is supposed to be an invariant + * on the properties of an instance state record. + * TODO Though note that only the virtual and durable exos currently + * enforce the `stateShape` invariant. The heap exos defined in this + * package currently ignore `stateShape`, but will enforce this in the future. + * + * @property {ReceivePower} [receiveRevoker] + * If a `receiveRevoker` function is provided, it will be called during + * definition of the exo class or exo class kit with a `Revoke` function. + * A `Revoke` function is a function of one argument. If you call the revoke + * function with a live instance of this exo class, or a live facet instance + * of this exo class kit, then it will "revoke" it and return true. Once + * revoked, this instance is no longer "live": Any attempt to invoke any of + * its methods will fail without further effect. + * + * @property {ReceivePower>} [receiveAmplifier] + * If a `receiveAmplifier` function is provided, it will be called during + * definition of the exo class kit with an `Amplify` function. If called + * during the definition of a normal exo or exo class, it will throw, since + * only exo kits can be amplified. + * An `Amplify` function is a function that takes a live facet instance of + * this class kit as an argument, in which case it will return the facets + * record, giving access to all the facet instances of the same cohort. */ /** @@ -194,7 +250,10 @@ harden(defineExoClass); * } | undefined} interfaceGuardKit * @param {I} init * @param {F & { [K in keyof F]: ThisType<{ facets: GuardedKit, state: ReturnType }> }} methodsKit - * @param {FarClassOptions, GuardedKit>>} [options] + * @param {FarClassOptions< + * KitContext, GuardedKit>, + * GuardedKit + * >} [options] * @returns {(...args: Parameters) => GuardedKit} */ export const defineExoClassKit = ( @@ -255,12 +314,11 @@ export const defineExoClassKit = ( } if (receiveAmplifier) { - const amplify = (aFacet, facetName) => { + const amplify = aFacet => { for (const contextMap of values(contextMapKit)) { if (contextMap.has(aFacet)) { - const otherFacet = contextMap.get(aFacet).facets[facetName]; - otherFacet || Fail`${q(facetName)} must be a facet name of ${q(tag)}`; - return otherFacet; + const { facets } = contextMap.get(aFacet); + return facets; } } throw Fail`Must be an unrevoked facet of ${q(tag)}: ${aFacet}`; diff --git a/packages/exo/test/test-amplify-heap-class-kits.js b/packages/exo/test/test-amplify-heap-class-kits.js index 06544866ed..581667f27c 100644 --- a/packages/exo/test/test-amplify-heap-class-kits.js +++ b/packages/exo/test/test-amplify-heap-class-kits.js @@ -73,34 +73,27 @@ test('test amplify defineExoClassKit', t => { }, }, ); - const { up: upCounter, down: downCounter } = makeCounterKit(3); + const counterKit = makeCounterKit(3); + const { up: upCounter, down: downCounter } = counterKit; t.is(upCounter.incr(5), 8); t.is(downCounter.decr(), 7); - t.throws(() => amp(upCounter, 'sideways'), { - message: '"sideways" must be a facet name of "Counter"', - }); - t.throws(() => amp(harden({}), 'down'), { + t.throws(() => amp(harden({})), { message: 'Must be an unrevoked facet of "Counter": {}', }); - t.is(amp(upCounter, 'down'), downCounter); - t.is(amp(upCounter, 'up'), upCounter); - t.is(amp(downCounter, 'up'), upCounter); - t.is(amp(downCounter, 'down'), downCounter); + t.deepEqual(amp(upCounter), counterKit); + t.deepEqual(amp(downCounter), counterKit); t.is(revoke(upCounter), true); - t.throws(() => amp(upCounter, 'down'), { - message: 'Must be an unrevoked facet of "Counter": "[Alleged: Counter up]"', - }); - t.throws(() => amp(upCounter, 'up'), { + t.throws(() => amp(upCounter), { message: 'Must be an unrevoked facet of "Counter": "[Alleged: Counter up]"', }); - t.is(amp(downCounter, 'up'), upCounter); + t.deepEqual(amp(downCounter), counterKit); t.throws(() => upCounter.incr(3), { message: '"In \\"incr\\" method of (Counter up)" may only be applied to a valid instance: "[Alleged: Counter up]"', }); - t.is(amp(downCounter, 'down'), downCounter); + t.deepEqual(amp(downCounter), counterKit); t.is(downCounter.decr(), 6); });