2021-06-07 03:09:53 +03:00
|
|
|
|
/**
|
|
|
|
|
* 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 * as fs from 'fs';
|
|
|
|
|
import * as path from 'path';
|
|
|
|
|
import { test, expect } from './playwright-test-fixtures';
|
|
|
|
|
|
|
|
|
|
test('should be able to define config', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = { timeout: 12345 };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
|
|
|
|
expect(testInfo.timeout).toBe(12345);
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should prioritize project timeout', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = { timeout: 500, projects: [{ timeout: 10000}, {}] };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
|
|
|
|
await new Promise(f => setTimeout(f, 1500));
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(1);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
expect(result.failed).toBe(1);
|
|
|
|
|
expect(result.output).toContain('Timeout of 500ms exceeded.');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should prioritize command line timeout over project timeout', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = { projects: [{ timeout: 10000}] };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
|
|
|
|
await new Promise(f => setTimeout(f, 1500));
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { timeout: '500' });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(1);
|
|
|
|
|
expect(result.failed).toBe(1);
|
|
|
|
|
expect(result.output).toContain('Timeout of 500ms exceeded.');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should read config from --config, resolve relative testDir', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'my.config.ts': `
|
|
|
|
|
import * as path from 'path';
|
|
|
|
|
module.exports = {
|
|
|
|
|
testDir: 'dir',
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('ignored', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
'dir/b.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('run', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
}, { config: 'my.config.ts' });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
expect(result.report.suites.length).toBe(1);
|
|
|
|
|
expect(result.report.suites[0].file).toBe('b.test.ts');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should default testDir to the config file', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'dir/my.config.ts': `
|
|
|
|
|
module.exports = {};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('ignored', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
'dir/b.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('run', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
}, { config: path.join('dir', 'my.config.ts') });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
expect(result.report.suites.length).toBe(1);
|
|
|
|
|
expect(result.report.suites[0].file).toBe('b.test.ts');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should be able to set reporters', async ({ runInlineTest }, testInfo) => {
|
|
|
|
|
const reportFile = testInfo.outputPath('my-report.json');
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = {
|
|
|
|
|
reporter: [
|
|
|
|
|
['json', { outputFile: ${JSON.stringify(reportFile)} }],
|
|
|
|
|
['list'],
|
|
|
|
|
]
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async () => {
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { reporter: '' });
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
const report = JSON.parse(fs.readFileSync(reportFile).toString());
|
|
|
|
|
expect(report.suites[0].file).toBe('a.test.ts');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should support different testDirs', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
import * as path from 'path';
|
|
|
|
|
module.exports = { projects: [
|
|
|
|
|
{ testDir: __dirname },
|
|
|
|
|
{ testDir: 'dir' },
|
|
|
|
|
] };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('runs once', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
'dir/b.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('runs twice', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(3);
|
|
|
|
|
|
|
|
|
|
expect(result.report.suites[0].specs[0].tests.length).toBe(1);
|
|
|
|
|
expect(result.report.suites[0].specs[0].title).toBe('runs once');
|
|
|
|
|
|
|
|
|
|
expect(result.report.suites[1].specs[0].tests.length).toBe(2);
|
|
|
|
|
expect(result.report.suites[1].specs[0].title).toBe('runs twice');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
test('should allow root testDir and use it for relative paths', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'config/config.ts': `
|
|
|
|
|
import * as path from 'path';
|
|
|
|
|
module.exports = {
|
|
|
|
|
testDir: path.join(__dirname, '..'),
|
|
|
|
|
projects: [{ testDir: path.join(__dirname, '..', 'dir') }]
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('fails', async ({}, testInfo) => {
|
|
|
|
|
expect(1 + 1).toBe(3);
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
'dir/a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('fails', async ({}, testInfo) => {
|
|
|
|
|
expect(1 + 1).toBe(3);
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
}, { config: path.join('config', 'config.ts') });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(1);
|
|
|
|
|
expect(result.passed).toBe(0);
|
|
|
|
|
expect(result.skipped).toBe(0);
|
|
|
|
|
expect(result.failed).toBe(1);
|
|
|
|
|
expect(result.output).toContain(`1) ${path.join('dir', 'a.test.ts')}:6:7 › fails`);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should throw when test() is called in config file', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
pwt.test('hey', () => {});
|
2021-06-07 03:09:53 +03:00
|
|
|
|
module.exports = {};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('test', async ({}) => {
|
|
|
|
|
});
|
|
|
|
|
`,
|
|
|
|
|
});
|
|
|
|
|
expect(result.output).toContain('test() can only be called in a test file');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should filter by project, case-insensitive', async ({ runInlineTest }) => {
|
|
|
|
|
const { passed, failed, output, skipped } = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = { projects: [
|
|
|
|
|
{ name: 'suite1' },
|
|
|
|
|
{ name: 'suite2' },
|
|
|
|
|
] };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
|
|
|
|
console.log(testInfo.project.name);
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { project: 'SUite2' });
|
|
|
|
|
expect(passed).toBe(1);
|
|
|
|
|
expect(failed).toBe(0);
|
|
|
|
|
expect(skipped).toBe(0);
|
|
|
|
|
expect(output).toContain('suite2');
|
|
|
|
|
expect(output).not.toContain('suite1');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should print nice error when project is unknown', async ({ runInlineTest }) => {
|
|
|
|
|
const { output, exitCode } = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = { projects: [
|
|
|
|
|
{ name: 'suite1' },
|
|
|
|
|
{ name: 'suite2' },
|
|
|
|
|
] };
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
|
|
|
|
console.log(testInfo.project.name);
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { project: 'suite3' });
|
|
|
|
|
expect(exitCode).toBe(1);
|
|
|
|
|
expect(output).toContain('Project "suite3" not found. Available named projects: "suite1", "suite2"');
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should work without config file', async ({ runInlineTest }) => {
|
|
|
|
|
const { exitCode, passed, failed, skipped } = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
throw new Error('This file should not be required');
|
|
|
|
|
`,
|
|
|
|
|
'dir/a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}) => {
|
|
|
|
|
test.expect(1 + 1).toBe(2);
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { config: 'dir' });
|
|
|
|
|
expect(exitCode).toBe(0);
|
|
|
|
|
expect(passed).toBe(1);
|
|
|
|
|
expect(failed).toBe(0);
|
|
|
|
|
expect(skipped).toBe(0);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should inerhit use options in projects', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = {
|
|
|
|
|
use: { foo: 'config' },
|
|
|
|
|
projects: [{
|
|
|
|
|
use: { bar: 'project' },
|
|
|
|
|
}]
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({ foo, bar }, testInfo) => {
|
|
|
|
|
test.expect(foo).toBe('config');
|
|
|
|
|
test.expect(bar).toBe('project');
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should work with undefined values and base', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = {
|
|
|
|
|
updateSnapshots: undefined,
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}, testInfo) => {
|
2021-06-16 17:51:54 +03:00
|
|
|
|
expect(testInfo.config.updateSnapshots).toBe('missing');
|
2021-06-07 03:09:53 +03:00
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, {}, { CI: '1' });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.passed).toBe(1);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
test('should work with custom reporter', async ({ runInlineTest }) => {
|
|
|
|
|
const result = await runInlineTest({
|
|
|
|
|
'reporter.ts': `
|
|
|
|
|
class Reporter {
|
|
|
|
|
constructor(options) {
|
|
|
|
|
this.options = options;
|
|
|
|
|
}
|
|
|
|
|
onBegin() {
|
|
|
|
|
console.log('\\n%%reporter-begin%%' + this.options.begin);
|
|
|
|
|
}
|
|
|
|
|
onTestBegin() {
|
|
|
|
|
console.log('\\n%%reporter-testbegin%%');
|
|
|
|
|
}
|
|
|
|
|
onStdOut() {
|
|
|
|
|
console.log('\\n%%reporter-stdout%%');
|
|
|
|
|
}
|
|
|
|
|
onStdErr() {
|
|
|
|
|
console.log('\\n%%reporter-stderr%%');
|
|
|
|
|
}
|
|
|
|
|
onTestEnd() {
|
|
|
|
|
console.log('\\n%%reporter-testend%%');
|
|
|
|
|
}
|
|
|
|
|
onTimeout() {
|
|
|
|
|
console.log('\\n%%reporter-timeout%%');
|
|
|
|
|
}
|
|
|
|
|
onError() {
|
|
|
|
|
console.log('\\n%%reporter-error%%');
|
|
|
|
|
}
|
2021-06-29 20:55:46 +03:00
|
|
|
|
async onEnd() {
|
|
|
|
|
await new Promise(f => setTimeout(f, 500));
|
2021-06-07 03:09:53 +03:00
|
|
|
|
console.log('\\n%%reporter-end%%' + this.options.end);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
export default Reporter;
|
|
|
|
|
`,
|
|
|
|
|
'playwright.config.ts': `
|
|
|
|
|
module.exports = {
|
|
|
|
|
reporter: [
|
|
|
|
|
[ './reporter.ts', { begin: 'begin', end: 'end' } ]
|
|
|
|
|
]
|
|
|
|
|
};
|
|
|
|
|
`,
|
|
|
|
|
'a.test.ts': `
|
2021-06-07 08:07:07 +03:00
|
|
|
|
const { test } = pwt;
|
2021-06-07 03:09:53 +03:00
|
|
|
|
test('pass', async ({}) => {
|
|
|
|
|
console.log('log');
|
|
|
|
|
console.error('error');
|
|
|
|
|
});
|
|
|
|
|
`
|
|
|
|
|
}, { reporter: '' });
|
|
|
|
|
|
|
|
|
|
expect(result.exitCode).toBe(0);
|
|
|
|
|
expect(result.output.split('\n').filter(line => line.startsWith('%%'))).toEqual([
|
|
|
|
|
'%%reporter-begin%%begin',
|
|
|
|
|
'%%reporter-testbegin%%',
|
|
|
|
|
'%%reporter-stdout%%',
|
|
|
|
|
'%%reporter-stderr%%',
|
|
|
|
|
'%%reporter-testend%%',
|
|
|
|
|
'%%reporter-end%%end',
|
|
|
|
|
]);
|
|
|
|
|
});
|