diff --git a/a3p-integration/README.md b/a3p-integration/README.md index 660c72ad36e..8442a58b8a8 100644 --- a/a3p-integration/README.md +++ b/a3p-integration/README.md @@ -102,9 +102,9 @@ code, and must be rebuilt every time there is a change. The `/scripts/generate-a3p-submission.sh` script contains commands to generate the core-eval content and move it to the expected proposal package's submission directory. It is executed as part of `a3p-integration`'s `build:submission` step. -Each proposal that requires such a build step should add a `submission` entry in -the `agoricProposal` section of package.json. This value specifies the name of -another entry which gives parameters to be passed to `generate-a3p-submission.sh`. +Each proposal that requires such a build step should add a `build:submission` +rule in its package.json to specify the details of proposals that require a +build step. ## Building synthetic-chain images diff --git a/a3p-integration/proposals/a:upgrade-next/package.json b/a3p-integration/proposals/a:upgrade-next/package.json index 067a04a3d5d..3222bead2b6 100644 --- a/a3p-integration/proposals/a:upgrade-next/package.json +++ b/a3p-integration/proposals/a:upgrade-next/package.json @@ -24,7 +24,8 @@ ] }, "scripts": { - "agops": "yarn --cwd /usr/src/agoric-sdk/ --silent agops" + "agops": "yarn --cwd /usr/src/agoric-sdk/ --silent agops", + "build:submission": "../../../scripts/generate-a3p-submission.sh probe-zcf-bundle a:upgrade-next probeZcfBundle probe-submission" }, "packageManager": "yarn@4.1.0" } diff --git a/a3p-integration/proposals/b:localchain/package.json b/a3p-integration/proposals/b:localchain/package.json index 057136a23d2..97a94ab3ba1 100644 --- a/a3p-integration/proposals/b:localchain/package.json +++ b/a3p-integration/proposals/b:localchain/package.json @@ -17,5 +17,8 @@ "!submission" ] }, + "scripts": { + "build:submission": "../../../scripts/generate-a3p-submission.sh test-localchain b:localchain" + }, "packageManager": "yarn@4.1.0" } diff --git a/packages/zoe/src/contractFacet/exit.js b/packages/zoe/src/contractFacet/exit.js index 2d541ba2806..ac1cd92791e 100644 --- a/packages/zoe/src/contractFacet/exit.js +++ b/packages/zoe/src/contractFacet/exit.js @@ -35,6 +35,7 @@ export const makeMakeExiter = baggage => { exit() { const { state } = this; state.zcfSeat.exit(); + state.zcfSeat = undefined; }, }, { diff --git a/packages/zoe/src/contractFacet/zcfSeat.js b/packages/zoe/src/contractFacet/zcfSeat.js index 2ba5f7e4796..15cc5403d12 100644 --- a/packages/zoe/src/contractFacet/zcfSeat.js +++ b/packages/zoe/src/contractFacet/zcfSeat.js @@ -159,6 +159,7 @@ export const createSeatManager = ( assertNoStagedAllocation(self); doExitSeat(self); E(zoeInstanceAdmin).exitSeat(zcfSeatToSeatHandle.get(self), completion); + zcfSeatToSeatHandle.delete(self); }, fail( reason = Error( @@ -179,6 +180,7 @@ export const createSeatManager = ( zcfSeatToSeatHandle.get(self), harden(reason), ); + zcfSeatToSeatHandle.delete(self); } return reason; }, diff --git a/packages/zoe/src/zoeService/originalZoeSeat.js b/packages/zoe/src/zoeService/originalZoeSeat.js new file mode 100644 index 00000000000..0187d07dcd3 --- /dev/null +++ b/packages/zoe/src/zoeService/originalZoeSeat.js @@ -0,0 +1,352 @@ +/* eslint @typescript-eslint/no-floating-promises: "warn" */ +import { SubscriberShape } from '@agoric/notifier'; +import { E } from '@endo/eventual-send'; +import { M, prepareExoClassKit } from '@agoric/vat-data'; +import { deeplyFulfilled } from '@endo/marshal'; +import { makePromiseKit } from '@endo/promise-kit'; + +import { satisfiesWant } from '../contractFacet/offerSafety.js'; +import '../types.js'; +import '../internal-types.js'; +import { + AmountKeywordRecordShape, + KeywordShape, + ExitObjectShape, + PaymentPKeywordRecordShape, +} from '../typeGuards.js'; + +const { Fail } = assert; + +export const coreUserSeatMethods = harden({ + getProposal: M.call().returns(M.promise()), + getPayouts: M.call().returns(M.promise()), + getPayout: M.call(KeywordShape).returns(M.promise()), + getOfferResult: M.call().returns(M.promise()), + hasExited: M.call().returns(M.promise()), + numWantsSatisfied: M.call().returns(M.promise()), + getFinalAllocation: M.call().returns(M.promise()), + getExitSubscriber: M.call().returns(M.any()), +}); + +export const ZoeUserSeatShape = M.interface('UserSeat', { + ...coreUserSeatMethods, + tryExit: M.call().returns(M.promise()), +}); + +export const OriginalZoeSeatIKit = harden({ + zoeSeatAdmin: M.interface('ZoeSeatAdmin', { + replaceAllocation: M.call(AmountKeywordRecordShape).returns(), + exit: M.call(M.any()).returns(), + fail: M.call(M.any()).returns(), + resolveExitAndResult: M.call({ + offerResultPromise: M.promise(), + exitObj: ExitObjectShape, + }).returns(), + getExitSubscriber: M.call().returns(SubscriberShape), + // The return promise is empty, but doExit relies on settlement as a signal + // that the payouts have settled. The exit publisher is notified after that. + finalPayouts: M.call(M.eref(PaymentPKeywordRecordShape)).returns( + M.promise(), + ), + }), + userSeat: ZoeUserSeatShape, +}); + +const assertHasNotExited = (c, msg) => { + !c.state.instanceAdminHelper.hasExited(c.facets.zoeSeatAdmin) || + assert(!c.state.instanceAdminHelper.hasExited(c.facets.zoeSeatAdmin), msg); +}; + +/** + * declareOldZoeSeatAdminKind declares an exo for the original kind of ZoeSeatKit. + * This version creates a reference cycle that garbage collection can't remove + * because it goes through weakMaps in two different Vats. We've defined a new + * Kind that doesn't have this problem, but we won't upgrade the existing + * objects, so the Kind must continue to be defined, but we don't return the + * maker function. + * + * The original ZoeSeatKit is an object that manages the state + * of a seat participating in a Zoe contract and return its two facets. + * + * The UserSeat is suitable to be handed to an agent outside zoe and the + * contract and allows them to query or monitor the current state, access the + * payouts and result, and call exit() if that's allowed for this seat. + * + * The zoeSeatAdmin is passed by Zoe to the ContractFacet (zcf), to allow zcf to + * query or update the allocation or exit the seat cleanly. + * + * @param {import('@agoric/vat-data').Baggage} baggage + * @param {() => PublishKit} makeDurablePublishKit + */ +export const declareOldZoeSeatAdminKind = (baggage, makeDurablePublishKit) => { + const doExit = ( + zoeSeatAdmin, + currentAllocation, + withdrawFacet, + instanceAdminHelper, + ) => { + /** @type {PaymentPKeywordRecord} */ + const payouts = withdrawFacet.withdrawPayments(currentAllocation); + return E.when( + zoeSeatAdmin.finalPayouts(payouts), + () => instanceAdminHelper.exitZoeSeatAdmin(zoeSeatAdmin), + () => instanceAdminHelper.exitZoeSeatAdmin(zoeSeatAdmin), + ); + }; + + // There is a race between resolveExitAndResult() and getOfferResult() that + // can be limited to when the adminFactory is paged in. If state.offerResult + // is defined, getOfferResult will return it. If it's not defined when + // getOfferResult is called, create a promiseKit, return the promise and store + // the kit here. When resolveExitAndResult() is called, it saves + // state.offerResult and resolves the promise if it exists, then removes the + // table entry. + /** + * @typedef {WeakMap} + */ + const ephemeralOfferResultStore = new WeakMap(); + + // notice that this returns a maker function which we drop on the floor. + prepareExoClassKit( + baggage, + 'ZoeSeatKit', + OriginalZoeSeatIKit, + /** + * + * @param {Allocation} initialAllocation + * @param {ProposalRecord} proposal + * @param {InstanceAdminHelper} instanceAdminHelper + * @param {WithdrawFacet} withdrawFacet + * @param {ERef} [exitObj] + * @param {boolean} [offerResultIsUndefined] + */ + ( + initialAllocation, + proposal, + instanceAdminHelper, + withdrawFacet, + exitObj = undefined, + // emptySeatKits start with offerResult validly undefined; others can set + // it to anything (including undefined) in resolveExitAndResult() + offerResultIsUndefined = false, + ) => { + const { publisher, subscriber } = makeDurablePublishKit(); + return { + currentAllocation: initialAllocation, + proposal, + exitObj, + offerResult: undefined, + offerResultStored: offerResultIsUndefined, + instanceAdminHelper, + withdrawFacet, + publisher, + subscriber, + payouts: harden({}), + exiting: false, + }; + }, + { + zoeSeatAdmin: { + replaceAllocation(replacementAllocation) { + const { state } = this; + assertHasNotExited( + this, + 'Cannot replace allocation. Seat has already exited', + ); + harden(replacementAllocation); + // Merging happens in ZCF, so replacementAllocation can + // replace the old allocation entirely. + + state.currentAllocation = replacementAllocation; + }, + exit(completion) { + const { state, facets } = this; + // Since this method doesn't wait, we could re-enter via exitAllSeats. + // If that happens, we shouldn't re-do any of the work. + if (state.exiting) { + return; + } + assertHasNotExited(this, 'Cannot exit seat. Seat has already exited'); + + state.exiting = true; + E.when( + doExit( + facets.zoeSeatAdmin, + state.currentAllocation, + state.withdrawFacet, + state.instanceAdminHelper, + ), + () => state.publisher.finish(completion), + ); + }, + fail(reason) { + const { state, facets } = this; + // Since this method doesn't wait, we could re-enter via failAllSeats. + // If that happens, we shouldn't re-do any of the work. + if (state.exiting) { + return; + } + + assertHasNotExited(this, 'Cannot fail seat. Seat has already exited'); + + state.exiting = true; + E.when( + doExit( + facets.zoeSeatAdmin, + state.currentAllocation, + state.withdrawFacet, + state.instanceAdminHelper, + ), + () => state.publisher.fail(reason), + () => state.publisher.fail(reason), + ); + }, + // called only for seats resulting from offers. + /** @param {HandleOfferResult} result */ + resolveExitAndResult({ offerResultPromise, exitObj }) { + const { state, facets } = this; + + !state.offerResultStored || + Fail`offerResultStored before offerResultPromise`; + + if (!ephemeralOfferResultStore.has(facets.userSeat)) { + // this was called before getOfferResult + const kit = makePromiseKit(); + kit.resolve(offerResultPromise); + ephemeralOfferResultStore.set(facets.userSeat, kit); + } + + const pKit = ephemeralOfferResultStore.get(facets.userSeat); + E.when( + offerResultPromise, + offerResult => { + // Resolve the ephemeral promise for offerResult + pKit.resolve(offerResult); + // Now we want to store the offerResult in `state` to get it off the heap, + // but we need to handle three cases: + // 1. already durable. (This includes being a remote presence.) + // 2. has promises for durable objects. + // 3. not durable even after resolving promises. + // For #1 we can assign directly, but we deeply await to also handle #2. + void E.when( + deeplyFulfilled(offerResult), + fulfilledOfferResult => { + try { + // In cases 1-2 this assignment will succeed. + state.offerResult = fulfilledOfferResult; + // If it doesn't, then these lines won't be reached so the + // flag will stay false and the promise will stay in the heap + state.offerResultStored = true; + ephemeralOfferResultStore.delete(facets.userSeat); + } catch (err) { + console.warn( + `non-durable offer result will be lost upon zoe vat termination: ${offerResult}`, + ); + } + }, + // no rejection handler because an offer result containing promises that reject + // is within spec + ); + }, + e => { + pKit.reject(e); + // NB: leave the rejected promise in the ephemeralOfferResultStore + // because it can't go in durable state + }, + ); + + state.exitObj = exitObj; + }, + getExitSubscriber() { + const { state } = this; + return state.subscriber; + }, + async finalPayouts(payments) { + const { state } = this; + + const settledPayouts = await deeplyFulfilled(payments); + state.payouts = settledPayouts; + }, + }, + userSeat: { + async getProposal() { + const { state } = this; + return state.proposal; + }, + async getPayouts() { + const { state } = this; + + return E.when( + state.subscriber.subscribeAfter(), + () => state.payouts, + () => state.payouts, + ); + }, + async getPayout(keyword) { + const { state } = this; + + // subscriber.subscribeAfter() only triggers after publisher.finish() + // in exit() or publisher.fail() in fail(). Both of those wait for + // doExit(), which ensures that finalPayouts() has set state.payouts. + return E.when( + state.subscriber.subscribeAfter(), + () => state.payouts[keyword], + () => state.payouts[keyword], + ); + }, + + async getOfferResult() { + const { state, facets } = this; + + if (state.offerResultStored) { + return state.offerResult; + } + + if (ephemeralOfferResultStore.has(facets.userSeat)) { + return ephemeralOfferResultStore.get(facets.userSeat).promise; + } + + const kit = makePromiseKit(); + ephemeralOfferResultStore.set(facets.userSeat, kit); + return kit.promise; + }, + async hasExited() { + const { state, facets } = this; + + return ( + state.exiting || + state.instanceAdminHelper.hasExited(facets.zoeSeatAdmin) + ); + }, + async tryExit() { + const { state } = this; + if (!state.exitObj) + throw Fail`exitObj must be initialized before use`; + assertHasNotExited(this, 'Cannot exit; seat has already exited'); + + return E(state.exitObj).exit(); + }, + async numWantsSatisfied() { + const { state } = this; + return E.when( + state.subscriber.subscribeAfter(), + () => satisfiesWant(state.proposal, state.currentAllocation), + () => satisfiesWant(state.proposal, state.currentAllocation), + ); + }, + getExitSubscriber() { + const { state } = this; + return state.subscriber; + }, + getFinalAllocation() { + const { state } = this; + return E.when( + state.subscriber.subscribeAfter(), + () => state.currentAllocation, + () => state.currentAllocation, + ); + }, + }, + }, + ); +}; diff --git a/packages/zoe/src/zoeService/zoeSeat.js b/packages/zoe/src/zoeService/zoeSeat.js index f58200e164e..363932fb89b 100644 --- a/packages/zoe/src/zoeService/zoeSeat.js +++ b/packages/zoe/src/zoeService/zoeSeat.js @@ -1,5 +1,5 @@ /* eslint @typescript-eslint/no-floating-promises: "warn" */ -import { prepareDurablePublishKit, SubscriberShape } from '@agoric/notifier'; +import { prepareDurablePublishKit } from '@agoric/notifier'; import { E } from '@endo/eventual-send'; import { M, prepareExoClassKit } from '@agoric/vat-data'; import { deeplyFulfilled } from '@endo/marshal'; @@ -9,47 +9,33 @@ import { satisfiesWant } from '../contractFacet/offerSafety.js'; import '../types.js'; import '../internal-types.js'; import { - AmountKeywordRecordShape, - KeywordShape, - ExitObjectShape, - PaymentPKeywordRecordShape, -} from '../typeGuards.js'; + declareOldZoeSeatAdminKind, + OriginalZoeSeatIKit, + ZoeUserSeatShape, + coreUserSeatMethods, +} from './originalZoeSeat.js'; const { Fail } = assert; -const ZoeSeatIKit = harden({ - zoeSeatAdmin: M.interface('ZoeSeatAdmin', { - replaceAllocation: M.call(AmountKeywordRecordShape).returns(), - exit: M.call(M.any()).returns(), - fail: M.call(M.any()).returns(), - resolveExitAndResult: M.call({ - offerResultPromise: M.promise(), - exitObj: ExitObjectShape, - }).returns(), - getExitSubscriber: M.call().returns(SubscriberShape), - // The return promise is empty, but doExit relies on settlement as a signal - // that the payouts have settled. The exit publisher is notified after that. - finalPayouts: M.call(M.eref(PaymentPKeywordRecordShape)).returns( - M.promise(), - ), - }), - userSeat: M.interface('UserSeat', { - getProposal: M.call().returns(M.promise()), - getPayouts: M.call().returns(M.promise()), - getPayout: M.call(KeywordShape).returns(M.promise()), - getOfferResult: M.call().returns(M.promise()), - hasExited: M.call().returns(M.promise()), - tryExit: M.call().returns(M.promise()), - numWantsSatisfied: M.call().returns(M.promise()), - getFinalAllocation: M.call().returns(M.promise()), - getExitSubscriber: M.call().returns(M.any()), +// ZoeSeatAdmin has the implementation of coreUserSeatMethods, but ZoeUserSeat +// is the facet shared with users. The latter transparently forwards to the +// former. + +const ZoeSeatAdmin = harden({ + userSeatAccess: M.interface('UserSeatAccess', { + ...coreUserSeatMethods, + initExitObjectSetter: M.call(M.any()).returns(), + assertHasNotExited: M.call(M.string()).returns(), }), + zoeSeatAdmin: OriginalZoeSeatIKit.zoeSeatAdmin, }); -const assertHasNotExited = (c, msg) => { - !c.state.instanceAdminHelper.hasExited(c.facets.zoeSeatAdmin) || - assert(!c.state.instanceAdminHelper.hasExited(c.facets.zoeSeatAdmin), msg); -}; +const ZoeUserSeat = harden({ + userSeat: ZoeUserSeatShape, + exitObjSetter: M.interface('exitObjSetter', { + setExitObject: M.call(M.or(M.remotable(), M.undefined())).returns(), + }), +}); /** * makeZoeSeatAdminFactory returns a maker for an object that manages the state @@ -70,6 +56,8 @@ export const makeZoeSeatAdminFactory = baggage => { 'zoe Seat publisher', ); + declareOldZoeSeatAdminKind(baggage, makeDurablePublishKit); + const doExit = ( zoeSeatAdmin, currentAllocation, @@ -97,17 +85,15 @@ export const makeZoeSeatAdminFactory = baggage => { */ const ephemeralOfferResultStore = new WeakMap(); - return prepareExoClassKit( + const makeZoeSeatAdmin = prepareExoClassKit( baggage, - 'ZoeSeatKit', - ZoeSeatIKit, + 'ZoeSeatAdmin', + ZoeSeatAdmin, /** - * * @param {Allocation} initialAllocation * @param {ProposalRecord} proposal * @param {InstanceAdminHelper} instanceAdminHelper * @param {WithdrawFacet} withdrawFacet - * @param {ERef} [exitObj] * @param {boolean} [offerResultIsUndefined] */ ( @@ -115,7 +101,6 @@ export const makeZoeSeatAdminFactory = baggage => { proposal, instanceAdminHelper, withdrawFacet, - exitObj = undefined, // emptySeatKits start with offerResult validly undefined; others can set // it to anything (including undefined) in resolveExitAndResult() offerResultIsUndefined = false, @@ -124,7 +109,6 @@ export const makeZoeSeatAdminFactory = baggage => { return { currentAllocation: initialAllocation, proposal, - exitObj, offerResult: undefined, offerResultStored: offerResultIsUndefined, instanceAdminHelper, @@ -133,14 +117,97 @@ export const makeZoeSeatAdminFactory = baggage => { subscriber, payouts: harden({}), exiting: false, + /** @type {{ setExitObject: (exitObj: ExitObj | undefined) => void} | undefined} */ + exitObjectSetter: undefined, }; }, { + // methods for userSeat to call + userSeatAccess: { + async getProposal() { + const { state } = this; + return state.proposal; + }, + async getPayouts() { + const { state } = this; + + return E.when( + state.subscriber.subscribeAfter(), + () => state.payouts, + () => state.payouts, + ); + }, + async getPayout(keyword) { + const { state } = this; + + // subscriber.subscribeAfter() only triggers after publisher.finish() + // in exit() or publisher.fail() in fail(). Both of those wait for + // doExit(), which ensures that finalPayouts() has set state.payouts. + return E.when( + state.subscriber.subscribeAfter(), + () => state.payouts[keyword], + () => state.payouts[keyword], + ); + }, + + async getOfferResult() { + const { state, facets } = this; + + if (state.offerResultStored) { + return state.offerResult; + } + + if (ephemeralOfferResultStore.has(facets.zoeSeatAdmin)) { + return ephemeralOfferResultStore.get(facets.zoeSeatAdmin).promise; + } + + const kit = makePromiseKit(); + ephemeralOfferResultStore.set(facets.zoeSeatAdmin, kit); + return kit.promise; + }, + async hasExited() { + const { state, facets } = this; + + return ( + state.exiting || + state.instanceAdminHelper.hasExited(facets.zoeSeatAdmin) + ); + }, + async numWantsSatisfied() { + const { state } = this; + return E.when( + state.subscriber.subscribeAfter(), + () => satisfiesWant(state.proposal, state.currentAllocation), + () => satisfiesWant(state.proposal, state.currentAllocation), + ); + }, + getExitSubscriber() { + const { state } = this; + return state.subscriber; + }, + getFinalAllocation() { + const { state } = this; + return E.when( + state.subscriber.subscribeAfter(), + () => state.currentAllocation, + () => state.currentAllocation, + ); + }, + initExitObjectSetter(setter) { + this.state.exitObjectSetter = setter; + }, + assertHasNotExited(msg) { + const { state, facets } = this; + const { instanceAdminHelper } = state; + const hasExited1 = instanceAdminHelper.hasExited(facets.zoeSeatAdmin); + + !hasExited1 || assert(!hasExited1, msg); + }, + }, zoeSeatAdmin: { replaceAllocation(replacementAllocation) { - const { state } = this; - assertHasNotExited( - this, + const { state, facets } = this; + facets.userSeatAccess.assertHasNotExited( 'Cannot replace allocation. Seat has already exited', ); harden(replacementAllocation); @@ -156,7 +223,9 @@ export const makeZoeSeatAdminFactory = baggage => { if (state.exiting) { return; } - assertHasNotExited(this, 'Cannot exit seat. Seat has already exited'); + facets.userSeatAccess.assertHasNotExited( + 'Cannot exit seat. Seat has already exited', + ); state.exiting = true; E.when( @@ -166,7 +235,13 @@ export const makeZoeSeatAdminFactory = baggage => { state.withdrawFacet, state.instanceAdminHelper, ), - () => state.publisher.finish(completion), + () => { + if (state.exitObjectSetter) { + state.exitObjectSetter.setExitObject(undefined); + state.exitObjectSetter = undefined; + } + return state.publisher.finish(completion); + }, ); }, fail(reason) { @@ -177,10 +252,12 @@ export const makeZoeSeatAdminFactory = baggage => { return; } - assertHasNotExited(this, 'Cannot fail seat. Seat has already exited'); + facets.userSeatAccess.assertHasNotExited( + 'Cannot fail seat. Seat has already exited', + ); state.exiting = true; - E.when( + void E.when( doExit( facets.zoeSeatAdmin, state.currentAllocation, @@ -190,6 +267,11 @@ export const makeZoeSeatAdminFactory = baggage => { () => state.publisher.fail(reason), () => state.publisher.fail(reason), ); + + if (state.exitObjectSetter) { + state.exitObjectSetter.setExitObject(undefined); + state.exitObjectSetter = undefined; + } }, // called only for seats resulting from offers. /** @param {HandleOfferResult} result */ @@ -199,15 +281,15 @@ export const makeZoeSeatAdminFactory = baggage => { !state.offerResultStored || Fail`offerResultStored before offerResultPromise`; - if (!ephemeralOfferResultStore.has(facets.userSeat)) { + if (!ephemeralOfferResultStore.has(facets.zoeSeatAdmin)) { // this was called before getOfferResult const kit = makePromiseKit(); kit.resolve(offerResultPromise); - ephemeralOfferResultStore.set(facets.userSeat, kit); + ephemeralOfferResultStore.set(facets.zoeSeatAdmin, kit); } - const pKit = ephemeralOfferResultStore.get(facets.userSeat); - E.when( + const pKit = ephemeralOfferResultStore.get(facets.zoeSeatAdmin); + void E.when( offerResultPromise, offerResult => { // Resolve the ephemeral promise for offerResult @@ -227,7 +309,7 @@ export const makeZoeSeatAdminFactory = baggage => { // If it doesn't, then these lines won't be reached so the // flag will stay false and the promise will stay in the heap state.offerResultStored = true; - ephemeralOfferResultStore.delete(facets.userSeat); + ephemeralOfferResultStore.delete(facets.zoeSeatAdmin); } catch (err) { console.warn( `non-durable offer result will be lost upon zoe vat termination: ${offerResult}`, @@ -245,7 +327,8 @@ export const makeZoeSeatAdminFactory = baggage => { }, ); - state.exitObj = exitObj; + // @ts-expect-error exitObjectSetter is set at birth. + state.exitObjectSetter.setExitObject(exitObj); }, getExitSubscriber() { const { state } = this; @@ -258,85 +341,100 @@ export const makeZoeSeatAdminFactory = baggage => { state.payouts = settledPayouts; }, }, + }, + ); + + const makeUserSeat = prepareExoClassKit( + baggage, + 'ZoeUserSeat', + ZoeUserSeat, + (userSeatAccess, exitObj) => { + return { + userSeatAccess, + exitObj, + }; + }, + { userSeat: { async getProposal() { - const { state } = this; - return state.proposal; + return this.state.userSeatAccess.getProposal(); }, async getPayouts() { - const { state } = this; - - return E.when( - state.subscriber.subscribeAfter(), - () => state.payouts, - () => state.payouts, - ); + return this.state.userSeatAccess.getPayouts(); }, async getPayout(keyword) { - const { state } = this; - - // subscriber.subscribeAfter() only triggers after publisher.finish() - // in exit() or publisher.fail() in fail(). Both of those wait for - // doExit(), which ensures that finalPayouts() has set state.payouts. - return E.when( - state.subscriber.subscribeAfter(), - () => state.payouts[keyword], - () => state.payouts[keyword], - ); + return this.state.userSeatAccess.getPayout(keyword); }, async getOfferResult() { - const { state, facets } = this; - - if (state.offerResultStored) { - return state.offerResult; - } - - if (ephemeralOfferResultStore.has(facets.userSeat)) { - return ephemeralOfferResultStore.get(facets.userSeat).promise; - } - - const kit = makePromiseKit(); - ephemeralOfferResultStore.set(facets.userSeat, kit); - return kit.promise; + return this.state.userSeatAccess.getOfferResult(); }, async hasExited() { - const { state, facets } = this; - - return ( - state.exiting || - state.instanceAdminHelper.hasExited(facets.zoeSeatAdmin) - ); + return this.state.userSeatAccess.hasExited(); }, async tryExit() { const { state } = this; + + state.userSeatAccess.assertHasNotExited( + 'Cannot exit; seat has already exited', + ); if (!state.exitObj) - throw Fail`exitObj must be initialized before use`; - assertHasNotExited(this, 'Cannot exit; seat has already exited'); + throw Fail`exitObj not initialized or already nullified`; + + const exitResult = E(state.exitObj).exit(); - return E(state.exitObj).exit(); + // unlink an un-collectible cycle. + state.exitObj = undefined; + + return exitResult; }, async numWantsSatisfied() { - const { state } = this; - return E.when( - state.subscriber.subscribeAfter(), - () => satisfiesWant(state.proposal, state.currentAllocation), - () => satisfiesWant(state.proposal, state.currentAllocation), - ); + return this.state.userSeatAccess.numWantsSatisfied(); }, getExitSubscriber() { - const { state } = this; - return state.subscriber; + return this.state.userSeatAccess.getExitSubscriber(); }, getFinalAllocation() { - const { state } = this; - return E.when( - state.subscriber.subscribeAfter(), - () => state.currentAllocation, - () => state.currentAllocation, - ); + return this.state.userSeatAccess.getFinalAllocation(); + }, + }, + exitObjSetter: { + setExitObject(exitObject) { + this.state.exitObj = exitObject; }, }, }, ); + + /** + * @param {Allocation} initialAllocation + * @param {ProposalRecord} proposal + * @param {InstanceAdminHelper} instanceAdminHelper + * @param {WithdrawFacet} withdrawFacet + * @param {ERef} [exitObj] + * @param {boolean} [offerResultIsUndefined] + */ + const makeZoeSeatAdminKit = ( + initialAllocation, + proposal, + instanceAdminHelper, + withdrawFacet, + exitObj = undefined, + offerResultIsUndefined = false, + ) => { + const { zoeSeatAdmin, userSeatAccess } = makeZoeSeatAdmin( + initialAllocation, + proposal, + instanceAdminHelper, + withdrawFacet, + offerResultIsUndefined, + ); + const { userSeat, exitObjSetter } = makeUserSeat(userSeatAccess, exitObj); + userSeatAccess.initExitObjectSetter(exitObjSetter); + + // The original makeZoeSeatAdminKit returned two facets of the same kind. + // This is returning two independent facets. + return { userSeat, zoeSeatAdmin }; + }; + return makeZoeSeatAdminKit; }; diff --git a/packages/zoe/test/unitTests/zcf/test-zcf.js b/packages/zoe/test/unitTests/zcf/test-zcf.js index a8e68a05d4f..9859a0500f7 100644 --- a/packages/zoe/test/unitTests/zcf/test-zcf.js +++ b/packages/zoe/test/unitTests/zcf/test-zcf.js @@ -992,7 +992,7 @@ test(`userSeat.getPayout() should throw from zcf.makeEmptySeatKit`, async t => { // @ts-expect-error deliberate invalid arguments for testing await t.throwsAsync(() => E(userSeat).getPayout(), { message: - 'In "getPayout" method of (ZoeSeatKit userSeat): Expected at least 1 arguments: []', + 'In "getPayout" method of (ZoeUserSeat userSeat): Expected at least 1 arguments: []', }); }); diff --git a/scripts/generate-a3p-submission-dirs.sh b/scripts/generate-a3p-submission-dirs.sh new file mode 100755 index 00000000000..6ed9d6bf3df --- /dev/null +++ b/scripts/generate-a3p-submission-dirs.sh @@ -0,0 +1,9 @@ +#!/bin/bash +set -ueo pipefail + +for proposal in ./proposals/?:* +do + cd $proposal + yarn run build:submission + cd - +done