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:
parent
1a9b93ca96
commit
e9988c2fc4
4 changed files with 45 additions and 38 deletions
|
@ -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();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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));
|
||||||
|
|
|
@ -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');
|
||||||
|
|
|
@ -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> (
|
||||||
|
|
Loading…
Reference in a new issue