2023-09-22 08:58:14 +00:00
|
|
|
#! /usr/bin/env -S node
|
|
|
|
|
2022-09-09 09:12:18 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2022 Google Inc. 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
|
|
|
|
*
|
|
|
|
* https://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.
|
|
|
|
*/
|
|
|
|
|
2023-03-24 14:31:47 +00:00
|
|
|
import {randomUUID} from 'crypto';
|
2023-02-15 23:09:31 +00:00
|
|
|
import fs from 'fs';
|
2023-09-27 14:02:55 +00:00
|
|
|
import {spawn} from 'node:child_process';
|
2023-02-15 23:09:31 +00:00
|
|
|
import os from 'os';
|
|
|
|
import path from 'path';
|
|
|
|
|
2023-09-25 12:00:09 +00:00
|
|
|
import {globSync} from 'glob';
|
2023-09-27 14:02:55 +00:00
|
|
|
import yargs from 'yargs';
|
|
|
|
import {hideBin} from 'yargs/helpers';
|
2023-09-25 12:00:09 +00:00
|
|
|
|
2022-09-08 10:32:39 +00:00
|
|
|
import {
|
|
|
|
zPlatform,
|
2023-09-22 08:58:14 +00:00
|
|
|
zTestSuiteFile,
|
|
|
|
type MochaResults,
|
|
|
|
type Platform,
|
|
|
|
type TestExpectation,
|
2023-09-15 11:00:20 +00:00
|
|
|
type TestSuite,
|
|
|
|
type TestSuiteFile,
|
2022-09-08 10:32:39 +00:00
|
|
|
} from './types.js';
|
|
|
|
import {
|
|
|
|
extendProcessEnv,
|
|
|
|
filterByParameters,
|
2023-09-22 08:58:14 +00:00
|
|
|
filterByPlatform,
|
2022-09-08 10:32:39 +00:00
|
|
|
getExpectationUpdates,
|
2023-02-28 11:55:20 +00:00
|
|
|
printSuggestions,
|
2023-09-22 08:58:14 +00:00
|
|
|
readJSON,
|
2023-03-24 14:31:47 +00:00
|
|
|
writeJSON,
|
2023-09-22 08:58:14 +00:00
|
|
|
type RecommendedExpectation,
|
2022-09-08 10:32:39 +00:00
|
|
|
} from './utils.js';
|
|
|
|
|
2023-09-27 14:02:55 +00:00
|
|
|
const {
|
|
|
|
_: mochaArgs,
|
|
|
|
testSuite: testSuiteId,
|
|
|
|
saveStatsTo,
|
|
|
|
cdpTests: includeCdpTests,
|
|
|
|
suggestions: provideSuggestions,
|
|
|
|
coverage: useCoverage,
|
|
|
|
minTests,
|
|
|
|
shard,
|
|
|
|
reporter,
|
|
|
|
} = yargs(hideBin(process.argv))
|
|
|
|
.parserConfiguration({'unknown-options-as-args': true})
|
|
|
|
.scriptName('@puppeteer/mocha-runner')
|
|
|
|
.option('coverage', {
|
|
|
|
boolean: true,
|
|
|
|
default: true,
|
|
|
|
})
|
|
|
|
.option('suggestions', {
|
|
|
|
boolean: true,
|
|
|
|
default: true,
|
|
|
|
})
|
|
|
|
.option('cdp-tests', {
|
|
|
|
boolean: true,
|
|
|
|
default: true,
|
|
|
|
})
|
|
|
|
.option('save-stats-to', {
|
|
|
|
string: true,
|
|
|
|
requiresArg: true,
|
|
|
|
})
|
|
|
|
.option('min-tests', {
|
|
|
|
number: true,
|
|
|
|
default: 0,
|
|
|
|
requiresArg: true,
|
|
|
|
})
|
|
|
|
.option('test-suite', {
|
|
|
|
string: true,
|
|
|
|
requiresArg: true,
|
|
|
|
})
|
|
|
|
.option('shard', {
|
|
|
|
string: true,
|
|
|
|
requiresArg: true,
|
|
|
|
})
|
|
|
|
.option('reporter', {
|
|
|
|
string: true,
|
|
|
|
requiresArg: true,
|
|
|
|
})
|
|
|
|
.parseSync();
|
|
|
|
|
2022-09-08 10:32:39 +00:00
|
|
|
function getApplicableTestSuites(
|
|
|
|
parsedSuitesFile: TestSuiteFile,
|
|
|
|
platform: Platform
|
|
|
|
): TestSuite[] {
|
|
|
|
let applicableSuites: TestSuite[] = [];
|
|
|
|
|
2023-09-27 14:02:55 +00:00
|
|
|
if (!testSuiteId) {
|
2022-09-08 10:32:39 +00:00
|
|
|
applicableSuites = filterByPlatform(parsedSuitesFile.testSuites, platform);
|
|
|
|
} else {
|
|
|
|
const testSuite = parsedSuitesFile.testSuites.find(suite => {
|
|
|
|
return suite.id === testSuiteId;
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!testSuite) {
|
|
|
|
console.error(`Test suite ${testSuiteId} is not defined`);
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!testSuite.platforms.includes(platform)) {
|
|
|
|
console.warn(
|
|
|
|
`Test suite ${testSuiteId} is not enabled for your platform. Running it anyway.`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
applicableSuites = [testSuite];
|
|
|
|
}
|
|
|
|
|
|
|
|
return applicableSuites;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function main() {
|
2023-09-27 14:02:55 +00:00
|
|
|
let statsPath = saveStatsTo;
|
|
|
|
if (statsPath && statsPath.includes('INSERTID')) {
|
|
|
|
statsPath = statsPath.replace(/INSERTID/gi, randomUUID());
|
2023-09-25 12:00:09 +00:00
|
|
|
}
|
|
|
|
|
2022-09-08 10:32:39 +00:00
|
|
|
const platform = zPlatform.parse(os.platform());
|
|
|
|
|
|
|
|
const expectations = readJSON(
|
|
|
|
path.join(process.cwd(), 'test', 'TestExpectations.json')
|
|
|
|
) as TestExpectation[];
|
|
|
|
|
|
|
|
const parsedSuitesFile = zTestSuiteFile.parse(
|
|
|
|
readJSON(path.join(process.cwd(), 'test', 'TestSuites.json'))
|
|
|
|
);
|
|
|
|
|
|
|
|
const applicableSuites = getApplicableTestSuites(parsedSuitesFile, platform);
|
|
|
|
|
|
|
|
console.log('Planning to run the following test suites', applicableSuites);
|
2023-09-27 14:02:55 +00:00
|
|
|
if (statsPath) {
|
|
|
|
console.log('Test stats will be saved to', statsPath);
|
2022-10-19 13:21:18 +00:00
|
|
|
}
|
2022-09-08 10:32:39 +00:00
|
|
|
|
|
|
|
let fail = false;
|
2023-03-20 09:59:50 +00:00
|
|
|
const recommendations: RecommendedExpectation[] = [];
|
2022-09-08 10:32:39 +00:00
|
|
|
try {
|
|
|
|
for (const suite of applicableSuites) {
|
|
|
|
const parameters = suite.parameters;
|
|
|
|
|
|
|
|
const applicableExpectations = filterByParameters(
|
|
|
|
filterByPlatform(expectations, platform),
|
|
|
|
parameters
|
2023-03-20 09:59:50 +00:00
|
|
|
).reverse();
|
2022-09-08 10:32:39 +00:00
|
|
|
|
2023-03-01 11:30:42 +00:00
|
|
|
// Add more logging when the GitHub Action Debugging option is set
|
|
|
|
// https://docs.github.com/en/actions/learn-github-actions/variables#default-environment-variables
|
|
|
|
const githubActionDebugging = process.env['RUNNER_DEBUG']
|
|
|
|
? {
|
|
|
|
DEBUG: 'puppeteer:*',
|
2023-03-07 16:00:05 +00:00
|
|
|
EXTRA_LAUNCH_OPTIONS: JSON.stringify({
|
2023-05-15 14:39:47 +00:00
|
|
|
dumpio: true,
|
2023-03-01 11:30:42 +00:00
|
|
|
extraPrefsFirefox: {
|
|
|
|
'remote.log.level': 'Trace',
|
|
|
|
},
|
2023-03-07 16:00:05 +00:00
|
|
|
}),
|
2023-03-01 11:30:42 +00:00
|
|
|
}
|
|
|
|
: {};
|
|
|
|
|
2022-09-08 10:32:39 +00:00
|
|
|
const env = extendProcessEnv([
|
|
|
|
...parameters.map(param => {
|
2023-02-28 11:55:20 +00:00
|
|
|
return parsedSuitesFile.parameterDefinitions[param];
|
2022-09-08 10:32:39 +00:00
|
|
|
}),
|
|
|
|
{
|
|
|
|
PUPPETEER_SKIPPED_TEST_CONFIG: JSON.stringify(
|
2022-09-09 09:12:18 +00:00
|
|
|
applicableExpectations.map(ex => {
|
|
|
|
return {
|
|
|
|
testIdPattern: ex.testIdPattern,
|
|
|
|
skip: ex.expectations.includes('SKIP'),
|
|
|
|
};
|
2022-09-08 10:32:39 +00:00
|
|
|
})
|
|
|
|
),
|
|
|
|
},
|
2023-03-01 11:30:42 +00:00
|
|
|
githubActionDebugging,
|
2022-09-08 10:32:39 +00:00
|
|
|
]);
|
|
|
|
|
|
|
|
const tmpDir = fs.mkdtempSync(
|
|
|
|
path.join(os.tmpdir(), 'puppeteer-test-runner-')
|
|
|
|
);
|
2023-09-27 14:02:55 +00:00
|
|
|
const tmpFilename = statsPath
|
|
|
|
? statsPath
|
2022-10-19 13:21:18 +00:00
|
|
|
: path.join(tmpDir, 'output.json');
|
2022-09-08 10:32:39 +00:00
|
|
|
console.log('Running', JSON.stringify(parameters), tmpFilename);
|
2022-09-15 05:59:11 +00:00
|
|
|
const args = [
|
|
|
|
'-u',
|
|
|
|
path.join(__dirname, 'interface.js'),
|
|
|
|
'-R',
|
2023-09-27 14:02:55 +00:00
|
|
|
!reporter ? path.join(__dirname, 'reporter.js') : reporter,
|
2022-09-15 05:59:11 +00:00
|
|
|
'-O',
|
2023-09-27 14:02:55 +00:00
|
|
|
`output=${tmpFilename}`,
|
2022-09-15 05:59:11 +00:00
|
|
|
];
|
2023-09-26 08:13:22 +00:00
|
|
|
|
|
|
|
const specPattern = 'test/build/**/*.spec.js';
|
|
|
|
const specs = globSync(specPattern, {
|
2023-09-27 14:02:55 +00:00
|
|
|
ignore: !includeCdpTests ? 'test/build/cdp/**/*.spec.js' : undefined,
|
2023-09-26 08:13:22 +00:00
|
|
|
}).sort((a, b) => {
|
|
|
|
return a.localeCompare(b);
|
|
|
|
});
|
2023-09-25 12:00:09 +00:00
|
|
|
if (shard) {
|
|
|
|
// Shard ID is 1-based.
|
|
|
|
const [shardId, shards] = shard.split('/').map(s => {
|
|
|
|
return Number(s);
|
|
|
|
}) as [number, number];
|
|
|
|
const argsLength = args.length;
|
|
|
|
for (let i = 0; i < specs.length; i++) {
|
|
|
|
if (i % shards === shardId - 1) {
|
|
|
|
args.push(specs[i]!);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (argsLength === args.length) {
|
|
|
|
throw new Error('Shard did not result in any test files');
|
|
|
|
}
|
|
|
|
console.log(
|
|
|
|
`Running shard ${shardId}/${shards}. Picked ${
|
|
|
|
args.length - argsLength
|
|
|
|
} files out of ${specs.length}.`
|
|
|
|
);
|
2023-09-26 08:13:22 +00:00
|
|
|
} else {
|
|
|
|
args.push(...specs);
|
2023-01-13 09:51:59 +00:00
|
|
|
}
|
2023-09-27 14:02:55 +00:00
|
|
|
const handle = spawn(
|
|
|
|
'npx',
|
|
|
|
[
|
|
|
|
...(useCoverage
|
|
|
|
? [
|
|
|
|
'c8',
|
|
|
|
'--check-coverage',
|
|
|
|
'--lines',
|
|
|
|
String(suite.expectedLineCoverage),
|
|
|
|
'npx',
|
|
|
|
]
|
|
|
|
: []),
|
|
|
|
'mocha',
|
|
|
|
...mochaArgs.map(String),
|
|
|
|
...args,
|
|
|
|
],
|
|
|
|
{
|
|
|
|
shell: true,
|
|
|
|
cwd: process.cwd(),
|
|
|
|
stdio: 'inherit',
|
|
|
|
env,
|
|
|
|
}
|
|
|
|
);
|
2022-09-08 10:32:39 +00:00
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
|
|
handle.on('error', err => {
|
|
|
|
reject(err);
|
|
|
|
});
|
|
|
|
handle.on('close', () => {
|
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
console.log('Finished', JSON.stringify(parameters));
|
|
|
|
try {
|
|
|
|
const results = readJSON(tmpFilename) as MochaResults;
|
2023-03-20 09:59:50 +00:00
|
|
|
const updates = getExpectationUpdates(results, applicableExpectations, {
|
|
|
|
platforms: [os.platform()],
|
|
|
|
parameters,
|
|
|
|
});
|
2023-06-20 11:56:11 +00:00
|
|
|
const totalTests = results.stats.tests;
|
2023-03-24 14:31:47 +00:00
|
|
|
results.parameters = parameters;
|
|
|
|
results.platform = platform;
|
|
|
|
results.date = new Date().toISOString();
|
2023-03-20 09:59:50 +00:00
|
|
|
if (updates.length > 0) {
|
2022-09-08 10:32:39 +00:00
|
|
|
fail = true;
|
2023-03-20 09:59:50 +00:00
|
|
|
recommendations.push(...updates);
|
2023-03-24 14:31:47 +00:00
|
|
|
results.updates = updates;
|
|
|
|
writeJSON(tmpFilename, results);
|
2022-09-08 10:32:39 +00:00
|
|
|
} else {
|
2023-09-25 12:00:09 +00:00
|
|
|
if (!shard && totalTests < minTests) {
|
2023-06-20 11:56:11 +00:00
|
|
|
fail = true;
|
|
|
|
console.log(
|
|
|
|
`Test run matches expectations but the number of discovered tests is too low (expected: ${minTests}, actual: ${totalTests}).`
|
|
|
|
);
|
|
|
|
writeJSON(tmpFilename, results);
|
|
|
|
continue;
|
|
|
|
}
|
2022-12-07 13:54:00 +00:00
|
|
|
console.log('Test run matches expectations');
|
2023-03-24 14:31:47 +00:00
|
|
|
writeJSON(tmpFilename, results);
|
2022-09-08 10:32:39 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
fail = true;
|
|
|
|
console.error(err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
fail = true;
|
|
|
|
console.error(err);
|
|
|
|
} finally {
|
2023-09-27 14:02:55 +00:00
|
|
|
if (!!provideSuggestions) {
|
2023-02-28 11:55:20 +00:00
|
|
|
printSuggestions(
|
|
|
|
recommendations,
|
|
|
|
'add',
|
|
|
|
'Add the following to TestExpectations.json to ignore the error:'
|
|
|
|
);
|
|
|
|
printSuggestions(
|
|
|
|
recommendations,
|
|
|
|
'remove',
|
|
|
|
'Remove the following from the TestExpectations.json to ignore the error:'
|
|
|
|
);
|
|
|
|
printSuggestions(
|
|
|
|
recommendations,
|
|
|
|
'update',
|
|
|
|
'Update the following expectations in the TestExpectations.json to ignore the error:'
|
|
|
|
);
|
2022-09-08 10:32:39 +00:00
|
|
|
}
|
|
|
|
process.exit(fail ? 1 : 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
main().catch(error => {
|
|
|
|
console.error(error);
|
|
|
|
process.exit(1);
|
|
|
|
});
|