mirror of
https://github.com/microsoft/playwright.git
synced 2025-06-26 21:40:17 +00:00

1. Fixtures defined in test.extend() can now have `{ option: true }` configuration that makes them overridable in the config. Options support all other properties of fixtures - value/function, scope, auto. ``` const test = base.extend<MyOptions>({ foo: ['default', { option: true }], }); ``` 2. test.declare() and project.define are removed. 3. project.use applies overrides to default option values and nothing else. Any test.extend() and test.use() calls take priority over config options. Required user changes: if someone used to define fixture options with test.extend(), overriding them in config will stop working. The solution is to add `{ option: true }`. ``` // Old code export const test = base.extend<{ myOption: number, myFixture: number }>({ myOption: 123, myFixture: ({ myOption }, use) => use(2 * myOption), }); // New code export const test = base.extend<{ myOption: number, myFixture: number }>({ myOption: [123, { option: true }], myFixture: ({ myOption }, use) => use(2 * myOption), }); ```
126 lines
4.6 KiB
TypeScript
126 lines
4.6 KiB
TypeScript
/**
|
|
* Copyright Microsoft Corporation. All rights reserved.
|
|
*
|
|
* 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 type { FullProject, Fixtures, FixturesWithLocation } from './types';
|
|
import { Suite, TestCase } from './test';
|
|
import { FixturePool, isFixtureOption } from './fixtures';
|
|
import { TestTypeImpl } from './testType';
|
|
|
|
export class ProjectImpl {
|
|
config: FullProject;
|
|
private index: number;
|
|
private testTypePools = new Map<TestTypeImpl, FixturePool>();
|
|
private testPools = new Map<TestCase, FixturePool>();
|
|
|
|
constructor(project: FullProject, index: number) {
|
|
this.config = project;
|
|
this.index = index;
|
|
}
|
|
|
|
private buildTestTypePool(testType: TestTypeImpl): FixturePool {
|
|
if (!this.testTypePools.has(testType)) {
|
|
const fixtures = this.resolveFixtures(testType, this.config.use);
|
|
const pool = new FixturePool(fixtures);
|
|
this.testTypePools.set(testType, pool);
|
|
}
|
|
return this.testTypePools.get(testType)!;
|
|
}
|
|
|
|
// TODO: we can optimize this function by building the pool inline in cloneSuite
|
|
private buildPool(test: TestCase): FixturePool {
|
|
if (!this.testPools.has(test)) {
|
|
let pool = this.buildTestTypePool(test._testType);
|
|
|
|
const parents: Suite[] = [];
|
|
for (let parent: Suite | undefined = test.parent; parent; parent = parent.parent)
|
|
parents.push(parent);
|
|
parents.reverse();
|
|
|
|
for (const parent of parents) {
|
|
if (parent._use.length)
|
|
pool = new FixturePool(parent._use, pool, parent._isDescribe);
|
|
for (const hook of parent._eachHooks)
|
|
pool.validateFunction(hook.fn, hook.type + ' hook', hook.location);
|
|
for (const hook of parent._allHooks)
|
|
pool.validateFunction(hook.fn, hook._type + ' hook', hook.location);
|
|
for (const modifier of parent._modifiers)
|
|
pool.validateFunction(modifier.fn, modifier.type + ' modifier', modifier.location);
|
|
}
|
|
|
|
pool.validateFunction(test.fn, 'Test', test.location);
|
|
this.testPools.set(test, pool);
|
|
}
|
|
return this.testPools.get(test)!;
|
|
}
|
|
|
|
private _cloneEntries(from: Suite, to: Suite, repeatEachIndex: number, filter: (test: TestCase) => boolean): boolean {
|
|
for (const entry of from._entries) {
|
|
if (entry instanceof Suite) {
|
|
const suite = entry._clone();
|
|
to._addSuite(suite);
|
|
if (!this._cloneEntries(entry, suite, repeatEachIndex, filter)) {
|
|
to._entries.pop();
|
|
to.suites.pop();
|
|
}
|
|
} else {
|
|
const test = entry._clone();
|
|
test.retries = this.config.retries;
|
|
test._id = `${entry._ordinalInFile}@${entry._requireFile}#run${this.index}-repeat${repeatEachIndex}`;
|
|
test._repeatEachIndex = repeatEachIndex;
|
|
test._projectIndex = this.index;
|
|
to._addTest(test);
|
|
if (!filter(test)) {
|
|
to._entries.pop();
|
|
to.tests.pop();
|
|
} else {
|
|
const pool = this.buildPool(entry);
|
|
test._workerHash = `run${this.index}-${pool.digest}-repeat${repeatEachIndex}`;
|
|
test._pool = pool;
|
|
}
|
|
}
|
|
}
|
|
if (!to._entries.length)
|
|
return false;
|
|
for (const hook of from._allHooks) {
|
|
const clone = hook._clone();
|
|
clone._pool = this.buildPool(hook);
|
|
clone._projectIndex = this.index;
|
|
to._addAllHook(clone);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
cloneFileSuite(suite: Suite, repeatEachIndex: number, filter: (test: TestCase) => boolean): Suite | undefined {
|
|
const result = suite._clone();
|
|
return this._cloneEntries(suite, result, repeatEachIndex, filter) ? result : undefined;
|
|
}
|
|
|
|
private resolveFixtures(testType: TestTypeImpl, configUse: Fixtures): FixturesWithLocation[] {
|
|
return testType.fixtures.map(f => {
|
|
const configKeys = new Set(Object.keys(configUse || {}));
|
|
const resolved = { ...f.fixtures };
|
|
for (const [key, value] of Object.entries(resolved)) {
|
|
if (!isFixtureOption(value) || !configKeys.has(key))
|
|
continue;
|
|
// Apply override from config file.
|
|
const override = (configUse as any)[key];
|
|
(resolved as any)[key] = [override, value[1]];
|
|
}
|
|
return { fixtures: resolved, location: f.location };
|
|
});
|
|
}
|
|
}
|