662 lines
17 KiB
TypeScript
662 lines
17 KiB
TypeScript
// Copyright 2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import assert from 'assert';
|
|
import fs from 'fs/promises';
|
|
import crypto from 'crypto';
|
|
import path from 'path';
|
|
import os from 'os';
|
|
import createDebug from 'debug';
|
|
import pTimeout from 'p-timeout';
|
|
import normalizePath from 'normalize-path';
|
|
import pixelmatch from 'pixelmatch';
|
|
import { PNG } from 'pngjs';
|
|
import type { Page } from 'playwright';
|
|
|
|
import type { Device, PrimaryDevice } from '@signalapp/mock-server';
|
|
import {
|
|
Server,
|
|
ServiceIdKind,
|
|
loadCertificates,
|
|
} from '@signalapp/mock-server';
|
|
import { MAX_READ_KEYS as MAX_STORAGE_READ_KEYS } from '../services/storageConstants';
|
|
import * as durations from '../util/durations';
|
|
import { drop } from '../util/drop';
|
|
import type { RendererConfigType } from '../types/RendererConfig';
|
|
import { App } from './playwright';
|
|
import { CONTACT_COUNT } from './benchmarks/fixtures';
|
|
|
|
export { App };
|
|
|
|
const debug = createDebug('mock:bootstrap');
|
|
|
|
const ELECTRON = path.join(
|
|
__dirname,
|
|
'..',
|
|
'..',
|
|
'node_modules',
|
|
'.bin',
|
|
'electron'
|
|
);
|
|
const CI_SCRIPT = path.join(__dirname, '..', '..', 'ci.js');
|
|
|
|
const CLOSE_TIMEOUT = 10 * 1000;
|
|
|
|
const CONTACT_FIRST_NAMES = [
|
|
'Alice',
|
|
'Bob',
|
|
'Charlie',
|
|
'Danielle',
|
|
'Elaine',
|
|
'Frankie',
|
|
'Grandma',
|
|
'Paul',
|
|
'Steve',
|
|
'William',
|
|
];
|
|
const CONTACT_LAST_NAMES = [
|
|
'Smith',
|
|
'Brown',
|
|
'Jones',
|
|
'Miller',
|
|
'Davis',
|
|
'Lopez',
|
|
'Gonzales',
|
|
'Singh',
|
|
'Baker',
|
|
'Farmer',
|
|
];
|
|
|
|
const CONTACT_SUFFIXES = [
|
|
'Sr.',
|
|
'Jr.',
|
|
'the 3rd',
|
|
'the 4th',
|
|
'the 5th',
|
|
'the 6th',
|
|
'the 7th',
|
|
'the 8th',
|
|
'the 9th',
|
|
'the 10th',
|
|
];
|
|
|
|
const CONTACT_NAMES = new Array<string>();
|
|
for (const firstName of CONTACT_FIRST_NAMES) {
|
|
for (const lastName of CONTACT_LAST_NAMES) {
|
|
CONTACT_NAMES.push(`${firstName} ${lastName}`);
|
|
}
|
|
}
|
|
|
|
for (const suffix of CONTACT_SUFFIXES) {
|
|
for (const firstName of CONTACT_FIRST_NAMES) {
|
|
for (const lastName of CONTACT_LAST_NAMES) {
|
|
CONTACT_NAMES.push(`${firstName} ${lastName}, ${suffix}`);
|
|
}
|
|
}
|
|
}
|
|
|
|
const MAX_CONTACTS = CONTACT_NAMES.length;
|
|
|
|
export type BootstrapOptions = Readonly<{
|
|
benchmark?: boolean;
|
|
|
|
linkedDevices?: number;
|
|
contactCount?: number;
|
|
contactsWithoutProfileKey?: number;
|
|
unknownContactCount?: number;
|
|
contactNames?: ReadonlyArray<string>;
|
|
contactPreKeyCount?: number;
|
|
}>;
|
|
|
|
type BootstrapInternalOptions = BootstrapOptions &
|
|
Readonly<{
|
|
benchmark: boolean;
|
|
linkedDevices: number;
|
|
contactCount: number;
|
|
contactsWithoutProfileKey: number;
|
|
unknownContactCount: number;
|
|
contactNames: ReadonlyArray<string>;
|
|
}>;
|
|
|
|
function sanitizePathComponent(component: string): string {
|
|
return normalizePath(component.replace(/[^a-z]+/gi, '-'));
|
|
}
|
|
|
|
//
|
|
// Bootstrap is a class that prepares mock server and desktop for running
|
|
// tests/benchmarks.
|
|
//
|
|
// In general, the usage pattern is:
|
|
//
|
|
// const bootstrap = new Bootstrap();
|
|
// await bootstrap.init();
|
|
// const app = await bootstrap.link();
|
|
// await bootstrap.teardown();
|
|
//
|
|
// Once initialized `bootstrap` variable will have following useful properties:
|
|
//
|
|
// - `server` - a mock server instance
|
|
// - `desktop` - a linked device representing currently running desktop instance
|
|
// - `phone` - a primary device representing desktop's primary
|
|
// - `contacts` - a list of primary devices for contacts that are synced over
|
|
// through contact sync
|
|
//
|
|
// `bootstrap.getTimestamp()` could be used to generate consecutive timestamp
|
|
// for sending messages.
|
|
//
|
|
// All phone numbers and uuids for all contacts and ourselves are random and not
|
|
// the same between different test runs.
|
|
//
|
|
export class Bootstrap {
|
|
public readonly server: Server;
|
|
|
|
private readonly options: BootstrapInternalOptions;
|
|
private privContacts?: ReadonlyArray<PrimaryDevice>;
|
|
private privContactsWithoutProfileKey?: ReadonlyArray<PrimaryDevice>;
|
|
private privUnknownContacts?: ReadonlyArray<PrimaryDevice>;
|
|
private privPhone?: PrimaryDevice;
|
|
private privDesktop?: Device;
|
|
private storagePath?: string;
|
|
private backupPath?: string;
|
|
private timestamp: number = Date.now() - durations.WEEK;
|
|
private lastApp?: App;
|
|
private readonly randomId = crypto.randomBytes(8).toString('hex');
|
|
|
|
constructor(options: BootstrapOptions = {}) {
|
|
this.server = new Server({
|
|
// Limit number of storage read keys for easier testing
|
|
maxStorageReadKeys: MAX_STORAGE_READ_KEYS,
|
|
});
|
|
|
|
this.options = {
|
|
linkedDevices: 5,
|
|
contactCount: CONTACT_COUNT,
|
|
contactsWithoutProfileKey: 0,
|
|
unknownContactCount: 0,
|
|
contactNames: CONTACT_NAMES,
|
|
benchmark: false,
|
|
|
|
...options,
|
|
};
|
|
|
|
const totalContactCount =
|
|
this.options.contactCount +
|
|
this.options.contactsWithoutProfileKey +
|
|
this.options.unknownContactCount;
|
|
assert(totalContactCount <= this.options.contactNames.length);
|
|
assert(totalContactCount <= MAX_CONTACTS);
|
|
}
|
|
|
|
public async init(): Promise<void> {
|
|
debug('initializing');
|
|
|
|
await this.server.listen(0);
|
|
|
|
const { port } = this.server.address();
|
|
debug('started server on port=%d', port);
|
|
|
|
const totalContactCount =
|
|
this.options.contactCount +
|
|
this.options.contactsWithoutProfileKey +
|
|
this.options.unknownContactCount;
|
|
|
|
const allContacts = await Promise.all(
|
|
this.options.contactNames
|
|
.slice(0, totalContactCount)
|
|
.map(async profileName => {
|
|
const primary = await this.server.createPrimaryDevice({
|
|
profileName,
|
|
});
|
|
|
|
for (let i = 0; i < this.options.linkedDevices; i += 1) {
|
|
// eslint-disable-next-line no-await-in-loop
|
|
await this.server.createSecondaryDevice(primary);
|
|
}
|
|
|
|
return primary;
|
|
})
|
|
);
|
|
|
|
this.privContacts = allContacts.splice(0, this.options.contactCount);
|
|
this.privContactsWithoutProfileKey = allContacts.splice(
|
|
0,
|
|
this.options.contactsWithoutProfileKey
|
|
);
|
|
this.privUnknownContacts = allContacts.splice(
|
|
0,
|
|
this.options.unknownContactCount
|
|
);
|
|
|
|
this.privPhone = await this.server.createPrimaryDevice({
|
|
profileName: 'Myself',
|
|
contacts: this.contacts,
|
|
contactsWithoutProfileKey: this.contactsWithoutProfileKey,
|
|
});
|
|
|
|
this.storagePath = await fs.mkdtemp(path.join(os.tmpdir(), 'mock-signal-'));
|
|
this.backupPath = await fs.mkdtemp(
|
|
path.join(os.tmpdir(), 'mock-signal-backup-')
|
|
);
|
|
|
|
debug('setting storage path=%j', this.storagePath);
|
|
}
|
|
|
|
public static benchmark(
|
|
fn: (bootstrap: Bootstrap) => Promise<void>,
|
|
timeout = 5 * durations.MINUTE
|
|
): void {
|
|
drop(Bootstrap.runBenchmark(fn, timeout));
|
|
}
|
|
|
|
public get logsDir(): string {
|
|
assert(
|
|
this.storagePath !== undefined,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
|
|
return path.join(this.storagePath, 'logs');
|
|
}
|
|
|
|
public getBackupPath(fileName: string): string {
|
|
assert(
|
|
this.backupPath !== undefined,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
|
|
return path.join(this.backupPath, fileName);
|
|
}
|
|
|
|
public eraseStorage(): Promise<void> {
|
|
return this.resetAppStorage();
|
|
}
|
|
|
|
private async resetAppStorage(): Promise<void> {
|
|
assert(
|
|
this.storagePath !== undefined,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
|
|
// Note that backupPath must remain unchanged!
|
|
await fs.rm(this.storagePath, { recursive: true });
|
|
this.storagePath = await fs.mkdtemp(path.join(os.tmpdir(), 'mock-signal-'));
|
|
}
|
|
|
|
public async teardown(): Promise<void> {
|
|
debug('tearing down');
|
|
|
|
await Promise.race([
|
|
Promise.all([
|
|
this.storagePath
|
|
? fs.rm(this.storagePath, { recursive: true })
|
|
: Promise.resolve(),
|
|
this.backupPath
|
|
? fs.rm(this.backupPath, { recursive: true })
|
|
: Promise.resolve(),
|
|
this.server.close(),
|
|
this.lastApp?.close(),
|
|
]),
|
|
new Promise(resolve => setTimeout(resolve, CLOSE_TIMEOUT).unref()),
|
|
]);
|
|
}
|
|
|
|
public async link(extraConfig?: Partial<RendererConfigType>): Promise<App> {
|
|
debug('linking');
|
|
|
|
const app = await this.startApp(extraConfig);
|
|
|
|
const window = await app.getWindow();
|
|
const qrCode = window.locator(
|
|
'.module-InstallScreenQrCodeNotScannedStep__qr-code__code'
|
|
);
|
|
const relinkButton = window.locator('.LeftPaneDialog__icon--relink');
|
|
await qrCode.or(relinkButton).waitFor();
|
|
if (await relinkButton.isVisible()) {
|
|
debug('unlinked, clicking left pane button');
|
|
await relinkButton.click();
|
|
await qrCode.waitFor();
|
|
}
|
|
|
|
const provision = await this.server.waitForProvision();
|
|
|
|
const provisionURL = await app.waitForProvisionURL();
|
|
|
|
this.privDesktop = await provision.complete({
|
|
provisionURL,
|
|
primaryDevice: this.phone,
|
|
});
|
|
|
|
debug('new desktop device %j', this.desktop.debugId);
|
|
|
|
const desktopKey = await this.desktop.popSingleUseKey();
|
|
await this.phone.addSingleUseKey(this.desktop, desktopKey);
|
|
|
|
for (const contact of this.allContacts) {
|
|
for (const serviceIdKind of [ServiceIdKind.ACI, ServiceIdKind.PNI]) {
|
|
// eslint-disable-next-line no-await-in-loop
|
|
const contactKey = await this.desktop.popSingleUseKey(serviceIdKind);
|
|
// eslint-disable-next-line no-await-in-loop
|
|
await contact.addSingleUseKey(this.desktop, contactKey, serviceIdKind);
|
|
}
|
|
}
|
|
|
|
await this.phone.waitForSync(this.desktop);
|
|
this.phone.resetSyncState(this.desktop);
|
|
|
|
debug('synced with %j', this.desktop.debugId);
|
|
|
|
return app;
|
|
}
|
|
|
|
public async linkAndClose(): Promise<void> {
|
|
const app = await this.link();
|
|
|
|
debug('closing the app after link');
|
|
await app.close();
|
|
}
|
|
|
|
public async startApp(
|
|
extraConfig?: Partial<RendererConfigType>
|
|
): Promise<App> {
|
|
assert(
|
|
this.storagePath !== undefined,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
|
|
debug('starting the app');
|
|
|
|
const { port } = this.server.address();
|
|
|
|
let startAttempts = 0;
|
|
const MAX_ATTEMPTS = 4;
|
|
let app: App | undefined;
|
|
while (!app) {
|
|
startAttempts += 1;
|
|
if (startAttempts > MAX_ATTEMPTS) {
|
|
throw new Error(
|
|
`App failed to start after ${MAX_ATTEMPTS} times, giving up`
|
|
);
|
|
}
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
const config = await this.generateConfig(port, extraConfig);
|
|
|
|
const startedApp = new App({
|
|
main: ELECTRON,
|
|
args: [CI_SCRIPT],
|
|
config,
|
|
});
|
|
|
|
try {
|
|
// eslint-disable-next-line no-await-in-loop
|
|
await startedApp.start();
|
|
} catch (error) {
|
|
// eslint-disable-next-line no-console
|
|
console.error(
|
|
`Failed to start the app, attempt ${startAttempts}, retrying`,
|
|
error
|
|
);
|
|
|
|
// eslint-disable-next-line no-await-in-loop
|
|
await this.resetAppStorage();
|
|
continue;
|
|
}
|
|
|
|
this.lastApp = startedApp;
|
|
startedApp.on('close', () => {
|
|
if (this.lastApp === startedApp) {
|
|
this.lastApp = undefined;
|
|
}
|
|
});
|
|
|
|
app = startedApp;
|
|
}
|
|
|
|
return app;
|
|
}
|
|
|
|
public getTimestamp(): number {
|
|
const result = this.timestamp;
|
|
this.timestamp += 1;
|
|
return result;
|
|
}
|
|
|
|
public async maybeSaveLogs(
|
|
test?: Mocha.Runnable,
|
|
app: App | undefined = this.lastApp
|
|
): Promise<void> {
|
|
const { FORCE_ARTIFACT_SAVE } = process.env;
|
|
if (test?.state !== 'passed' || FORCE_ARTIFACT_SAVE) {
|
|
await this.saveLogs(app, test?.fullTitle());
|
|
}
|
|
}
|
|
|
|
public async saveLogs(
|
|
app: App | undefined = this.lastApp,
|
|
testName?: string
|
|
): Promise<void> {
|
|
const outDir = await this.getArtifactsDir(testName);
|
|
if (outDir == null) {
|
|
return;
|
|
}
|
|
|
|
// eslint-disable-next-line no-console
|
|
console.error(`Saving logs to ${outDir}`);
|
|
|
|
const { logsDir } = this;
|
|
await fs.rename(logsDir, path.join(outDir, 'logs'));
|
|
|
|
const page = await app?.getWindow();
|
|
if (process.env.TRACING) {
|
|
await page
|
|
?.context()
|
|
.tracing.stop({ path: path.join(outDir, 'trace.zip') });
|
|
}
|
|
if (app) {
|
|
const window = await app.getWindow();
|
|
const screenshot = await window.screenshot();
|
|
await fs.writeFile(path.join(outDir, 'screenshot.png'), screenshot);
|
|
}
|
|
}
|
|
|
|
public async createScreenshotComparator(
|
|
app: App,
|
|
callback: (
|
|
page: Page,
|
|
snapshot: (name: string) => Promise<void>
|
|
) => Promise<void>,
|
|
test?: Mocha.Runnable
|
|
): Promise<(app: App) => Promise<void>> {
|
|
const snapshots = new Array<{ name: string; data: Buffer }>();
|
|
|
|
const window = await app.getWindow();
|
|
await callback(window, async (name: string) => {
|
|
debug('creating screenshot');
|
|
snapshots.push({ name, data: await window.screenshot() });
|
|
});
|
|
|
|
let index = 0;
|
|
|
|
return async (anotherApp: App): Promise<void> => {
|
|
const anotherWindow = await anotherApp.getWindow();
|
|
await callback(anotherWindow, async (name: string) => {
|
|
index += 1;
|
|
|
|
const before = snapshots.shift();
|
|
assert(before != null, 'No previous snapshot');
|
|
assert.strictEqual(before.name, name, 'Wrong snapshot order');
|
|
|
|
const after = await anotherWindow.screenshot();
|
|
|
|
const beforePng = PNG.sync.read(before.data);
|
|
const afterPng = PNG.sync.read(after);
|
|
|
|
const { width, height } = beforePng;
|
|
const diffPng = new PNG({ width, height });
|
|
|
|
const numPixels = pixelmatch(
|
|
beforePng.data,
|
|
afterPng.data,
|
|
diffPng.data,
|
|
width,
|
|
height,
|
|
{}
|
|
);
|
|
|
|
if (numPixels === 0) {
|
|
debug('no screenshot difference');
|
|
return;
|
|
}
|
|
|
|
debug('screenshot difference', numPixels);
|
|
|
|
const outDir = await this.getArtifactsDir(test?.fullTitle());
|
|
if (outDir != null) {
|
|
debug('saving screenshots and diff');
|
|
const prefix = `${index}-${sanitizePathComponent(name)}`;
|
|
await fs.writeFile(
|
|
path.join(outDir, `${prefix}-before.png`),
|
|
before.data
|
|
);
|
|
await fs.writeFile(path.join(outDir, `${prefix}-after.png`), after);
|
|
await fs.writeFile(
|
|
path.join(outDir, `${prefix}-diff.png`),
|
|
PNG.sync.write(diffPng)
|
|
);
|
|
}
|
|
|
|
assert.strictEqual(numPixels, 0, 'Expected no pixels to be different');
|
|
});
|
|
};
|
|
}
|
|
|
|
//
|
|
// Getters
|
|
//
|
|
|
|
public get phone(): PrimaryDevice {
|
|
assert(
|
|
this.privPhone,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
return this.privPhone;
|
|
}
|
|
|
|
public get desktop(): Device {
|
|
assert(
|
|
this.privDesktop,
|
|
'Bootstrap has to be linked first, see: bootstrap.link()'
|
|
);
|
|
return this.privDesktop;
|
|
}
|
|
|
|
public get contacts(): ReadonlyArray<PrimaryDevice> {
|
|
assert(
|
|
this.privContacts,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
return this.privContacts;
|
|
}
|
|
|
|
public get contactsWithoutProfileKey(): ReadonlyArray<PrimaryDevice> {
|
|
assert(
|
|
this.privContactsWithoutProfileKey,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
return this.privContactsWithoutProfileKey;
|
|
}
|
|
public get unknownContacts(): ReadonlyArray<PrimaryDevice> {
|
|
assert(
|
|
this.privUnknownContacts,
|
|
'Bootstrap has to be initialized first, see: bootstrap.init()'
|
|
);
|
|
return this.privUnknownContacts;
|
|
}
|
|
|
|
public get allContacts(): ReadonlyArray<PrimaryDevice> {
|
|
return [
|
|
...this.contacts,
|
|
...this.contactsWithoutProfileKey,
|
|
...this.unknownContacts,
|
|
];
|
|
}
|
|
|
|
//
|
|
// Private
|
|
//
|
|
|
|
private async getArtifactsDir(
|
|
testName?: string
|
|
): Promise<string | undefined> {
|
|
const { ARTIFACTS_DIR } = process.env;
|
|
if (!ARTIFACTS_DIR) {
|
|
// eslint-disable-next-line no-console
|
|
console.error(
|
|
'Not saving artifacts. Please set ARTIFACTS_DIR env variable'
|
|
);
|
|
return undefined;
|
|
}
|
|
|
|
const normalizedPath = testName
|
|
? `${this.randomId}-${sanitizePathComponent(testName)}`
|
|
: this.randomId;
|
|
|
|
const outDir = path.join(ARTIFACTS_DIR, normalizedPath);
|
|
await fs.mkdir(outDir, { recursive: true });
|
|
|
|
return outDir;
|
|
}
|
|
|
|
private static async runBenchmark(
|
|
fn: (bootstrap: Bootstrap) => Promise<void>,
|
|
timeout: number
|
|
): Promise<void> {
|
|
const bootstrap = new Bootstrap({
|
|
benchmark: true,
|
|
});
|
|
|
|
await bootstrap.init();
|
|
|
|
try {
|
|
await pTimeout(fn(bootstrap), timeout);
|
|
if (process.env.FORCE_ARTIFACT_SAVE) {
|
|
await bootstrap.saveLogs();
|
|
}
|
|
} catch (error) {
|
|
await bootstrap.saveLogs();
|
|
throw error;
|
|
} finally {
|
|
await bootstrap.teardown();
|
|
}
|
|
}
|
|
|
|
private async generateConfig(
|
|
port: number,
|
|
extraConfig?: Partial<RendererConfigType>
|
|
): Promise<string> {
|
|
const url = `https://127.0.0.1:${port}`;
|
|
return JSON.stringify({
|
|
...(await loadCertificates()),
|
|
|
|
forcePreloadBundle: this.options.benchmark,
|
|
ciMode: 'full',
|
|
|
|
buildExpiration: Date.now() + durations.MONTH,
|
|
storagePath: this.storagePath,
|
|
storageProfile: 'mock',
|
|
serverUrl: url,
|
|
storageUrl: url,
|
|
cdn: {
|
|
'0': url,
|
|
'2': url,
|
|
},
|
|
updatesEnabled: false,
|
|
|
|
directoreType: 'cdsi',
|
|
directoryCDSIUrl: url,
|
|
directoryCDSIMRENCLAVE:
|
|
'51133fecb3fa18aaf0c8f64cb763656d3272d9faaacdb26ae7df082e414fb142',
|
|
|
|
...extraConfig,
|
|
});
|
|
}
|
|
}
|