fix(expect): report expect "Timed out" when it actually does (#22174)
Previously, it would say "Timed out" when page was closed at test timeout, or not say "Timed out" when at least one element matched. Fixes #21664.
This commit is contained in:
parent
26c00a97a5
commit
ab85b23e67
|
|
@ -41,6 +41,7 @@ import type { ScreenshotOptions } from './screenshotter';
|
||||||
import type { InputFilesItems } from './dom';
|
import type { InputFilesItems } from './dom';
|
||||||
import { asLocator } from '../utils/isomorphic/locatorGenerators';
|
import { asLocator } from '../utils/isomorphic/locatorGenerators';
|
||||||
import { FrameSelectors } from './frameSelectors';
|
import { FrameSelectors } from './frameSelectors';
|
||||||
|
import { TimeoutError } from '../common/errors';
|
||||||
|
|
||||||
type ContextData = {
|
type ContextData = {
|
||||||
contextPromise: ManualPromise<dom.FrameExecutionContext | Error>;
|
contextPromise: ManualPromise<dom.FrameExecutionContext | Error>;
|
||||||
|
|
@ -1435,7 +1436,7 @@ export class Frame extends SdkObject {
|
||||||
const result: { matches: boolean, received?: any, log?: string[], timedOut?: boolean } = { matches: options.isNot, log: metadata.log };
|
const result: { matches: boolean, received?: any, log?: string[], timedOut?: boolean } = { matches: options.isNot, log: metadata.log };
|
||||||
if (lastIntermediateResult.isSet)
|
if (lastIntermediateResult.isSet)
|
||||||
result.received = lastIntermediateResult.received;
|
result.received = lastIntermediateResult.received;
|
||||||
else
|
if (e instanceof TimeoutError)
|
||||||
result.timedOut = true;
|
result.timedOut = true;
|
||||||
return result;
|
return result;
|
||||||
});
|
});
|
||||||
|
|
|
||||||
52
tests/page/expect-timeout.spec.ts
Normal file
52
tests/page/expect-timeout.spec.ts
Normal file
|
|
@ -0,0 +1,52 @@
|
||||||
|
/**
|
||||||
|
* Copyright (c) Microsoft Corporation.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { stripAnsi } from '../config/utils';
|
||||||
|
import { test, expect } from './pageTest';
|
||||||
|
|
||||||
|
test('should print timed out error message', async ({ page }) => {
|
||||||
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
|
const error = await expect(page.locator('no-such-thing')).toHaveText('hey', { timeout: 1000 }).catch(e => e);
|
||||||
|
expect(stripAnsi(error.message)).toContain('Timed out 1000ms waiting for expect(received).toHaveText(expected)');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should print timed out error message when value does not match', async ({ page }) => {
|
||||||
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
|
const error = await expect(page.locator('div')).toHaveText('hey', { timeout: 1000 }).catch(e => e);
|
||||||
|
expect(stripAnsi(error.message)).toContain('Timed out 1000ms waiting for expect(received).toHaveText(expected)');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should print timed out error message with impossible timeout', async ({ page }) => {
|
||||||
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
|
const error = await expect(page.locator('no-such-thing')).toHaveText('hey', { timeout: 1 }).catch(e => e);
|
||||||
|
expect(stripAnsi(error.message)).toContain('Timed out 1ms waiting for expect(received).toHaveText(expected)');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should print timed out error message when value does not match with impossible timeout', async ({ page }) => {
|
||||||
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
|
const error = await expect(page.locator('div')).toHaveText('hey', { timeout: 1 }).catch(e => e);
|
||||||
|
expect(stripAnsi(error.message)).toContain('Timed out 1ms waiting for expect(received).toHaveText(expected)');
|
||||||
|
});
|
||||||
|
|
||||||
|
test('should not print timed out error message when page closes', async ({ page }) => {
|
||||||
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
|
const [error] = await Promise.all([
|
||||||
|
expect(page.locator('div')).toHaveText('hey', { timeout: 100000 }).catch(e => e),
|
||||||
|
page.close(),
|
||||||
|
]);
|
||||||
|
expect(stripAnsi(error.message)).toContain('expect.toHaveText with timeout 100000ms');
|
||||||
|
expect(stripAnsi(error.message)).not.toContain('Timed out');
|
||||||
|
});
|
||||||
|
|
@ -611,21 +611,23 @@ test('should print expected/received on Ctrl+C', async ({ runInlineTest }) => {
|
||||||
expect(result.output).toContain('Received string: "Text content"');
|
expect(result.output).toContain('Received string: "Text content"');
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should print timed out error message', async ({ runInlineTest }) => {
|
test('should not print timed out error message when test times out', async ({ runInlineTest }) => {
|
||||||
const result = await runInlineTest({
|
const result = await runInlineTest({
|
||||||
'a.test.ts': `
|
'a.test.ts': `
|
||||||
import { test, expect } from '@playwright/test';
|
import { test, expect } from '@playwright/test';
|
||||||
|
|
||||||
test('fail', async ({ page }) => {
|
test('fail', async ({ page }) => {
|
||||||
await page.setContent('<div id=node>Text content</div>');
|
await page.setContent('<div id=node>Text content</div>');
|
||||||
await expect(page.locator('no-such-thing')).toHaveText('hey', { timeout: 1000 });
|
await expect(page.locator('no-such-thing')).toHaveText('hey', { timeout: 5000 });
|
||||||
});
|
});
|
||||||
`,
|
`,
|
||||||
}, { workers: 1 });
|
}, { workers: 1, timeout: 3000 });
|
||||||
expect(result.failed).toBe(1);
|
expect(result.failed).toBe(1);
|
||||||
expect(result.exitCode).toBe(1);
|
expect(result.exitCode).toBe(1);
|
||||||
const output = result.output;
|
const output = result.output;
|
||||||
expect(output).toContain('Timed out 1000ms waiting for expect(received).toHaveText(expected)');
|
expect(output).toContain('Test timeout of 3000ms exceeded');
|
||||||
|
expect(output).not.toContain('Timed out 5000ms waiting for expect');
|
||||||
|
expect(output).toContain('Error: expect(received).toHaveText(expected)');
|
||||||
});
|
});
|
||||||
|
|
||||||
test('should not leak long expect message strings', async ({ runInlineTest }) => {
|
test('should not leak long expect message strings', async ({ runInlineTest }) => {
|
||||||
|
|
|
||||||
|
|
@ -53,7 +53,7 @@ test('should update trace live', async ({ runUITest, server }) => {
|
||||||
).toHaveText([
|
).toHaveText([
|
||||||
/browserContext.newPage[\d.]+m?s/,
|
/browserContext.newPage[\d.]+m?s/,
|
||||||
/page.gotohttp:\/\/localhost:\d+\/one.html/
|
/page.gotohttp:\/\/localhost:\d+\/one.html/
|
||||||
]);
|
], { timeout: 15000 });
|
||||||
|
|
||||||
await expect(
|
await expect(
|
||||||
listItem.locator(':scope.selected'),
|
listItem.locator(':scope.selected'),
|
||||||
|
|
@ -138,7 +138,7 @@ test('should preserve action list selection upon live trace update', async ({ ru
|
||||||
/browserContext.newPage[\d.]+m?s/,
|
/browserContext.newPage[\d.]+m?s/,
|
||||||
/page.gotoabout:blank[\d.]+m?s/,
|
/page.gotoabout:blank[\d.]+m?s/,
|
||||||
/page.setContent[\d.]+m?s/,
|
/page.setContent[\d.]+m?s/,
|
||||||
]);
|
], { timeout: 15000 });
|
||||||
|
|
||||||
// Manually select page.goto.
|
// Manually select page.goto.
|
||||||
await page.getByTestId('action-list').getByText('page.goto').click();
|
await page.getByTestId('action-list').getByText('page.goto').click();
|
||||||
|
|
@ -199,7 +199,7 @@ test('should update tracing network live', async ({ runUITest, server }) => {
|
||||||
/browserContext.newPage[\d.]+m?s/,
|
/browserContext.newPage[\d.]+m?s/,
|
||||||
/page.gotohttp:\/\/localhost:\d+\/one.html[\d.]+m?s/,
|
/page.gotohttp:\/\/localhost:\d+\/one.html[\d.]+m?s/,
|
||||||
/page.setContent[\d.]+m?s/,
|
/page.setContent[\d.]+m?s/,
|
||||||
]);
|
], { timeout: 15000 });
|
||||||
|
|
||||||
// Once page.setContent is visible, we can be sure that page.goto has all required
|
// Once page.setContent is visible, we can be sure that page.goto has all required
|
||||||
// resources in the trace. Switch to it and check that everything renders.
|
// resources in the trace. Switch to it and check that everything renders.
|
||||||
|
|
|
||||||
|
|
@ -602,7 +602,7 @@ test('should run CT on changed deps', async ({ runWatchTest, writeFiles }) => {
|
||||||
|
|
||||||
await testProcess.waitForOutput(`src${path.sep}button.spec.tsx:4:11 › pass`);
|
await testProcess.waitForOutput(`src${path.sep}button.spec.tsx:4:11 › pass`);
|
||||||
expect(testProcess.output).not.toContain(`src${path.sep}link.spec.tsx`);
|
expect(testProcess.output).not.toContain(`src${path.sep}link.spec.tsx`);
|
||||||
await testProcess.waitForOutput('Error: expect(received).toHaveText(expected)');
|
await testProcess.waitForOutput('Error: Timed out 1000ms waiting for expect(received).toHaveText(expected)');
|
||||||
await testProcess.waitForOutput('Waiting for file changes.');
|
await testProcess.waitForOutput('Waiting for file changes.');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue