signal-desktop/ts/test-mock/benchmarks/group_send_bench.ts

260 lines
6.8 KiB
TypeScript
Raw Normal View History

// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
/* eslint-disable no-await-in-loop, no-console */
import assert from 'assert';
import {
StorageState,
EnvelopeType,
ReceiptType,
} from '@signalapp/mock-server';
import {
Bootstrap,
debug,
RUN_COUNT,
GROUP_SIZE,
2023-12-13 19:47:51 +00:00
CONVERSATION_SIZE,
DISCARD_COUNT,
2023-12-13 19:47:51 +00:00
GROUP_DELIVERY_RECEIPTS,
2024-09-06 17:52:19 +00:00
BLOCKED_COUNT,
} from './fixtures';
import { stats } from '../../util/benchmark/stats';
2023-12-13 19:47:51 +00:00
import { sleep } from '../../util/sleep';
2024-04-03 17:17:39 +00:00
import { typeIntoInput } from '../helpers';
import { MINUTE } from '../../util/durations';
const LAST_MESSAGE = 'start sending messages now';
2023-03-13 23:41:47 +00:00
Bootstrap.benchmark(async (bootstrap: Bootstrap): Promise<void> => {
const { contacts, phone } = bootstrap;
const members = [...contacts].slice(0, GROUP_SIZE);
const GROUP_NAME = 'Mock Group';
const group = await phone.createGroup({
title: GROUP_NAME,
members: [phone, ...members],
});
await phone.setStorageState(
StorageState.getEmpty()
.addGroup(group, { whitelisted: true })
.pinGroup(group)
);
2023-03-13 23:41:47 +00:00
const app = await bootstrap.link();
2022-07-08 20:46:25 +00:00
2023-03-13 23:41:47 +00:00
const { server, desktop } = bootstrap;
const [first] = members;
2023-03-13 23:41:47 +00:00
const messages = new Array<Buffer>();
debug('encrypting');
// Fill left pane
2023-12-13 19:47:51 +00:00
for (const contact of members.slice(0, CONVERSATION_SIZE).reverse()) {
2023-03-13 23:41:47 +00:00
const messageTimestamp = bootstrap.getTimestamp();
2023-03-13 23:41:47 +00:00
messages.push(
await contact.encryptText(desktop, `hello from: ${contact.profileName}`, {
timestamp: messageTimestamp,
sealed: true,
})
);
messages.push(
await phone.encryptSyncRead(desktop, {
timestamp: bootstrap.getTimestamp(),
messages: [
{
2023-08-16 20:54:39 +00:00
senderAci: contact.device.aci,
timestamp: messageTimestamp,
2023-03-13 23:41:47 +00:00
},
],
})
);
}
2024-09-06 17:52:19 +00:00
assert.ok(
BLOCKED_COUNT < members.length - 1,
'Must block fewer members than are in the group'
);
const unblockedMembers = members.slice(0, members.length - BLOCKED_COUNT);
const blockedMembers = members.slice(members.length - BLOCKED_COUNT);
if (blockedMembers.length > 0) {
let state = await phone.expectStorageState('blocking');
for (const member of blockedMembers) {
state = state.addContact(member, {
blocked: true,
});
}
await phone.setStorageState(state);
}
2023-03-13 23:41:47 +00:00
// Fill group
for (let i = 0; i < CONVERSATION_SIZE; i += 1) {
2024-09-06 17:52:19 +00:00
const contact = unblockedMembers[i % unblockedMembers.length];
2023-03-13 23:41:47 +00:00
const messageTimestamp = bootstrap.getTimestamp();
const isLast = i === CONVERSATION_SIZE - 1;
messages.push(
await contact.encryptText(
desktop,
isLast ? LAST_MESSAGE : `#${i} from: ${contact.profileName}`,
{
timestamp: messageTimestamp,
sealed: true,
group,
}
)
);
2024-09-06 17:52:19 +00:00
// Last message should trigger an unread indicator
if (!isLast) {
messages.push(
await phone.encryptSyncRead(desktop, {
timestamp: bootstrap.getTimestamp(),
messages: [
{
senderAci: contact.device.aci,
timestamp: messageTimestamp,
},
],
})
);
}
2023-03-13 23:41:47 +00:00
}
debug('encrypted');
2023-03-13 23:41:47 +00:00
await Promise.all(messages.map(message => server.send(desktop, message)));
2023-03-13 23:41:47 +00:00
const window = await app.getWindow();
2023-03-13 23:41:47 +00:00
debug('opening conversation');
{
const leftPane = window.locator('#LeftPane');
2023-03-13 23:41:47 +00:00
const item = leftPane
.locator(
2024-09-06 17:52:19 +00:00
`.module-conversation-list__item--contact-or-conversation[data-testid="${group.id}"]`
2023-03-13 23:41:47 +00:00
)
.first();
2024-09-06 17:52:19 +00:00
// Wait for unread indicator to give desktop time to process messages without
// the timeline open
await item
.locator(
'.module-conversation-list__item--contact-or-conversation__content'
)
.locator(
'.module-conversation-list__item--contact-or-conversation__unread-indicator'
)
.first()
.waitFor();
await item.click();
}
2024-09-06 17:52:19 +00:00
debug('scrolling to bottom of timeline');
await window
.locator('.module-timeline__messages__at-bottom-detector')
.scrollIntoViewIfNeeded();
debug('finding message in timeline');
{
const item = window
.locator(`.module-message >> text="${LAST_MESSAGE}"`)
.first();
await item.click({ timeout: MINUTE });
2023-03-13 23:41:47 +00:00
}
const deltaList = new Array<number>();
2023-12-13 19:47:51 +00:00
const input = await app.waitForEnabledComposer();
function sendReceiptsInBatches({
receipts,
batchSize,
nextBatchSize,
runId,
delay,
}: {
receipts: Array<Buffer>;
batchSize: number;
nextBatchSize: number;
runId: number;
delay: number;
}) {
const receiptsToSend = receipts.splice(0, batchSize);
debug(`sending ${receiptsToSend.length} receipts for runId ${runId}`);
receiptsToSend.forEach(delivery => server.send(desktop, delivery));
if (receipts.length) {
setTimeout(
() =>
sendReceiptsInBatches({
receipts,
batchSize: nextBatchSize,
nextBatchSize,
runId,
delay,
}),
delay
);
}
}
let receiptsFromPreviousMessage: Array<Buffer> = [];
2023-03-13 23:41:47 +00:00
for (let runId = 0; runId < RUN_COUNT + DISCARD_COUNT; runId += 1) {
2023-12-13 19:47:51 +00:00
debug(`sending previous ${receiptsFromPreviousMessage.length} receipts`);
// deliver up to 256 receipts at once (max that server will send) and then in chunks
// of 30 every 200ms to approximate real behavior as we acknowledge each batch
sendReceiptsInBatches({
receipts: receiptsFromPreviousMessage,
batchSize: 256,
nextBatchSize: 30,
delay: 100,
runId,
});
2023-03-13 23:41:47 +00:00
debug('entering message text');
2024-04-03 17:17:39 +00:00
await typeIntoInput(input, `my message ${runId}`);
2023-03-13 23:41:47 +00:00
await input.press('Enter');
debug('waiting for message on server side');
const { body, source, envelopeType } = await first.waitForMessage();
assert.strictEqual(body, `my message ${runId}`);
assert.strictEqual(source, desktop);
assert.strictEqual(envelopeType, EnvelopeType.SenderKey);
2023-03-13 23:41:47 +00:00
debug('waiting for timing from the app');
const { timestamp, delta } = await app.waitForMessageSend();
2024-02-27 16:01:25 +00:00
if (GROUP_DELIVERY_RECEIPTS > 1) {
// Sleep to allow any receipts from previous rounds to be processed
await sleep(1000);
}
2023-03-13 23:41:47 +00:00
2023-12-13 19:47:51 +00:00
debug('sending delivery receipts');
receiptsFromPreviousMessage = await Promise.all(
members.slice(0, GROUP_DELIVERY_RECEIPTS).map(member =>
member.encryptReceipt(desktop, {
timestamp: timestamp + 1,
messageTimestamps: [timestamp],
type: ReceiptType.Delivery,
})
)
);
2023-03-13 23:41:47 +00:00
if (runId >= DISCARD_COUNT) {
deltaList.push(delta);
console.log('run=%d info=%j', runId - DISCARD_COUNT, { delta });
} else {
console.log('discarded=%d info=%j', runId, { delta });
}
}
2023-03-13 23:41:47 +00:00
console.log('stats info=%j', { delta: stats(deltaList, [99, 99.8]) });
});