mirror of
https://git.tuxpa.in/a/code-server.git
synced 2025-01-02 15:28:45 +00:00
142 lines
5.2 KiB
TypeScript
142 lines
5.2 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';
|
|
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 optimistLib from 'optimist';
|
|
|
|
const optimist = optimistLib
|
|
.describe('workspacePath', 'path to the workspace to open in the test').string('workspacePath')
|
|
.describe('extensionDevelopmentPath', 'path to the extension to test').string('extensionDevelopmentPath')
|
|
.describe('extensionTestsPath', 'path to the extension tests').string('extensionTestsPath')
|
|
.describe('debug', 'do not run browsers headless').boolean('debug')
|
|
.describe('browser', 'browser in which integration tests should run').string('browser').default('browser', 'chromium')
|
|
.describe('help', 'show the help').alias('help', 'h');
|
|
|
|
if (optimist.argv.help) {
|
|
optimist.showHelp();
|
|
process.exit(0);
|
|
}
|
|
|
|
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 args = process.platform === 'linux' && browserType === 'chromium' ? ['--no-sandbox'] : undefined; // disable sandbox to run chrome on certain Linux distros
|
|
const browser = await playwright[browserType].launch({ headless: !Boolean(optimist.argv.debug), args });
|
|
const context = await browser.newContext();
|
|
const page = await context.newPage();
|
|
await page.setViewportSize({ width, height });
|
|
|
|
const host = endpoint.host;
|
|
const protocol = 'vscode-remote';
|
|
|
|
const testWorkspaceUri = url.format({ pathname: URI.file(path.resolve(optimist.argv.workspacePath)).path, protocol, host, slashes: true });
|
|
const testExtensionUri = url.format({ pathname: URI.file(path.resolve(optimist.argv.extensionDevelopmentPath)).path, protocol, host, slashes: true });
|
|
const testFilesUri = url.format({ pathname: URI.file(path.resolve(optimist.argv.extensionTestsPath)).path, protocol, host, slashes: true });
|
|
|
|
const folderParam = testWorkspaceUri;
|
|
const payloadParam = `[["extensionDevelopmentPath","${testExtensionUri}"],["extensionTestsPath","${testFilesUri}"],["enableProposedApi",""]]`;
|
|
|
|
await page.goto(`${endpoint.href}&folder=${folderParam}&payload=${payloadParam}`);
|
|
|
|
await page.exposeFunction('codeAutomationLog', (type: string, args: any[]) => {
|
|
console[type](...args);
|
|
});
|
|
|
|
page.on('console', async (msg: playwright.ConsoleMessage) => {
|
|
const msgText = msg.text();
|
|
if (msgText.indexOf('vscode:exit') >= 0) {
|
|
try {
|
|
await browser.close();
|
|
} catch (error) {
|
|
console.error(`Error when closing browser: ${error}`);
|
|
}
|
|
|
|
try {
|
|
await pkill(server.pid);
|
|
} catch (error) {
|
|
console.error(`Error when killing server process tree: ${error}`);
|
|
}
|
|
|
|
process.exit(msgText === 'vscode:exit 0' ? 0 : 1);
|
|
}
|
|
});
|
|
}
|
|
|
|
function pkill(pid: number): Promise<void> {
|
|
return new Promise((c, e) => {
|
|
kill(pid, error => error ? e(error) : c());
|
|
});
|
|
}
|
|
|
|
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_AGENT_FOLDER: userDataDir,
|
|
VSCODE_BROWSER: browserType,
|
|
...process.env
|
|
};
|
|
|
|
let serverLocation: string;
|
|
if (process.env.VSCODE_REMOTE_SERVER_PATH) {
|
|
serverLocation = path.join(process.env.VSCODE_REMOTE_SERVER_PATH, `server.${process.platform === 'win32' ? 'cmd' : 'sh'}`);
|
|
|
|
console.log(`Starting built server from '${serverLocation}'`);
|
|
} else {
|
|
serverLocation = path.join(__dirname, '..', '..', '..', '..', `resources/server/web.${process.platform === 'win32' ? 'bat' : 'sh'}`);
|
|
process.env.VSCODE_DEV = '1';
|
|
|
|
console.log(`Starting server out of sources from '${serverLocation}'`);
|
|
}
|
|
|
|
let serverProcess = cp.spawn(
|
|
serverLocation,
|
|
['--browser', 'none', '--driver', 'web', '--enable-proposed-api'],
|
|
{ env }
|
|
);
|
|
|
|
serverProcess?.stderr?.on('data', error => console.log(`Server stderr: ${error}`));
|
|
|
|
if (optimist.argv.debug) {
|
|
serverProcess?.stdout?.on('data', data => console.log(`Server stdout: ${data}`));
|
|
}
|
|
|
|
process.on('exit', () => serverProcess.kill());
|
|
process.on('SIGINT', () => serverProcess.kill());
|
|
process.on('SIGTERM', () => serverProcess.kill());
|
|
|
|
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(optimist.argv.browser).then(async ({ endpoint, server }) => {
|
|
return runTestsInBrowser(optimist.argv.browser, endpoint, server);
|
|
}, error => {
|
|
console.error(error);
|
|
process.exit(1);
|
|
});
|