Skip to content

Commit

Permalink
fix(endo): Synchronize host cancellation with formula graph
Browse files Browse the repository at this point in the history
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.
  • Loading branch information
rekmarks committed Feb 22, 2024
1 parent 93ba613 commit 8b10d5f
Show file tree
Hide file tree
Showing 5 changed files with 25 additions and 16 deletions.
10 changes: 9 additions & 1 deletion packages/daemon/src/daemon.js
Original file line number Diff line number Diff line change
Expand Up @@ -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<>} */ (
Expand Down Expand Up @@ -1085,8 +1093,8 @@ const makeDaemonCore = async (
const makeMailbox = makeMailboxMaker({
getFormulaIdentifierForRef,
provideValueForFormulaIdentifier,
provideControllerForFormulaIdentifier,
provideControllerForFormulaIdentifierAndResolveHandle,
cancelValue,
});

const makeIdentifiedGuestController = makeGuestMaker({
Expand Down
10 changes: 3 additions & 7 deletions packages/daemon/src/mail.js
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down Expand Up @@ -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']} */
Expand Down
11 changes: 9 additions & 2 deletions packages/daemon/src/mutex.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,14 @@
import { makeQueue } from '@endo/stream';

/**
* @returns {{ lock: () => Promise<void>, unlock: () => void, enqueue: (asyncFn: () => Promise<any>) => Promise<any> }}
* @typedef {object} Mutex
* @property {() => Promise<void>} lock
* @property {() => void} unlock
* @property {(asyncFn?: () => Promise<any>) => Promise<any>} enqueue
*/

/**
* @returns {Mutex}
*/
export const makeMutex = () => {
/** @type {import('@endo/stream').AsyncQueue<void>} */
Expand All @@ -21,7 +28,7 @@ export const makeMutex = () => {
enqueue: async asyncFn => {
await lock();
try {
return await asyncFn();
return await (asyncFn ? asyncFn() : null);
} finally {
unlock();
}
Expand Down
4 changes: 4 additions & 0 deletions packages/daemon/src/types.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -291,6 +291,10 @@ export type ProvideControllerForFormulaIdentifier = (
export type ProvideControllerForFormulaIdentifierAndResolveHandle = (
formulaIdentifier: string,
) => Promise<Controller>;
export type CancelValue = (
formulaIdentifier: string,
reason: Error,
) => Promise<void>;

/**
* A handle is used to create a pointer to a formula without exposing it directly.
Expand Down
6 changes: 0 additions & 6 deletions packages/daemon/test/test-endo.js
Original file line number Diff line number Diff line change
Expand Up @@ -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');
});
Expand Down Expand Up @@ -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');
});
Expand Down

0 comments on commit 8b10d5f

Please sign in to comment.