From 8b10d5fbb9429ae983ca57756f802948d69339e4 Mon Sep 17 00:00:00 2001 From: Erik Marks Date: Tue, 20 Feb 2024 21:36:54 -0800 Subject: [PATCH] fix(endo): Synchronize host cancellation with formula graph Synchronizes the host's `cancel()` with the formula graph by awaiting the formula graph mutex in a new daemon method, `cancelValue()`. Modifies the mutex implementation to permit calling `enqueue()` without specifying function to call. --- packages/daemon/src/daemon.js | 10 +++++++++- packages/daemon/src/mail.js | 10 +++------- packages/daemon/src/mutex.js | 11 +++++++++-- packages/daemon/src/types.d.ts | 4 ++++ packages/daemon/test/test-endo.js | 6 ------ 5 files changed, 25 insertions(+), 16 deletions(-) diff --git a/packages/daemon/src/daemon.js b/packages/daemon/src/daemon.js index d8972a1ef6..9adc13dd09 100644 --- a/packages/daemon/src/daemon.js +++ b/packages/daemon/src/daemon.js @@ -653,6 +653,14 @@ const makeDaemonCore = async ( return controller; }; + /** @type {import('./types.js').CancelValue} */ + const cancelValue = async (formulaIdentifier, reason) => { + await formulaGraphMutex.enqueue(); + const controller = provideControllerForFormulaIdentifier(formulaIdentifier); + console.log('Cancelled:'); + return controller.context.cancel(reason); + }; + /** @type {import('./types.js').ProvideValueForFormulaIdentifier} */ const provideValueForFormulaIdentifier = formulaIdentifier => { const controller = /** @type {import('./types.js').Controller<>} */ ( @@ -1085,8 +1093,8 @@ const makeDaemonCore = async ( const makeMailbox = makeMailboxMaker({ getFormulaIdentifierForRef, provideValueForFormulaIdentifier, - provideControllerForFormulaIdentifier, provideControllerForFormulaIdentifierAndResolveHandle, + cancelValue, }); const makeIdentifiedGuestController = makeGuestMaker({ diff --git a/packages/daemon/src/mail.js b/packages/daemon/src/mail.js index a98f8657cd..914969c984 100644 --- a/packages/daemon/src/mail.js +++ b/packages/daemon/src/mail.js @@ -11,15 +11,15 @@ const { quote: q } = assert; /** * @param {object} args * @param {import('./types.js').ProvideValueForFormulaIdentifier} args.provideValueForFormulaIdentifier - * @param {import('./types.js').ProvideControllerForFormulaIdentifier} args.provideControllerForFormulaIdentifier * @param {import('./types.js').GetFormulaIdentifierForRef} args.getFormulaIdentifierForRef * @param {import('./types.js').ProvideControllerForFormulaIdentifierAndResolveHandle} args.provideControllerForFormulaIdentifierAndResolveHandle + * @param {import('./types.js').CancelValue} args.cancelValue */ export const makeMailboxMaker = ({ getFormulaIdentifierForRef, provideValueForFormulaIdentifier, - provideControllerForFormulaIdentifier, provideControllerForFormulaIdentifierAndResolveHandle, + cancelValue, }) => { /** * @param {object} args @@ -81,11 +81,7 @@ export const makeMailboxMaker = ({ if (formulaIdentifier === undefined) { throw new TypeError(`Unknown pet name: ${q(petName)}`); } - // Behold, recursion: - const controller = - provideControllerForFormulaIdentifier(formulaIdentifier); - console.log('Cancelled:'); - return controller.context.cancel(reason); + return cancelValue(formulaIdentifier, reason); }; /** @type {import('./types.js').Mail['list']} */ diff --git a/packages/daemon/src/mutex.js b/packages/daemon/src/mutex.js index f222fc54f5..f531f20082 100644 --- a/packages/daemon/src/mutex.js +++ b/packages/daemon/src/mutex.js @@ -1,7 +1,14 @@ import { makeQueue } from '@endo/stream'; /** - * @returns {{ lock: () => Promise, unlock: () => void, enqueue: (asyncFn: () => Promise) => Promise }} + * @typedef {object} Mutex + * @property {() => Promise} lock + * @property {() => void} unlock + * @property {(asyncFn?: () => Promise) => Promise} enqueue + */ + +/** + * @returns {Mutex} */ export const makeMutex = () => { /** @type {import('@endo/stream').AsyncQueue} */ @@ -21,7 +28,7 @@ export const makeMutex = () => { enqueue: async asyncFn => { await lock(); try { - return await asyncFn(); + return await (asyncFn ? asyncFn() : null); } finally { unlock(); } diff --git a/packages/daemon/src/types.d.ts b/packages/daemon/src/types.d.ts index 0da63437ab..7d1d07f84d 100644 --- a/packages/daemon/src/types.d.ts +++ b/packages/daemon/src/types.d.ts @@ -291,6 +291,10 @@ export type ProvideControllerForFormulaIdentifier = ( export type ProvideControllerForFormulaIdentifierAndResolveHandle = ( formulaIdentifier: string, ) => Promise; +export type CancelValue = ( + formulaIdentifier: string, + reason: Error, +) => Promise; /** * A handle is used to create a pointer to a formula without exposing it directly. diff --git a/packages/daemon/test/test-endo.js b/packages/daemon/test/test-endo.js index 541bd56816..d5b27e9a82 100644 --- a/packages/daemon/test/test-endo.js +++ b/packages/daemon/test/test-endo.js @@ -846,9 +846,6 @@ test('unconfined service can respond to cancellation', async t => { ['caplet'], ['context-consumer'], ); - // TODO:cancel This should not be necessary. - // eslint-disable-next-line no-undef - await new Promise(resolve => setTimeout(resolve, 100)); await E(host).cancel('context-consumer'); t.is(await result, 'cancelled'); }); @@ -882,9 +879,6 @@ test('confined service can respond to cancellation', async t => { ['caplet'], ['context-consumer'], ); - // TODO:cancel This should not be necessary. - // eslint-disable-next-line no-undef - await new Promise(resolve => setTimeout(resolve, 100)); await E(host).cancel('context-consumer'); t.is(await result, 'cancelled'); });