mirror of
https://github.com/microsoft/playwright.git
synced 2025-01-07 11:46:42 +03:00
chore: move "service" mode tests to experimental grid (#9531)
This patch moves service tests to use experimental grid. Test changes: - `proxy.spec.ts` - proxy setup is not supported at all in service mode. This is because we already use proxy for browsers to properly resolve localhost. - `har.spec.ts` - tests disabled since connection port is different.
This commit is contained in:
parent
e47cc5186b
commit
01c702adbb
@ -18,9 +18,8 @@ import { Fixtures, _baseTest } from '@playwright/test';
|
|||||||
import * as path from 'path';
|
import * as path from 'path';
|
||||||
import * as fs from 'fs';
|
import * as fs from 'fs';
|
||||||
import { installCoverageHooks } from './coverage';
|
import { installCoverageHooks } from './coverage';
|
||||||
import * as childProcess from 'child_process';
|
|
||||||
import { start } from 'playwright-core/lib/outofprocess';
|
import { start } from 'playwright-core/lib/outofprocess';
|
||||||
import { PlaywrightClient } from 'playwright-core/lib/remote/playwrightClient';
|
import { GridClient } from 'playwright-core/src/grid/gridClient';
|
||||||
import type { LaunchOptions } from 'playwright-core';
|
import type { LaunchOptions } from 'playwright-core';
|
||||||
import { commonFixtures, CommonFixtures, serverFixtures, ServerFixtures, ServerOptions } from './commonFixtures';
|
import { commonFixtures, CommonFixtures, serverFixtures, ServerFixtures, ServerOptions } from './commonFixtures';
|
||||||
|
|
||||||
@ -57,36 +56,15 @@ class DriverMode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class ServiceMode {
|
class ServiceMode {
|
||||||
private _client: import('playwright-core/src/remote/playwrightClient').PlaywrightClient;
|
private _gridClient: GridClient;
|
||||||
private _serviceProcess: childProcess.ChildProcess;
|
|
||||||
|
|
||||||
async setup(workerIndex: number) {
|
async setup(workerIndex: number) {
|
||||||
const port = 10507 + workerIndex;
|
this._gridClient = await GridClient.connect('http://localhost:3333');
|
||||||
this._serviceProcess = childProcess.fork(path.join(__dirname, '..', '..', 'packages', 'playwright-core', 'lib', 'cli', 'cli.js'), ['run-server', String(port)], {
|
return this._gridClient.playwright();
|
||||||
stdio: 'pipe'
|
|
||||||
});
|
|
||||||
this._serviceProcess.stderr.pipe(process.stderr);
|
|
||||||
await new Promise<void>(f => {
|
|
||||||
this._serviceProcess.stdout.on('data', data => {
|
|
||||||
if (data.toString().includes('Listening on'))
|
|
||||||
f();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
this._serviceProcess.on('exit', this._onExit);
|
|
||||||
this._client = await PlaywrightClient.connect({ wsEndpoint: `ws://localhost:${port}/ws` });
|
|
||||||
return this._client.playwright();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async teardown() {
|
async teardown() {
|
||||||
await this._client.close();
|
await this._gridClient.close();
|
||||||
this._serviceProcess.removeListener('exit', this._onExit);
|
|
||||||
const processExited = new Promise(f => this._serviceProcess.on('exit', f));
|
|
||||||
this._serviceProcess.kill();
|
|
||||||
await processExited;
|
|
||||||
}
|
|
||||||
|
|
||||||
private _onExit(exitCode: number, signal: string) {
|
|
||||||
throw new Error(`Server closed with exitCode=${exitCode} signal=${signal}`);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,6 +60,11 @@ const config: Config<CommonOptions & PlaywrightOptions> = {
|
|||||||
[ 'json', { outputFile: path.join(outputDir, 'report.json') } ],
|
[ 'json', { outputFile: path.join(outputDir, 'report.json') } ],
|
||||||
] : 'line',
|
] : 'line',
|
||||||
projects: [],
|
projects: [],
|
||||||
|
webServer: mode === 'service' ? {
|
||||||
|
command: 'npx playwright experimental-grid-server',
|
||||||
|
port: 3333,
|
||||||
|
reuseExistingServer: true,
|
||||||
|
} : undefined,
|
||||||
};
|
};
|
||||||
|
|
||||||
const browserNames = ['chromium', 'webkit', 'firefox'] as BrowserName[];
|
const browserNames = ['chromium', 'webkit', 'firefox'] as BrowserName[];
|
||||||
|
@ -402,17 +402,18 @@ it('should not contain internal pages', async ({ browserName, contextFactory, se
|
|||||||
expect(log.pages.length).toBe(1);
|
expect(log.pages.length).toBe(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should have connection details', async ({ contextFactory, server, browserName, platform }, testInfo) => {
|
it('should have connection details', async ({ contextFactory, server, browserName, platform, mode }, testInfo) => {
|
||||||
const { page, getLog } = await pageWithHar(contextFactory, testInfo);
|
const { page, getLog } = await pageWithHar(contextFactory, testInfo);
|
||||||
await page.goto(server.EMPTY_PAGE);
|
await page.goto(server.EMPTY_PAGE);
|
||||||
const log = await getLog();
|
const log = await getLog();
|
||||||
const { serverIPAddress, _serverPort: port, _securityDetails: securityDetails } = log.entries[0];
|
const { serverIPAddress, _serverPort: port, _securityDetails: securityDetails } = log.entries[0];
|
||||||
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(port).toBe(server.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(port).toBe(server.PORT);
|
||||||
expect(securityDetails).toEqual({});
|
expect(securityDetails).toEqual({});
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should have security details', async ({ contextFactory, httpsServer, browserName, platform }, testInfo) => {
|
it('should have security details', async ({ contextFactory, httpsServer, browserName, platform, mode }, testInfo) => {
|
||||||
it.fail(browserName === 'webkit' && platform === 'linux', 'https://github.com/microsoft/playwright/issues/6759');
|
it.fail(browserName === 'webkit' && platform === 'linux', 'https://github.com/microsoft/playwright/issues/6759');
|
||||||
it.fail(browserName === 'webkit' && platform === 'win32');
|
it.fail(browserName === 'webkit' && platform === 'win32');
|
||||||
|
|
||||||
@ -421,14 +422,15 @@ it('should have security details', async ({ contextFactory, httpsServer, browser
|
|||||||
const log = await getLog();
|
const log = await getLog();
|
||||||
const { serverIPAddress, _serverPort: port, _securityDetails: securityDetails } = log.entries[0];
|
const { serverIPAddress, _serverPort: port, _securityDetails: securityDetails } = log.entries[0];
|
||||||
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(port).toBe(httpsServer.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(port).toBe(httpsServer.PORT);
|
||||||
if (browserName === 'webkit' && platform === 'darwin')
|
if (browserName === 'webkit' && platform === 'darwin')
|
||||||
expect(securityDetails).toEqual({ protocol: 'TLS 1.3', subjectName: 'puppeteer-tests', validFrom: 1550084863, validTo: 33086084863 });
|
expect(securityDetails).toEqual({ protocol: 'TLS 1.3', subjectName: 'puppeteer-tests', validFrom: 1550084863, validTo: 33086084863 });
|
||||||
else
|
else
|
||||||
expect(securityDetails).toEqual({ issuer: 'puppeteer-tests', protocol: 'TLS 1.3', subjectName: 'puppeteer-tests', validFrom: 1550084863, validTo: 33086084863 });
|
expect(securityDetails).toEqual({ issuer: 'puppeteer-tests', protocol: 'TLS 1.3', subjectName: 'puppeteer-tests', validFrom: 1550084863, validTo: 33086084863 });
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should have connection details for redirects', async ({ contextFactory, server, browserName }, testInfo) => {
|
it('should have connection details for redirects', async ({ contextFactory, server, browserName, mode }, testInfo) => {
|
||||||
server.setRedirect('/foo.html', '/empty.html');
|
server.setRedirect('/foo.html', '/empty.html');
|
||||||
const { page, getLog } = await pageWithHar(contextFactory, testInfo);
|
const { page, getLog } = await pageWithHar(contextFactory, testInfo);
|
||||||
await page.goto(server.PREFIX + '/foo.html');
|
await page.goto(server.PREFIX + '/foo.html');
|
||||||
@ -442,15 +444,17 @@ it('should have connection details for redirects', async ({ contextFactory, serv
|
|||||||
expect(detailsFoo._serverPort).toBeUndefined();
|
expect(detailsFoo._serverPort).toBeUndefined();
|
||||||
} else {
|
} else {
|
||||||
expect(detailsFoo.serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(detailsFoo.serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(detailsFoo._serverPort).toBe(server.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(detailsFoo._serverPort).toBe(server.PORT);
|
||||||
}
|
}
|
||||||
|
|
||||||
const detailsEmpty = log.entries[1];
|
const detailsEmpty = log.entries[1];
|
||||||
expect(detailsEmpty.serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(detailsEmpty.serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(detailsEmpty._serverPort).toBe(server.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(detailsEmpty._serverPort).toBe(server.PORT);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should have connection details for failed requests', async ({ contextFactory, server, browserName, platform }, testInfo) => {
|
it('should have connection details for failed requests', async ({ contextFactory, server, browserName, platform, mode }, testInfo) => {
|
||||||
server.setRoute('/one-style.css', (_, res) => {
|
server.setRoute('/one-style.css', (_, res) => {
|
||||||
res.setHeader('Content-Type', 'text/css');
|
res.setHeader('Content-Type', 'text/css');
|
||||||
res.connection.destroy();
|
res.connection.destroy();
|
||||||
@ -460,14 +464,16 @@ it('should have connection details for failed requests', async ({ contextFactory
|
|||||||
const log = await getLog();
|
const log = await getLog();
|
||||||
const { serverIPAddress, _serverPort: port } = log.entries[0];
|
const { serverIPAddress, _serverPort: port } = log.entries[0];
|
||||||
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(serverIPAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(port).toBe(server.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(port).toBe(server.PORT);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return server address directly from response', async ({ page, server }) => {
|
it('should return server address directly from response', async ({ page, server, mode }) => {
|
||||||
const response = await page.goto(server.EMPTY_PAGE);
|
const response = await page.goto(server.EMPTY_PAGE);
|
||||||
const { ipAddress, port } = await response.serverAddr();
|
const { ipAddress, port } = await response.serverAddr();
|
||||||
expect(ipAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
expect(ipAddress).toMatch(/^127\.0\.0\.1|\[::1\]/);
|
||||||
expect(port).toBe(server.PORT);
|
if (mode !== 'service')
|
||||||
|
expect(port).toBe(server.PORT);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should return security details directly from response', async ({ contextFactory, httpsServer, browserName, platform }) => {
|
it('should return security details directly from response', async ({ contextFactory, httpsServer, browserName, platform }) => {
|
||||||
|
@ -289,10 +289,12 @@ it('should throw if networkidle2 is passed as an option', async ({ page, server
|
|||||||
expect(error.message).toContain(`waitUntil: expected one of (load|domcontentloaded|networkidle)`);
|
expect(error.message).toContain(`waitUntil: expected one of (load|domcontentloaded|networkidle)`);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should fail when main resources failed to load', async ({ page, browserName, isWindows }) => {
|
it('should fail when main resources failed to load', async ({ page, browserName, isWindows, mode }) => {
|
||||||
let error = null;
|
let error = null;
|
||||||
await page.goto('http://localhost:44123/non-existing-url').catch(e => error = e);
|
await page.goto('http://localhost:44123/non-existing-url').catch(e => error = e);
|
||||||
if (browserName === 'chromium')
|
if (mode === 'service')
|
||||||
|
expect(error.message).toContain('net::ERR_SOCKS_CONNECTION_FAILED');
|
||||||
|
else if (browserName === 'chromium')
|
||||||
expect(error.message).toContain('net::ERR_CONNECTION_REFUSED');
|
expect(error.message).toContain('net::ERR_CONNECTION_REFUSED');
|
||||||
else if (browserName === 'webkit' && isWindows)
|
else if (browserName === 'webkit' && isWindows)
|
||||||
expect(error.message).toContain(`Couldn\'t connect to server`);
|
expect(error.message).toContain(`Couldn\'t connect to server`);
|
||||||
|
@ -18,6 +18,8 @@ import { playwrightTest as it, expect } from './config/browserTest';
|
|||||||
import socks from 'socksv5';
|
import socks from 'socksv5';
|
||||||
import net from 'net';
|
import net from 'net';
|
||||||
|
|
||||||
|
it.skip(({ mode }) => mode === 'service');
|
||||||
|
|
||||||
it('should throw for bad server value', async ({ browserType, browserOptions }) => {
|
it('should throw for bad server value', async ({ browserType, browserOptions }) => {
|
||||||
const error = await browserType.launch({
|
const error = await browserType.launch({
|
||||||
...browserOptions,
|
...browserOptions,
|
||||||
|
Loading…
Reference in New Issue
Block a user