Skip to content

Commit

Permalink
test(multichain-testing): stir for no AVA timeout in long awaits
Browse files Browse the repository at this point in the history
  • Loading branch information
michaelfig committed Oct 7, 2024
1 parent 7940ef4 commit 6b38ae8
Show file tree
Hide file tree
Showing 7 changed files with 118 additions and 12 deletions.
3 changes: 1 addition & 2 deletions multichain-testing/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -58,8 +58,7 @@
"**/*.test.ts"
],
"concurrency": 1,
"serial": true,
"timeout": "125s"
"serial": true
},
"eslintConfig": {
"root": true,
Expand Down
5 changes: 4 additions & 1 deletion multichain-testing/test/stake-ica.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -243,7 +243,10 @@ const stakeScenario = test.macro(async (t, scenario: StakeIcaScenario) => {

console.log('waiting for unbonding period');
// XXX reference `120000` from chain state + `maxClockSkew`
await sleep(120000);
await sleep(120000, {
stir: description =>
t.pass(`stirring while waiting for unbonding period: ${description}`),
});
const { balances: rewardsWithUndelegations } = await retryUntilCondition(
() => queryClient.queryBalances(address),
({ balances }) => {
Expand Down
26 changes: 26 additions & 0 deletions multichain-testing/test/stir.test.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
import test from '@endo/ses-ava/prepare-endo.js';
import { sleep } from '../tools/sleep.js';

test('sleep without tripping AVA timeout', async t => {
const stirred: string[] = [];
t.timeout(500);
const sleepTime = 4_000;
await sleep(sleepTime, {
log: t.log,
stirEveryMs: 300,
stir: description => {
stirred.push(description);
t.pass(description);
},
});
t.like(stirred, {
length: 15,
0: `sleeping for ${sleepTime}ms...`,
1: `while sleeping (1/14)`,
2: `while sleeping (2/14)`,
// ...
13: `while sleeping (13/14)`,
14: `while sleeping (14/14)`,
15: undefined,
});
});
14 changes: 12 additions & 2 deletions multichain-testing/test/support.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import { makeAgdTools } from '../tools/agd-tools.js';
import { type E2ETools } from '../tools/e2e-tools.js';
import { makeGetFile, makeSetupRegistry } from '../tools/registry.js';
import { generateMnemonic } from '../tools/wallet.js';
import { makeRetryUntilCondition } from '../tools/sleep.js';
import { makeRetryUntilCondition, sleep } from '../tools/sleep.js';
import { makeDeployBuilder } from '../tools/deploy.js';
import { makeHermes } from '../tools/hermes-tools.js';

Expand Down Expand Up @@ -55,12 +55,22 @@ const makeKeyring = async (

export const commonSetup = async (t: ExecutionContext) => {
const { useChain } = await setupRegistry();
const tools = await makeAgdTools(t.log, childProcess);
const delay = (ms: number) =>
sleep(ms, {
log: t.log,
stir: description => t.pass(`stirring in commonSetup ${description}`),
setTimeout: globalThis.setTimeout,
});
const tools = await makeAgdTools(t.log, { ...childProcess, delay });
const keyring = await makeKeyring(tools);
const deployBuilder = makeDeployBuilder(tools, fse.readJSON, execa);
const retryUntilCondition = makeRetryUntilCondition({
log: t.log,
setTimeout: globalThis.setTimeout,
setInterval: globalThis.setInterval,
clearInterval: globalThis.clearInterval,
stir: description =>
t.pass(`stirring in retryUntilCondition ${description}`),
});
const hermes = makeHermes(childProcess);

Expand Down
6 changes: 5 additions & 1 deletion multichain-testing/tools/agd-tools.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,14 +6,18 @@ export const makeAgdTools = async (
{
execFile,
execFileSync,
}: Pick<typeof import('child_process'), 'execFile' | 'execFileSync'>,
delay,
}: Pick<typeof import('child_process'), 'execFile' | 'execFileSync'> & {
delay?: (ms: number) => Promise<void>;
},
) => {
const bundleCache = await unsafeMakeBundleCache('bundles');
const tools = await makeE2ETools(log, bundleCache, {
execFileSync,
execFile,
fetch,
setTimeout,
delay,
});
return tools;
};
Expand Down
5 changes: 3 additions & 2 deletions multichain-testing/tools/e2e-tools.js
Original file line number Diff line number Diff line change
Expand Up @@ -412,7 +412,8 @@ const runCoreEval = async (
* @param {typeof import('child_process').execFileSync} io.execFileSync
* @param {typeof import('child_process').execFile} io.execFile
* @param {typeof window.fetch} io.fetch
* @param {typeof window.setTimeout} io.setTimeout
* @param {typeof window.setTimeout} [io.setTimeout]
* @param {(ms: number) => Promise<void>} [io.delay]
* @param {string} [io.bundleDir]
* @param {string} [io.rpcAddress]
* @param {string} [io.apiAddress]
Expand All @@ -427,12 +428,12 @@ export const makeE2ETools = async (
setTimeout,
rpcAddress = 'http://localhost:26657',
apiAddress = 'http://localhost:1317',
delay = ms => new Promise(resolve => setTimeout(resolve, ms)),
},
) => {
const agd = makeAgd({ execFileSync }).withOpts({ keyringBackend: 'test' });
const rpc = makeHttpClient(rpcAddress, fetch);
const lcd = makeAPI(apiAddress, { fetch });
const delay = ms => new Promise(resolve => setTimeout(resolve, ms));

const explainDelay = (ms, info) => {
if (typeof info === 'object' && Object.keys(info).length > 0) {
Expand Down
71 changes: 67 additions & 4 deletions multichain-testing/tools/sleep.ts
Original file line number Diff line number Diff line change
@@ -1,21 +1,68 @@
const ambientSetInterval = globalThis.setInterval;
const ambientClearInterval = globalThis.clearInterval;
const ambientSetTimeout = globalThis.setTimeout;

// Derived from the fact that AVA's "default timeout is 10 seconds."
// https://github.com/avajs/ava/blob/main/docs/07-test-timeouts.md#test-timeouts
const DEFAULT_STIR_EVERY_MS = 8_000;

type Log = (...values: unknown[]) => void;

type SleepOptions = {
log?: Log;
clearInterval?: typeof ambientClearInterval;
setInterval?: typeof ambientSetInterval;
setTimeout?: typeof ambientSetTimeout;
stirEveryMs?: number;
/**
* Call every `stirEveryMs` during sleeping so that the sleeper isn't timed
* out by, e.g. a test runner.
* @param description - A bit of context as to why we're stirring.
*/
stir?: (description: string) => void;
};

export const sleep = (
/**
*
* @param {number} ms Sleep duration in milliseconds
* @param {SleepOptions} opts
* @returns {Promise<void>}
*/
const deepSleep = (
ms: number,
{ log = () => {}, setTimeout = ambientSetTimeout }: SleepOptions = {},
{ setTimeout = ambientSetTimeout }: SleepOptions = {},
) =>
new Promise(resolve => {
log(`Sleeping for ${ms}ms...`);
setTimeout(resolve, ms);
});

export const sleep = async (
ms: number,
{
log = () => {},
stir,
stirEveryMs = DEFAULT_STIR_EVERY_MS,
setTimeout = ambientSetTimeout,
}: SleepOptions = {},
) => {
log(`Sleeping for ${ms}ms...`);
await (stir && stir(`sleeping for ${ms}ms...`));
let remaining = ms;
let nStirs = 0;
const totalStirs = Math.ceil(ms / stirEveryMs);
const doStir = async () => {
nStirs += 1;
await (stir && stir(`while sleeping (${nStirs}/${totalStirs})`));
};
while (stir && remaining >= stirEveryMs) {
remaining -= stirEveryMs;
await deepSleep(stirEveryMs, { setTimeout });
await doStir();
}
await deepSleep(remaining, { setTimeout });
await doStir();
};

export type RetryOptions = {
maxRetries?: number;
retryIntervalMs?: number;
Expand All @@ -29,13 +76,27 @@ const retryUntilCondition = async <T>(
maxRetries = 6,
retryIntervalMs = 3500,
log = () => {},
stirEveryMs,
stir,
setTimeout = ambientSetTimeout,
setInterval = ambientSetInterval,
clearInterval = ambientClearInterval,
}: RetryOptions = {},
): Promise<T> => {
console.log({ maxRetries, retryIntervalMs, message });
let retries = 0;

while (retries < maxRetries) {
let conditionStirring;
if (stir) {
let nStirs = 0;
conditionStirring = setInterval(() => {
nStirs += 1;
stir(
`during ${message} condition (${retries + 1}/${maxRetries})#${nStirs}`,
);
}, stirEveryMs);
}
try {
const result = await operation();
if (condition(result)) {
Expand All @@ -47,13 +108,15 @@ const retryUntilCondition = async <T>(
} else {
log(`Unknown error: ${String(error)}`);
}
} finally {
conditionStirring === undefined || clearInterval(conditionStirring);
}

retries++;
console.log(
`Retry ${retries}/${maxRetries} - Waiting for ${retryIntervalMs}ms for ${message}...`,
);
await sleep(retryIntervalMs, { log, setTimeout });
await sleep(retryIntervalMs, { log, setTimeout, stirEveryMs, stir });
}

throw Error(`${message} condition failed after ${maxRetries} retries.`);
Expand Down

0 comments on commit 6b38ae8

Please sign in to comment.