test: fixup flaky tests (#44380)

* test: fixup flaky test

Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org>

* test: disable flaky protocol speed test on macOS

Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org>

* test: fixup flaky test in api-browser-window-spec.ts

Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org>

* test: update waitUntil to handle async functions

---------

Co-authored-by: trop[bot] <37223003+trop[bot]@users.noreply.github.com>
Co-authored-by: John Kleinschmidt <jkleinsc@electronjs.org>
This commit is contained in:
trop[bot] 2024-10-24 15:21:55 +02:00 committed by GitHub
parent 1a9b93ca96
commit e9988c2fc4
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 45 additions and 38 deletions

View file

@ -15,7 +15,7 @@ import { setTimeout } from 'node:timers/promises';
import { emittedUntil, emittedNTimes } from './lib/events-helpers'; import { emittedUntil, emittedNTimes } from './lib/events-helpers';
import { HexColors, hasCapturableScreen, ScreenCapture } from './lib/screen-helpers'; import { HexColors, hasCapturableScreen, ScreenCapture } from './lib/screen-helpers';
import { ifit, ifdescribe, defer, listen } from './lib/spec-helpers'; import { ifit, ifdescribe, defer, listen, waitUntil } from './lib/spec-helpers';
import { closeWindow, closeAllWindows } from './lib/window-helpers'; import { closeWindow, closeAllWindows } from './lib/window-helpers';
const fixtures = path.resolve(__dirname, 'fixtures'); const fixtures = path.resolve(__dirname, 'fixtures');
@ -5978,8 +5978,10 @@ describe('BrowserWindow module', () => {
w.webContents.on('enter-html-full-screen', async () => { w.webContents.on('enter-html-full-screen', async () => {
enterCount++; enterCount++;
if (w.isFullScreen()) reject(new Error('w.isFullScreen should be false')); if (w.isFullScreen()) reject(new Error('w.isFullScreen should be false'));
const isFS = await w.webContents.executeJavaScript('!!document.fullscreenElement'); await waitUntil(async () => {
if (!isFS) reject(new Error('Document should have fullscreen element')); const isFS = await w.webContents.executeJavaScript('!!document.fullscreenElement');
return isFS === true;
});
checkDone(); checkDone();
}); });

View file

@ -1737,7 +1737,8 @@ describe('protocol module', () => {
}); });
// TODO(nornagon): this test doesn't pass on Linux currently, investigate. // TODO(nornagon): this test doesn't pass on Linux currently, investigate.
ifit(process.platform !== 'linux')('is fast', async () => { // test is also flaky on CI on macOS so it is currently disabled there as well.
ifit(process.platform !== 'linux' && (!process.env.CI || process.platform !== 'darwin'))('is fast', async () => {
// 128 MB of spaces. // 128 MB of spaces.
const chunk = new Uint8Array(128 * 1024 * 1024); const chunk = new Uint8Array(128 * 1024 * 1024);
chunk.fill(' '.charCodeAt(0)); chunk.fill(' '.charCodeAt(0));

View file

@ -15,7 +15,7 @@ import * as path from 'node:path';
import { setTimeout } from 'node:timers/promises'; import { setTimeout } from 'node:timers/promises';
import * as url from 'node:url'; import * as url from 'node:url';
import { ifit, ifdescribe, defer, itremote, listen, startRemoteControlApp } from './lib/spec-helpers'; import { ifit, ifdescribe, defer, itremote, listen, startRemoteControlApp, waitUntil } from './lib/spec-helpers';
import { closeAllWindows } from './lib/window-helpers'; import { closeAllWindows } from './lib/window-helpers';
import { PipeTransport } from './pipe-transport'; import { PipeTransport } from './pipe-transport';
@ -2946,10 +2946,12 @@ describe('iframe using HTML fullscreen API while window is OS-fullscreened', ()
); );
await once(w.webContents, 'leave-html-full-screen'); await once(w.webContents, 'leave-html-full-screen');
const width = await w.webContents.executeJavaScript( await expect(waitUntil(async () => {
"document.querySelector('iframe').offsetWidth" const width = await w.webContents.executeJavaScript(
); "document.querySelector('iframe').offsetWidth"
expect(width).to.equal(0); );
return width === 0;
})).to.eventually.be.fulfilled();
w.setFullScreen(false); w.setFullScreen(false);
await once(w, 'leave-full-screen'); await once(w, 'leave-full-screen');

View file

@ -9,6 +9,7 @@ import * as http2 from 'node:http2';
import * as https from 'node:https'; import * as https from 'node:https';
import * as net from 'node:net'; import * as net from 'node:net';
import * as path from 'node:path'; import * as path from 'node:path';
import { setTimeout } from 'node:timers/promises';
import * as url from 'node:url'; import * as url from 'node:url';
import * as v8 from 'node:v8'; import * as v8 from 'node:v8';
@ -94,49 +95,50 @@ export async function startRemoteControlApp (extraArgs: string[] = [], options?:
} }
export function waitUntil ( export function waitUntil (
callback: () => boolean, callback: () => boolean|Promise<boolean>,
opts: { rate?: number, timeout?: number } = {} opts: { rate?: number, timeout?: number } = {}
) { ) {
const { rate = 10, timeout = 10000 } = opts; const { rate = 10, timeout = 10000 } = opts;
return new Promise<void>((resolve, reject) => { return (async () => {
let intervalId: NodeJS.Timeout | undefined; // eslint-disable-line prefer-const const ac = new AbortController();
let timeoutId: NodeJS.Timeout | undefined; const signal = ac.signal;
let checkCompleted = false;
let timedOut = false;
const cleanup = () => { const check = async () => {
if (intervalId) clearInterval(intervalId);
if (timeoutId) clearTimeout(timeoutId);
};
const check = () => {
let result; let result;
try { try {
result = callback(); result = await callback();
} catch (e) { } catch (e) {
cleanup(); ac.abort();
reject(e); throw e;
return;
} }
if (result === true) { return result;
cleanup();
resolve();
return true;
}
}; };
if (check()) { setTimeout(timeout, { signal })
return; .then(() => {
timedOut = true;
checkCompleted = true;
});
while (checkCompleted === false) {
const checkSatisfied = await check();
if (checkSatisfied === true) {
ac.abort();
checkCompleted = true;
return;
} else {
await setTimeout(rate);
}
} }
intervalId = setInterval(check, rate); if (timedOut) {
throw new Error(`waitUntil timed out after ${timeout}ms`);
timeoutId = setTimeout(() => { }
timeoutId = undefined; })();
cleanup();
reject(new Error(`waitUntil timed out after ${timeout}ms`));
}, timeout);
});
} }
export async function repeatedly<T> ( export async function repeatedly<T> (