mirror of
https://github.com/microsoft/playwright.git
synced 2025-01-07 11:46:42 +03:00
parent
5953472899
commit
bf35da3656
4
package-lock.json
generated
4
package-lock.json
generated
@ -8894,6 +8894,7 @@
|
|||||||
},
|
},
|
||||||
"node_modules/socksv5/node_modules/ipv6": {
|
"node_modules/socksv5/node_modules/ipv6": {
|
||||||
"version": "3.1.1",
|
"version": "3.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ipv6/-/ipv6-3.1.1.tgz",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"inBundle": true,
|
"inBundle": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
@ -8912,6 +8913,7 @@
|
|||||||
},
|
},
|
||||||
"node_modules/socksv5/node_modules/ipv6/node_modules/sprintf": {
|
"node_modules/socksv5/node_modules/ipv6/node_modules/sprintf": {
|
||||||
"version": "0.1.3",
|
"version": "0.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/sprintf/-/sprintf-0.1.3.tgz",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"inBundle": true,
|
"inBundle": true,
|
||||||
"engines": {
|
"engines": {
|
||||||
@ -17858,6 +17860,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"ipv6": {
|
"ipv6": {
|
||||||
"version": "3.1.1",
|
"version": "3.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/ipv6/-/ipv6-3.1.1.tgz",
|
||||||
"bundled": true,
|
"bundled": true,
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"requires": {
|
"requires": {
|
||||||
@ -17868,6 +17871,7 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"sprintf": {
|
"sprintf": {
|
||||||
"version": "0.1.3",
|
"version": "0.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/sprintf/-/sprintf-0.1.3.tgz",
|
||||||
"bundled": true,
|
"bundled": true,
|
||||||
"dev": true
|
"dev": true
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,7 @@ import * as fs from 'fs';
|
|||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import { promisify } from 'util';
|
import { promisify } from 'util';
|
||||||
import { Dispatcher, TestGroup } from './dispatcher';
|
import { Dispatcher, TestGroup } from './dispatcher';
|
||||||
import { createMatcher, FilePatternFilter, monotonicTime } from './util';
|
import { createFileMatcher, createTitleMatcher, FilePatternFilter, monotonicTime } from './util';
|
||||||
import { TestCase, Suite } from './test';
|
import { TestCase, Suite } from './test';
|
||||||
import { Loader } from './loader';
|
import { Loader } from './loader';
|
||||||
import { Reporter } from '../../types/testReporter';
|
import { Reporter } from '../../types/testReporter';
|
||||||
@ -131,7 +131,7 @@ export class Runner {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async _run(list: boolean, testFileReFilters: FilePatternFilter[], projectNames?: string[]): Promise<RunResult> {
|
async _run(list: boolean, testFileReFilters: FilePatternFilter[], projectNames?: string[]): Promise<RunResult> {
|
||||||
const testFileFilter = testFileReFilters.length ? createMatcher(testFileReFilters.map(e => e.re)) : () => true;
|
const testFileFilter = testFileReFilters.length ? createFileMatcher(testFileReFilters.map(e => e.re)) : () => true;
|
||||||
const config = this._loader.fullConfig();
|
const config = this._loader.fullConfig();
|
||||||
|
|
||||||
let projectsToFind: Set<string> | undefined;
|
let projectsToFind: Set<string> | undefined;
|
||||||
@ -169,8 +169,8 @@ export class Runner {
|
|||||||
if (!fs.statSync(testDir).isDirectory())
|
if (!fs.statSync(testDir).isDirectory())
|
||||||
throw new Error(`${testDir} is not a directory`);
|
throw new Error(`${testDir} is not a directory`);
|
||||||
const allFiles = await collectFiles(project.config.testDir);
|
const allFiles = await collectFiles(project.config.testDir);
|
||||||
const testMatch = createMatcher(project.config.testMatch);
|
const testMatch = createFileMatcher(project.config.testMatch);
|
||||||
const testIgnore = createMatcher(project.config.testIgnore);
|
const testIgnore = createFileMatcher(project.config.testIgnore);
|
||||||
const testFileExtension = (file: string) => ['.js', '.ts', '.mjs'].includes(path.extname(file));
|
const testFileExtension = (file: string) => ['.js', '.ts', '.mjs'].includes(path.extname(file));
|
||||||
const testFiles = allFiles.filter(file => !testIgnore(file) && testMatch(file) && testFileFilter(file) && testFileExtension(file));
|
const testFiles = allFiles.filter(file => !testIgnore(file) && testMatch(file) && testFileFilter(file) && testFileExtension(file));
|
||||||
files.set(project, testFiles);
|
files.set(project, testFiles);
|
||||||
@ -210,8 +210,8 @@ export class Runner {
|
|||||||
fileSuites.set(fileSuite._requireFile, fileSuite);
|
fileSuites.set(fileSuite._requireFile, fileSuite);
|
||||||
|
|
||||||
const outputDirs = new Set<string>();
|
const outputDirs = new Set<string>();
|
||||||
const grepMatcher = createMatcher(config.grep);
|
const grepMatcher = createTitleMatcher(config.grep);
|
||||||
const grepInvertMatcher = config.grepInvert ? createMatcher(config.grepInvert) : null;
|
const grepInvertMatcher = config.grepInvert ? createTitleMatcher(config.grepInvert) : null;
|
||||||
const rootSuite = new Suite('');
|
const rootSuite = new Suite('');
|
||||||
for (const project of projects) {
|
for (const project of projects) {
|
||||||
const projectSuite = new Suite(project.config.name);
|
const projectSuite = new Suite(project.config.name);
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
import type { TestInfoImpl } from './types';
|
import type { TestInfoImpl } from './types';
|
||||||
import util from 'util';
|
import util from 'util';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
import url from 'url';
|
||||||
import type { TestError, Location } from './types';
|
import type { TestError, Location } from './types';
|
||||||
import { default as minimatch } from 'minimatch';
|
import { default as minimatch } from 'minimatch';
|
||||||
import { errors } from '../..';
|
import { errors } from '../..';
|
||||||
@ -91,7 +92,7 @@ export type FilePatternFilter = {
|
|||||||
line: number | null;
|
line: number | null;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function createMatcher(patterns: string | RegExp | (string | RegExp)[]): Matcher {
|
export function createFileMatcher(patterns: string | RegExp | (string | RegExp)[]): Matcher {
|
||||||
const reList: RegExp[] = [];
|
const reList: RegExp[] = [];
|
||||||
const filePatterns: string[] = [];
|
const filePatterns: string[] = [];
|
||||||
for (const pattern of Array.isArray(patterns) ? patterns : [patterns]) {
|
for (const pattern of Array.isArray(patterns) ? patterns : [patterns]) {
|
||||||
@ -104,17 +105,38 @@ export function createMatcher(patterns: string | RegExp | (string | RegExp)[]):
|
|||||||
filePatterns.push(pattern);
|
filePatterns.push(pattern);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return (filePath: string) => {
|
||||||
|
for (const re of reList) {
|
||||||
|
re.lastIndex = 0;
|
||||||
|
if (re.test(filePath))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
// Windows might still recieve unix style paths from Cygwin or Git Bash.
|
||||||
|
// Check against the file url as well.
|
||||||
|
if (path.sep === '\\') {
|
||||||
|
const fileURL = url.pathToFileURL(filePath).href;
|
||||||
|
for (const re of reList) {
|
||||||
|
re.lastIndex = 0;
|
||||||
|
if (re.test(fileURL))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (const pattern of filePatterns) {
|
||||||
|
if (minimatch(filePath, pattern, { nocase: true, dot: true }))
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createTitleMatcher(patterns: RegExp | RegExp[]): Matcher {
|
||||||
|
const reList = Array.isArray(patterns) ? patterns : [patterns];
|
||||||
return (value: string) => {
|
return (value: string) => {
|
||||||
for (const re of reList) {
|
for (const re of reList) {
|
||||||
re.lastIndex = 0;
|
re.lastIndex = 0;
|
||||||
if (re.test(value))
|
if (re.test(value))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
for (const pattern of filePatterns) {
|
|
||||||
if (minimatch(value, pattern, { nocase: true, dot: true }))
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -418,3 +418,28 @@ test('should match in dot-directories', async ({ runInlineTest }) => {
|
|||||||
expect(result.report.suites.map(s => s.file).sort()).toEqual(['.dir/a.test.ts', '.dir/b.test.js']);
|
expect(result.report.suites.map(s => s.file).sort()).toEqual(['.dir/a.test.ts', '.dir/b.test.js']);
|
||||||
expect(result.exitCode).toBe(0);
|
expect(result.exitCode).toBe(0);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test('should always work with unix separators', async ({ runInlineTest }) => {
|
||||||
|
// Cygwin or Git Bash might send us a path with unix separators.
|
||||||
|
const result = await runInlineTest({
|
||||||
|
'playwright.config.ts': `
|
||||||
|
import * as path from 'path';
|
||||||
|
module.exports = { testDir: path.join(__dirname, 'dir') };
|
||||||
|
`,
|
||||||
|
'dir/a.test.ts': `
|
||||||
|
const { test } = pwt;
|
||||||
|
test('pass', ({}) => {});
|
||||||
|
`,
|
||||||
|
'dir/b.test.ts': `
|
||||||
|
const { test } = pwt;
|
||||||
|
test('pass', ({}) => {});
|
||||||
|
`,
|
||||||
|
'a.test.ts': `
|
||||||
|
const { test } = pwt;
|
||||||
|
test('pass', ({}) => {});
|
||||||
|
`
|
||||||
|
}, { args: [`dir/a`] });
|
||||||
|
expect(result.passed).toBe(1);
|
||||||
|
expect(result.report.suites.map(s => s.file).sort()).toEqual(['a.test.ts']);
|
||||||
|
expect(result.exitCode).toBe(0);
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user