mirror of
https://github.com/microsoft/vscode.git
synced 2026-04-22 09:38:38 +01:00
* move build dependencies to devDependencies * build: bump tree-sitter * dependencies cleanup * remove leftover file
229 lines
7.7 KiB
TypeScript
229 lines
7.7 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import * as path from 'path';
|
|
import * as cp from 'child_process';
|
|
import * as playwright from '@playwright/test';
|
|
import * as url from 'url';
|
|
import * as tmp from 'tmp';
|
|
import * as rimraf from 'rimraf';
|
|
import { URI } from 'vscode-uri';
|
|
import * as kill from 'tree-kill';
|
|
import * as minimist from 'minimist';
|
|
import { promisify } from 'util';
|
|
import { promises } from 'fs';
|
|
|
|
const root = path.join(__dirname, '..', '..', '..', '..');
|
|
const logsPath = path.join(root, '.build', 'logs', 'integration-tests-browser');
|
|
|
|
const args = minimist(process.argv.slice(2), {
|
|
string: [
|
|
// path to the workspace (folder or *.code-workspace file) to open in the test
|
|
'workspacePath',
|
|
// path to the extension to test
|
|
'extensionDevelopmentPath',
|
|
// path to the extension tests
|
|
'extensionTestsPath',
|
|
// browser in which integration tests should run
|
|
'browser',
|
|
],
|
|
boolean: [
|
|
'help',
|
|
// do not run browsers headless
|
|
'debug',
|
|
],
|
|
alias: {
|
|
h: 'help'
|
|
},
|
|
default: {
|
|
'browser': 'chromium'
|
|
}
|
|
});
|
|
|
|
if (args.help) {
|
|
console.error(`Integration test runner for VS Code in the browser
|
|
Usage: node integration-tests-browser/out/index.js [options]
|
|
|
|
--workspacePath <path> Path to the workspace (folder or *.code-workspace file) to open in the test
|
|
--extensionDevelopmentPath <path> Path to the extension to test
|
|
--extensionTestsPath <path> Path to the extension tests
|
|
--browser <browser> Browser in which integration tests should run
|
|
--debug Do not run browsers headless
|
|
--help Print this help message
|
|
`);
|
|
|
|
process.exit(1);
|
|
}
|
|
|
|
const width = 1200;
|
|
const height = 800;
|
|
|
|
type BrowserType = 'chromium' | 'firefox' | 'webkit';
|
|
|
|
async function runTestsInBrowser(browserType: BrowserType, endpoint: url.UrlWithStringQuery, server: cp.ChildProcess): Promise<void> {
|
|
const browser = await playwright[browserType].launch({ headless: !Boolean(args.debug) });
|
|
const context = await browser.newContext();
|
|
|
|
const page = await context.newPage();
|
|
await page.setViewportSize({ width, height });
|
|
|
|
page.on('pageerror', async error => console.error(`Playwright ERROR: page error: ${error}`));
|
|
page.on('crash', page => console.error('Playwright ERROR: page crash'));
|
|
page.on('response', async response => {
|
|
if (response.status() >= 400) {
|
|
console.error(`Playwright ERROR: HTTP status ${response.status()} for ${response.url()}`);
|
|
}
|
|
});
|
|
page.on('console', async msg => {
|
|
try {
|
|
if (msg.type() === 'error' || msg.type() === 'warning') {
|
|
consoleLogFn(msg)(msg.text(), await Promise.all(msg.args().map(async arg => await arg.jsonValue())));
|
|
}
|
|
} catch (err) {
|
|
console.error('Error logging console', err);
|
|
}
|
|
});
|
|
page.on('requestfailed', e => {
|
|
console.error('Request Failed', e.url(), e.failure()?.errorText);
|
|
});
|
|
|
|
await page.exposeFunction('codeAutomationLog', (type: string, args: any[]) => {
|
|
console[type](...args);
|
|
});
|
|
|
|
await page.exposeFunction('codeAutomationExit', async (code: number, logs: Array<{ readonly relativePath: string; readonly contents: string }>) => {
|
|
try {
|
|
for (const log of logs) {
|
|
const absoluteLogsPath = path.join(logsPath, log.relativePath);
|
|
|
|
await promises.mkdir(path.dirname(absoluteLogsPath), { recursive: true });
|
|
await promises.writeFile(absoluteLogsPath, log.contents);
|
|
}
|
|
} catch (error) {
|
|
console.error(`Error saving web client logs (${error})`);
|
|
}
|
|
|
|
try {
|
|
await browser.close();
|
|
} catch (error) {
|
|
console.error(`Error when closing browser: ${error}`);
|
|
}
|
|
|
|
try {
|
|
await promisify(kill)(server.pid!);
|
|
} catch (error) {
|
|
console.error(`Error when killing server process tree (pid: ${server.pid}): ${error}`);
|
|
}
|
|
|
|
process.exit(code);
|
|
});
|
|
|
|
const host = endpoint.host;
|
|
const protocol = 'vscode-remote';
|
|
|
|
const testWorkspacePath = URI.file(path.resolve(args.workspacePath)).path;
|
|
const testExtensionUri = url.format({ pathname: URI.file(path.resolve(args.extensionDevelopmentPath)).path, protocol, host, slashes: true });
|
|
const testFilesUri = url.format({ pathname: URI.file(path.resolve(args.extensionTestsPath)).path, protocol, host, slashes: true });
|
|
|
|
const payloadParam = `[["extensionDevelopmentPath","${testExtensionUri}"],["extensionTestsPath","${testFilesUri}"],["enableProposedApi",""],["webviewExternalEndpointCommit","ef65ac1ba57f57f2a3961bfe94aa20481caca4c6"],["skipWelcome","true"]]`;
|
|
|
|
if (path.extname(testWorkspacePath) === '.code-workspace') {
|
|
await page.goto(`${endpoint.href}&workspace=${testWorkspacePath}&payload=${payloadParam}`);
|
|
} else {
|
|
await page.goto(`${endpoint.href}&folder=${testWorkspacePath}&payload=${payloadParam}`);
|
|
}
|
|
}
|
|
|
|
function consoleLogFn(msg: playwright.ConsoleMessage) {
|
|
const type = msg.type();
|
|
const candidate = console[type];
|
|
if (candidate) {
|
|
return candidate;
|
|
}
|
|
|
|
if (type === 'warning') {
|
|
return console.warn;
|
|
}
|
|
|
|
return console.log;
|
|
}
|
|
|
|
async function launchServer(browserType: BrowserType): Promise<{ endpoint: url.UrlWithStringQuery; server: cp.ChildProcess }> {
|
|
|
|
// Ensure a tmp user-data-dir is used for the tests
|
|
const tmpDir = tmp.dirSync({ prefix: 't' });
|
|
const testDataPath = tmpDir.name;
|
|
process.once('exit', () => rimraf.sync(testDataPath));
|
|
|
|
const userDataDir = path.join(testDataPath, 'd');
|
|
|
|
const env = {
|
|
VSCODE_BROWSER: browserType,
|
|
...process.env
|
|
};
|
|
|
|
const serverArgs = ['--enable-proposed-api', '--disable-telemetry', '--server-data-dir', userDataDir, '--accept-server-license-terms', '--disable-workspace-trust'];
|
|
|
|
let serverLocation: string;
|
|
if (process.env.VSCODE_REMOTE_SERVER_PATH) {
|
|
const { serverApplicationName } = require(path.join(process.env.VSCODE_REMOTE_SERVER_PATH, 'product.json'));
|
|
serverLocation = path.join(process.env.VSCODE_REMOTE_SERVER_PATH, 'bin', `${serverApplicationName}${process.platform === 'win32' ? '.cmd' : ''}`);
|
|
|
|
if (args.debug) {
|
|
console.log(`Starting built server from '${serverLocation}'`);
|
|
}
|
|
} else {
|
|
serverLocation = path.join(root, `scripts/code-server.${process.platform === 'win32' ? 'bat' : 'sh'}`);
|
|
process.env.VSCODE_DEV = '1';
|
|
|
|
if (args.debug) {
|
|
console.log(`Starting server out of sources from '${serverLocation}'`);
|
|
}
|
|
}
|
|
|
|
const serverLogsPath = path.join(logsPath, 'server');
|
|
console.log(`Storing log files into '${serverLogsPath}'`);
|
|
serverArgs.push('--logsPath', serverLogsPath);
|
|
|
|
const stdio: cp.StdioOptions = args.debug ? 'pipe' : ['ignore', 'pipe', 'ignore'];
|
|
|
|
const serverProcess = cp.spawn(
|
|
serverLocation,
|
|
serverArgs,
|
|
{ env, stdio }
|
|
);
|
|
|
|
if (args.debug) {
|
|
serverProcess.stderr!.on('data', error => console.log(`Server stderr: ${error}`));
|
|
serverProcess.stdout!.on('data', data => console.log(`Server stdout: ${data}`));
|
|
}
|
|
|
|
process.on('exit', () => serverProcess.kill());
|
|
process.on('SIGINT', () => {
|
|
serverProcess.kill();
|
|
process.exit(128 + 2); // https://nodejs.org/docs/v14.16.0/api/process.html#process_signal_events
|
|
});
|
|
process.on('SIGTERM', () => {
|
|
serverProcess.kill();
|
|
process.exit(128 + 15); // https://nodejs.org/docs/v14.16.0/api/process.html#process_signal_events
|
|
});
|
|
|
|
return new Promise(c => {
|
|
serverProcess.stdout!.on('data', data => {
|
|
const matches = data.toString('ascii').match(/Web UI available at (.+)/);
|
|
if (matches !== null) {
|
|
c({ endpoint: url.parse(matches[1]), server: serverProcess });
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
launchServer(args.browser).then(async ({ endpoint, server }) => {
|
|
return runTestsInBrowser(args.browser, endpoint, server);
|
|
}, error => {
|
|
console.error(error);
|
|
process.exit(1);
|
|
});
|