mirror of
https://github.com/microsoft/playwright.git
synced 2024-12-14 21:53:35 +03:00
382 lines
9.7 KiB
JavaScript
382 lines
9.7 KiB
JavaScript
/**
|
|
* 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.
|
|
*/
|
|
|
|
// @ts-check
|
|
|
|
const child_process = require('child_process');
|
|
const path = require('path');
|
|
const chokidar = require('chokidar');
|
|
const fs = require('fs');
|
|
const { workspace } = require('../workspace');
|
|
|
|
/**
|
|
* @typedef {{
|
|
* command: string,
|
|
* args: string[],
|
|
* shell: boolean,
|
|
* env?: NodeJS.ProcessEnv,
|
|
* cwd?: string,
|
|
* concurrent?: boolean,
|
|
* }} Step
|
|
*/
|
|
|
|
/**
|
|
* @typedef {{
|
|
* files: string,
|
|
* from: string,
|
|
* to: string,
|
|
* ignored?: string[],
|
|
* }} CopyFile
|
|
*/
|
|
|
|
/**
|
|
* @typedef {{
|
|
* inputs: string[],
|
|
* mustExist?: string[],
|
|
* script?: string,
|
|
* command?: string,
|
|
* args?: string[],
|
|
* cwd?: string,
|
|
* }} OnChange
|
|
*/
|
|
|
|
/** @type {Step[]} */
|
|
const steps = [];
|
|
/** @type {OnChange[]} */
|
|
const onChanges = [];
|
|
/** @type {CopyFile[]} */
|
|
const copyFiles = [];
|
|
|
|
const watchMode = process.argv.slice(2).includes('--watch');
|
|
const lintMode = process.argv.slice(2).includes('--lint');
|
|
const withSourceMaps = process.argv.slice(2).includes('--sourcemap') || watchMode;
|
|
const ROOT = path.join(__dirname, '..', '..');
|
|
|
|
/**
|
|
* @param {string} relative
|
|
* @returns {string}
|
|
*/
|
|
function filePath(relative) {
|
|
return path.join(ROOT, ...relative.split('/'));
|
|
}
|
|
|
|
/**
|
|
* @param {string} path
|
|
* @returns {string}
|
|
*/
|
|
function quotePath(path) {
|
|
return "\"" + path + "\"";
|
|
}
|
|
|
|
/**
|
|
* @param {Step} step
|
|
*/
|
|
function runStep(step) {
|
|
console.log(`==== Running ${step.command} ${step.args.join(' ')} in ${step.cwd || process.cwd()}`);
|
|
const out = child_process.spawnSync(step.command, step.args, {
|
|
stdio: 'inherit',
|
|
shell: step.shell,
|
|
env: {
|
|
...process.env,
|
|
...step.env
|
|
},
|
|
cwd: step.cwd,
|
|
});
|
|
if (out.status)
|
|
process.exit(out.status);
|
|
}
|
|
|
|
async function runWatch() {
|
|
/** @param {OnChange} onChange */
|
|
function runOnChange(onChange) {
|
|
const paths = onChange.inputs;
|
|
const mustExist = onChange.mustExist || [];
|
|
let timeout;
|
|
function callback() {
|
|
timeout = undefined;
|
|
for (const fileMustExist of mustExist) {
|
|
if (!fs.existsSync(filePath(fileMustExist)))
|
|
return;
|
|
}
|
|
if (onChange.script)
|
|
child_process.spawnSync('node', [onChange.script], { stdio: 'inherit' });
|
|
else
|
|
child_process.spawnSync(onChange.command, onChange.args || [], { stdio: 'inherit', cwd: onChange.cwd, shell: true });
|
|
}
|
|
// chokidar will report all files as added in a sync loop, throttle those.
|
|
const reschedule = () => {
|
|
if (timeout)
|
|
clearTimeout(timeout);
|
|
timeout = setTimeout(callback, 500);
|
|
};
|
|
chokidar.watch([...paths, ...mustExist, onChange.script].filter(Boolean).map(filePath)).on('all', reschedule);
|
|
callback();
|
|
}
|
|
|
|
for (const { files, from, to, ignored } of copyFiles) {
|
|
const watcher = chokidar.watch([filePath(files)], { ignored });
|
|
watcher.on('all', (event, file) => {
|
|
copyFile(file, from, to);
|
|
});
|
|
}
|
|
|
|
for (const step of steps) {
|
|
if (!step.concurrent)
|
|
runStep(step);
|
|
}
|
|
|
|
/** @type{import('child_process').ChildProcess[]} */
|
|
const spawns = [];
|
|
for (const step of steps) {
|
|
if (!step.concurrent)
|
|
continue;
|
|
spawns.push(child_process.spawn(step.command, step.args, {
|
|
stdio: 'inherit',
|
|
shell: step.shell,
|
|
env: {
|
|
...process.env,
|
|
...step.env,
|
|
},
|
|
cwd: step.cwd,
|
|
}));
|
|
}
|
|
process.on('exit', () => spawns.forEach(s => s.kill()));
|
|
for (const onChange of onChanges)
|
|
runOnChange(onChange);
|
|
}
|
|
|
|
async function runBuild() {
|
|
for (const { files, from, to, ignored } of copyFiles) {
|
|
const watcher = chokidar.watch([filePath(files)], {
|
|
ignored
|
|
});
|
|
watcher.on('add', file => {
|
|
copyFile(file, from, to);
|
|
});
|
|
await new Promise(x => watcher.once('ready', x));
|
|
watcher.close();
|
|
}
|
|
for (const step of steps)
|
|
runStep(step);
|
|
for (const onChange of onChanges) {
|
|
if (onChange.script)
|
|
runStep({ command: 'node', args: [filePath(onChange.script)], shell: false });
|
|
else
|
|
runStep({ command: onChange.command, args: onChange.args, shell: true, cwd: onChange.cwd });
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} file
|
|
* @param {string} from
|
|
* @param {string} to
|
|
*/
|
|
function copyFile(file, from, to) {
|
|
const destination = path.resolve(filePath(to), path.relative(filePath(from), file));
|
|
fs.mkdirSync(path.dirname(destination), { recursive: true });
|
|
fs.copyFileSync(file, destination);
|
|
}
|
|
|
|
const bundles = [];
|
|
for (const pkg of workspace.packages()) {
|
|
const bundlesDir = path.join(pkg.path, 'bundles');
|
|
if (!fs.existsSync(bundlesDir))
|
|
continue;
|
|
for (const bundle of fs.readdirSync(bundlesDir))
|
|
bundles.push(path.join(bundlesDir, bundle));
|
|
}
|
|
|
|
// Update test runner.
|
|
steps.push({
|
|
command: 'npm',
|
|
args: ['ci', '--save=false', '--fund=false', '--audit=false'],
|
|
shell: true,
|
|
cwd: path.join(__dirname, '..', '..', 'tests', 'playwright-test', 'stable-test-runner'),
|
|
});
|
|
|
|
// Update bundles.
|
|
for (const bundle of bundles) {
|
|
steps.push({
|
|
command: 'npm',
|
|
args: ['ci', '--save=false', '--fund=false', '--audit=false', '--omit=optional'],
|
|
shell: true,
|
|
cwd: bundle,
|
|
});
|
|
}
|
|
|
|
// Generate third party licenses for bundles.
|
|
steps.push({
|
|
command: 'node',
|
|
args: [path.resolve(__dirname, '../generate_third_party_notice.js')],
|
|
shell: true,
|
|
});
|
|
|
|
// Build injected scripts.
|
|
steps.push({
|
|
command: 'node',
|
|
args: ['utils/generate_injected.js'],
|
|
shell: true,
|
|
});
|
|
|
|
// Run Babel.
|
|
for (const pkg of workspace.packages()) {
|
|
if (!fs.existsSync(path.join(pkg.path, 'src')))
|
|
continue;
|
|
steps.push({
|
|
command: 'npx',
|
|
args: [
|
|
'babel',
|
|
...(watchMode ? ['-w'] : []),
|
|
...(withSourceMaps ? ['--source-maps'] : []),
|
|
'--extensions', '.ts',
|
|
'--out-dir', quotePath(path.join(pkg.path, 'lib')),
|
|
'--ignore', '"packages/playwright-core/src/server/injected/**/*"',
|
|
quotePath(path.join(pkg.path, 'src')),
|
|
],
|
|
shell: true,
|
|
concurrent: true,
|
|
});
|
|
}
|
|
|
|
// Build/watch bundles.
|
|
for (const bundle of bundles) {
|
|
steps.push({
|
|
command: 'npm',
|
|
args: [
|
|
'run',
|
|
watchMode ? 'watch' : 'build',
|
|
...(withSourceMaps ? ['--', '--sourcemap'] : [])
|
|
],
|
|
shell: true,
|
|
cwd: bundle,
|
|
concurrent: true,
|
|
});
|
|
}
|
|
|
|
// Build/watch web packages.
|
|
for (const webPackage of ['html-reporter', 'recorder', 'trace-viewer']) {
|
|
steps.push({
|
|
command: 'npx',
|
|
args: [
|
|
'vite',
|
|
'build',
|
|
...(watchMode ? ['--watch', '--minify=false'] : []),
|
|
...(withSourceMaps ? ['--sourcemap'] : []),
|
|
],
|
|
shell: true,
|
|
cwd: path.join(__dirname, '..', '..', 'packages', webPackage),
|
|
concurrent: true,
|
|
});
|
|
}
|
|
// Build/watch trace viewer service worker.
|
|
steps.push({
|
|
command: 'npx',
|
|
args: [
|
|
'vite',
|
|
'--config',
|
|
'vite.sw.config.ts',
|
|
'build',
|
|
...(watchMode ? ['--watch', '--minify=false'] : []),
|
|
...(withSourceMaps ? ['--sourcemap'] : []),
|
|
],
|
|
shell: true,
|
|
cwd: path.join(__dirname, '..', '..', 'packages', 'trace-viewer'),
|
|
concurrent: true,
|
|
});
|
|
|
|
|
|
// Generate injected.
|
|
onChanges.push({
|
|
inputs: [
|
|
'packages/playwright-core/src/server/injected/**',
|
|
'packages/playwright-core/src/utils/isomorphic/**',
|
|
'utils/generate_injected.js',
|
|
],
|
|
script: 'utils/generate_injected.js',
|
|
});
|
|
|
|
// Generate channels.
|
|
onChanges.push({
|
|
inputs: [
|
|
'packages/protocol/src/protocol.yml'
|
|
],
|
|
script: 'utils/generate_channels.js',
|
|
});
|
|
|
|
// Generate types.
|
|
onChanges.push({
|
|
inputs: [
|
|
'docs/src/api/',
|
|
'docs/src/test-api/',
|
|
'docs/src/test-reporter-api/',
|
|
'utils/generate_types/overrides.d.ts',
|
|
'utils/generate_types/overrides-test.d.ts',
|
|
'utils/generate_types/overrides-testReporter.d.ts',
|
|
'utils/generate_types/exported.json',
|
|
'packages/playwright-core/src/server/chromium/protocol.d.ts',
|
|
],
|
|
mustExist: [
|
|
'packages/playwright-core/lib/server/deviceDescriptors.js',
|
|
'packages/playwright-core/lib/server/deviceDescriptorsSource.json',
|
|
],
|
|
script: 'utils/generate_types/index.js',
|
|
});
|
|
|
|
// The recorder and trace viewer have an app_icon.png that needs to be copied.
|
|
copyFiles.push({
|
|
files: 'packages/playwright-core/src/server/chromium/*.png',
|
|
from: 'packages/playwright-core/src',
|
|
to: 'packages/playwright-core/lib',
|
|
});
|
|
|
|
// Babel doesn't touch JS files, so copy them manually.
|
|
// For example: diff_match_patch.js
|
|
copyFiles.push({
|
|
files: 'packages/playwright-core/src/**/*.js',
|
|
from: 'packages/playwright-core/src',
|
|
to: 'packages/playwright-core/lib',
|
|
ignored: ['**/.eslintrc.js', '**/injected/**/*']
|
|
});
|
|
|
|
// Sometimes we require JSON files that babel ignores.
|
|
// For example, deviceDescriptorsSource.json
|
|
copyFiles.push({
|
|
files: 'packages/playwright-core/src/**/*.json',
|
|
ignored: ['**/injected/**/*'],
|
|
from: 'packages/playwright-core/src',
|
|
to: 'packages/playwright-core/lib',
|
|
});
|
|
|
|
if (lintMode) {
|
|
// Run TypeScript for type checking.
|
|
steps.push({
|
|
command: 'npx',
|
|
args: ['tsc', ...(watchMode ? ['-w'] : []), '-p', quotePath(filePath('.'))],
|
|
shell: true,
|
|
concurrent: true,
|
|
});
|
|
for (const webPackage of ['html-reporter', 'recorder', 'trace-viewer']) {
|
|
steps.push({
|
|
command: 'npx',
|
|
args: ['tsc', ...(watchMode ? ['-w'] : []), '-p', quotePath(filePath(`packages/${webPackage}`))],
|
|
shell: true,
|
|
concurrent: true,
|
|
});
|
|
}
|
|
}
|
|
|
|
watchMode ? runWatch() : runBuild();
|