/** * 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 path from 'path'; import { test, expect, stripAnsi } from './playwright-test-fixtures'; test('should support toHaveCount', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent(''); const locator = page.locator('option'); let done = false; const promise = expect(locator).toHaveCount(2).then(() => { done = true; }); await page.waitForTimeout(1000); expect(done).toBe(false); await page.setContent(''); await promise; expect(done).toBe(true); }); test('pass zero', async ({ page }) => { await page.setContent('
'); const locator = page.locator('span'); await expect(locator).toHaveCount(0); await expect(locator).not.toHaveCount(1); }); test('eventually pass zero', async ({ page }) => { await page.setContent('
'); const locator = page.locator('span'); setTimeout(() => page.evaluate(() => div.textContent = '').catch(() => {}), 200); await expect(locator).toHaveCount(0); await expect(locator).not.toHaveCount(1); }); test('eventually pass non-zero', async ({ page }) => { await page.setContent(''); setTimeout(async () => { await page.setContent(""); }, 500); const locator = page.locator('li'); await expect(locator).toHaveCount(2); }); test('eventually pass not non-zero', async ({ page }) => { await page.setContent(''); setTimeout(async () => { await page.setContent(""); }, 500); const locator = page.locator('li'); await expect(locator).not.toHaveCount(2); }); test('fail zero', async ({ page }) => { await page.setContent('
'); const locator = page.locator('span'); await expect(locator).toHaveCount(0, { timeout: 500 }); }); test('fail zero 2', async ({ page }) => { await page.setContent('
'); const locator = page.locator('span'); await expect(locator).not.toHaveCount(1, { timeout: 500 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(result.passed).toBe(5); expect(result.failed).toBe(2); expect(result.exitCode).toBe(1); expect(output).toContain('Expected: 0'); expect(output).toContain('Received: 1'); expect(output).toContain('expect.toHaveCount with timeout 500ms'); }); test('should support toHaveJSProperty', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = { a: 1, b: 'string', c: new Date(1627503992000) }); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', { a: 1, b: 'string', c: new Date(1627503992000) }); }); test('fail', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = { a: 1, b: 'string', c: new Date(1627503992000) }); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', { a: 1, b: 'string', c: new Date(1627503992001) }, { timeout: 1000 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('- "c"'); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should support toHaveJSProperty with builtin types', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass string', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = 'string'); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', 'string'); }); test('fail string', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = 'string'); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', 'error', {timeout: 1000}); }); test('pass number', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = 2021); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', 2021); }); test('fail number', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = 2021); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', 1, {timeout: 1000}); }); test('pass boolean', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = true); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', true); }); test('fail boolean', async ({ page }) => { await page.setContent('
'); await page.$eval('div', e => e.foo = false); const locator = page.locator('div'); await expect(locator).toHaveJSProperty('foo', true, {timeout: 1000}); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(result.passed).toBe(3); expect(result.failed).toBe(3); expect(result.exitCode).toBe(1); expect(output).toContain('Expected: "error"'); expect(output).toContain('Received: "string"'); expect(output).toContain('Expected: 1'); expect(output).toContain('Received: 2021'); expect(output).toContain('Expected: true'); expect(output).toContain('Received: false'); }); test('should support toHaveClass', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent('
'); const locator = page.locator('div'); await expect(locator).toHaveClass('foo bar baz'); }); test('fail', async ({ page }) => { await page.setContent('
'); const locator = page.locator('div'); await expect(locator).toHaveClass('foo bar baz', { timeout: 1000 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(locator).toHaveClass'); expect(output).toContain('Expected string: \"foo bar baz\"'); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should support toHaveClass w/ array', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent('
'); const locator = page.locator('div'); await expect(locator).toHaveClass(['foo', 'bar', /[a-z]az/]); }); test('fail', async ({ page }) => { await page.setContent('
'); const locator = page.locator('div'); await expect(locator).toHaveClass(['foo', 'bar', /[a-z]az/], { timeout: 1000 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(received).toHaveClass(expected)'); expect(output).toContain('- /[a-z]az/,'); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should support toHaveTitle', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent(' Hello world'); await expect(page).toHaveTitle('Hello world'); }); test('fail', async ({ page }) => { await page.setContent('Bye'); await expect(page).toHaveTitle('Hello', { timeout: 100 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(page).toHaveTitle'); expect(output).toContain('Expected string: \"Hello\"'); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should support toHaveURL', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.goto('data:text/html,
A
'); await expect(page).toHaveURL('data:text/html,
A
'); }); test('fail', async ({ page }) => { await page.goto('data:text/html,
B
'); await expect(page).toHaveURL('wrong', { timeout: 100 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(page).toHaveURL'); expect(output).toContain('Expected string: \"wrong\"'); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should support toHaveURL with baseURL from webServer', async ({ runInlineTest }, testInfo) => { const port = testInfo.workerIndex + 10500; const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.goto('/foobar'); await expect(page).toHaveURL('/foobar'); await expect(page).toHaveURL('http://localhost:${port}/foobar'); }); test('fail', async ({ page }) => { await page.goto('/foobar'); await expect(page).toHaveURL('/kek', { timeout: 100 }); }); `, 'playwright.config.ts': ` module.exports = { webServer: { command: 'node ${JSON.stringify(path.join(__dirname, 'assets', 'simple-server.js'))} ${port}', port: ${port}, }, }; `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(page).toHaveURL'); expect(output).toContain(`Expected string: \"http://localhost:${port}/kek\"`); expect(result.passed).toBe(1); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should respect expect.timeout', async ({ runInlineTest }) => { const result = await runInlineTest({ 'playwright.config.js': `module.exports = { expect: { timeout: 1000 } }`, 'a.test.ts': ` const { test } = pwt; test('timeout', async ({ page }) => { await page.goto('data:text/html,
A
'); await Promise.all([ expect(page).toHaveURL('data:text/html,
B
'), new Promise(f => setTimeout(f, 2000)).then(() => expect(true).toBe(false)) ]); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); expect(output).toContain('expect(received).toHaveURL(expected)'); expect(output).toContain('expect.toHaveURL with timeout 1000ms'); expect(result.failed).toBe(1); expect(result.exitCode).toBe(1); }); test('should log scale the time', async ({ runInlineTest }) => { const result = await runInlineTest({ 'a.test.ts': ` const { test } = pwt; test('pass', async ({ page }) => { await page.setContent('
Wrong
'); await expect(page.locator('div')).toHaveText('Text', { timeout: 2000 }); }); `, }, { workers: 1 }); const output = stripAnsi(result.output); const tokens = output.split('unexpected value'); // Log scale: 0, 100, 250, 500, 1000, 1000, should be less than 8. expect(tokens.length).toBeGreaterThan(1); expect(tokens.length).toBeLessThan(8); expect(result.passed).toBe(0); expect(result.exitCode).toBe(1); });