signal-desktop/ts/test-mock/storage/fixtures.ts

126 lines
3.3 KiB
TypeScript
Raw Normal View History

// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import createDebug from 'debug';
import type { Group, PrimaryDevice } from '@signalapp/mock-server';
import { StorageState, Proto } from '@signalapp/mock-server';
import { App } from '../playwright';
import { Bootstrap } from '../bootstrap';
2022-04-13 00:50:17 +00:00
import type { BootstrapOptions } from '../bootstrap';
import { MY_STORY_ID } from '../../types/Stories';
import { uuidToBytes } from '../../util/uuidToBytes';
2022-04-20 19:35:53 +00:00
export const debug = createDebug('mock:test:storage');
export { App, Bootstrap };
const GROUP_SIZE = 8;
const IdentifierType = Proto.ManifestRecord.Identifier.Type;
export type InitStorageResultType = Readonly<{
bootstrap: Bootstrap;
app: App;
group: Group;
2022-04-13 00:50:17 +00:00
members: ReadonlyArray<PrimaryDevice>;
}>;
//
// This function creates an initial storage service state that includes:
//
// - All contacts from contact sync (first contact pinned)
// - A pinned group with GROUP_SIZE members (from the contacts)
// - Account with e164 and profileKey
//
// In addition to above, this function will queue one incoming message in the
// group, and one for the first contact (so that both will appear in the left
// pane).
2022-04-13 00:50:17 +00:00
export async function initStorage(
options?: BootstrapOptions
): Promise<InitStorageResultType> {
// Creates primary device, contacts
2022-04-13 00:50:17 +00:00
const bootstrap = new Bootstrap(options);
await bootstrap.init();
2022-07-08 20:46:25 +00:00
try {
// Populate storage service
const { contacts, phone } = bootstrap;
2022-07-08 20:46:25 +00:00
const [firstContact] = contacts;
2022-07-08 20:46:25 +00:00
const members = [...contacts].slice(0, GROUP_SIZE);
2022-07-08 20:46:25 +00:00
const group = await phone.createGroup({
title: 'Mock Group',
members: [phone, ...members],
});
2022-07-08 20:46:25 +00:00
let state = StorageState.getEmpty();
2022-07-08 20:46:25 +00:00
state = state.updateAccount({
profileKey: phone.profileKey.serialize(),
e164: phone.device.number,
2022-08-03 17:10:49 +00:00
givenName: phone.profileName,
2022-07-08 20:46:25 +00:00
});
2022-07-08 20:46:25 +00:00
state = state
.addGroup(group, {
whitelisted: true,
})
.pinGroup(group);
2022-07-08 20:46:25 +00:00
for (const contact of contacts) {
state = state.addContact(contact, {
identityState: Proto.ContactRecord.IdentityState.VERIFIED,
whitelisted: true,
2022-07-08 20:46:25 +00:00
identityKey: contact.publicKey.serialize(),
profileKey: contact.profileKey.serialize(),
2022-08-03 17:10:49 +00:00
givenName: contact.profileName,
2022-07-08 20:46:25 +00:00
});
}
2022-07-08 20:46:25 +00:00
state = state.pin(firstContact);
state = state.addRecord({
type: IdentifierType.STORY_DISTRIBUTION_LIST,
record: {
storyDistributionList: {
allowsReplies: true,
identifier: uuidToBytes(MY_STORY_ID),
isBlockList: true,
name: MY_STORY_ID,
2023-08-16 20:54:39 +00:00
recipientServiceIds: [],
},
},
});
2022-07-08 20:46:25 +00:00
await phone.setStorageState(state);
2022-07-08 20:46:25 +00:00
// Link new device
const app = await bootstrap.link();
2022-07-08 20:46:25 +00:00
const { desktop } = bootstrap;
2022-07-08 20:46:25 +00:00
// Send a message to the group and the first contact
const contactSend = contacts[0].sendText(desktop, 'hello from contact', {
timestamp: bootstrap.getTimestamp(),
sealed: true,
});
2022-07-08 20:46:25 +00:00
const groupSend = members[0].sendText(desktop, 'hello in group', {
timestamp: bootstrap.getTimestamp(),
sealed: true,
group,
});
2022-07-08 20:46:25 +00:00
await Promise.all([contactSend, groupSend]);
2022-07-08 20:46:25 +00:00
return { bootstrap, app, group, members };
} catch (error) {
await bootstrap.saveLogs();
throw error;
}
}