Skip to content

Commit

Permalink
Merge pull request #1870 from balena-io/expectNewSettledTasks
Browse files Browse the repository at this point in the history
tests: Fix flakyness of service install task expectations
  • Loading branch information
flowzone-app[bot] authored Nov 28, 2024
2 parents 0c738a4 + 9cbe2f0 commit 8b2c1cc
Show file tree
Hide file tree
Showing 3 changed files with 55 additions and 21 deletions.
35 changes: 22 additions & 13 deletions test/25_service-installs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,10 @@ import * as fixtures from './test-lib/fixtures.js';
import { assertExists, expectToEventually } from './test-lib/common.js';
import * as config from '../src/lib/config.js';
import { supertest } from './test-lib/supertest.js';
import { expectNewTasks, resetLatestTaskIds } from './test-lib/api-helpers.js';
import {
expectNewSettledTasks,
resetLatestTaskIds,
} from './test-lib/api-helpers.js';

export default () => {
versions.test((version, pineTest) => {
Expand Down Expand Up @@ -77,7 +80,7 @@ export default () => {
ctx.app1Service1.id,
);
});
await expectNewTasks(
await expectNewSettledTasks(
'create_service_installs',
isServiceInstallEnabled
? [
Expand Down Expand Up @@ -120,7 +123,7 @@ export default () => {
ctx.app1Service2.id,
);
});
await expectNewTasks(
await expectNewSettledTasks(
'create_service_installs',
isServiceInstallEnabled
? [
Expand Down Expand Up @@ -162,7 +165,7 @@ export default () => {
ctx.app1Service3.id,
);
});
await expectNewTasks(
await expectNewSettledTasks(
'create_service_installs',
isServiceInstallEnabled
? [
Expand All @@ -188,6 +191,20 @@ export default () => {
})
.expect(200);

await expectNewSettledTasks(
'create_service_installs',
isServiceInstallEnabled
? [
{
is_executed_with__parameter_set: {
devices: [ctx.device.id],
},
status: 'succeeded',
},
]
: [],
);

await pineUser
.patch({
resource: 'device',
Expand All @@ -212,18 +229,10 @@ export default () => {
ctx.app2Service1.id,
);
});
await expectNewTasks(
await expectNewSettledTasks(
'create_service_installs',
isServiceInstallEnabled
? [
// the first one is from unpinning
{
is_executed_with__parameter_set: {
devices: [ctx.device.id],
},
status: 'succeeded',
},
// the second one is from moving application
{
is_executed_with__parameter_set: {
devices: [ctx.device.id],
Expand Down
23 changes: 19 additions & 4 deletions test/test-lib/api-helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -269,11 +269,11 @@ const latestTaskIdByHandler: Dictionary<string | undefined> = {};

export const expectNewTasks = async (
handler: string,
expectedUsage: TaskExpectation[],
expectedTasks: TaskExpectation[],
) => {
const tasks = await expectTasks(
handler,
expectedUsage,
expectedTasks,
latestTaskIdByHandler[handler],
);
const lastTaskId = tasks.at(-1)?.id;
Expand All @@ -282,7 +282,15 @@ export const expectNewTasks = async (
}
};

export const resetLatestTaskIds = async (handler: string) => {
export const expectNewSettledTasks = async (
handler: string,
expectedTasks: TaskExpectation[],
) => {
await waitUntilTasksFinish(handler);
await expectNewTasks(handler, expectedTasks);
};

const waitUntilTasksFinish = async function (handler: string) {
await expectToEventually(async () => {
const runningTasks = await sbvrUtils.api.tasks.get({
resource: 'task',
Expand All @@ -297,8 +305,15 @@ export const resetLatestTaskIds = async (handler: string) => {
$orderby: { id: 'desc' },
},
});
expect(runningTasks).to.have.lengthOf(0);
expect(
runningTasks,
`Found still queued '${handler}' tasks`,
).to.have.lengthOf(0);
});
};

export const resetLatestTaskIds = async (handler: string) => {
await waitUntilTasksFinish(handler);
const [latestTask] = await sbvrUtils.api.tasks.get({
resource: 'task',
passthrough: { req: permissions.rootRead },
Expand Down
18 changes: 14 additions & 4 deletions test/test-lib/common.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import { expect } from 'chai';
import { stripIndent } from 'common-tags';
import { setTimeout } from 'timers/promises';
import { TypedError } from 'typed-error';
import { ThisShouldNeverHappenError } from '../../src/infra/error-handling/index.js';

type PredicateFunction = () => Resolvable<boolean>;

Expand Down Expand Up @@ -107,15 +108,24 @@ export async function expectToEventually<T>(
attempts = 20,
interval = 100,
): Promise<T> {
let error: Error | undefined;
for (let i = 0; i < attempts; i++) {
try {
return await fn();

// eslint-disable-next-line @typescript-eslint/no-unused-vars -- So that the interface is already well defined.
} catch (_e) {
} catch (e) {
if (!(e instanceof Error)) {
throw ThisShouldNeverHappenError(
'Thrown error is not an instanceof Error',
);
}
error = e;
console.log(`⌚ Waiting ${interval}ms (${i}/${attempts})...`);
await setTimeout(interval);
}
}
throw new Error('Expectation failed');
if (error == null) {
// error should always be assigned
throw ThisShouldNeverHappenError('Unexpected test error');
}
throw new Error(`Expectation failed: ${error.message}`, { cause: error });
}

0 comments on commit 8b2c1cc

Please sign in to comment.