signal-desktop/ts/test-mock/pnp/send_gv2_invite_test.ts

202 lines
5.3 KiB
TypeScript
Raw Normal View History

2022-04-20 19:35:53 +00:00
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { assert } from 'chai';
import type { PrimaryDevice, Group } from '@signalapp/mock-server';
2022-07-28 16:35:29 +00:00
import { StorageState, Proto, UUIDKind } from '@signalapp/mock-server';
2022-04-20 19:35:53 +00:00
import createDebug from 'debug';
import * as durations from '../../util/durations';
import { Bootstrap } from '../bootstrap';
import type { App } from '../bootstrap';
import { MY_STORIES_ID } from '../../types/Stories';
import { uuidToBytes } from '../../util/uuidToBytes';
const IdentifierType = Proto.ManifestRecord.Identifier.Type;
2022-04-20 19:35:53 +00:00
export const debug = createDebug('mock:test:gv2');
2022-09-21 16:18:48 +00:00
describe('pnp/send gv2 invite', function needsName() {
2022-04-20 19:35:53 +00:00
this.timeout(durations.MINUTE);
let bootstrap: Bootstrap;
let app: App;
2022-04-27 00:30:44 +00:00
let aciContact: PrimaryDevice;
2022-04-20 19:35:53 +00:00
let pniContact: PrimaryDevice;
beforeEach(async () => {
bootstrap = new Bootstrap();
await bootstrap.init();
2022-04-27 00:30:44 +00:00
const { phone, server } = bootstrap;
2022-04-20 19:35:53 +00:00
let state = StorageState.getEmpty();
state = state.updateAccount({
profileKey: phone.profileKey.serialize(),
e164: phone.device.number,
});
2022-04-27 00:30:44 +00:00
aciContact = await server.createPrimaryDevice({
profileName: 'ACI Contact',
});
state = state.addContact(aciContact, {
2022-04-20 19:35:53 +00:00
identityState: Proto.ContactRecord.IdentityState.VERIFIED,
whitelisted: true,
2022-04-27 00:30:44 +00:00
identityKey: aciContact.publicKey.serialize(),
profileKey: aciContact.profileKey.serialize(),
2022-04-20 19:35:53 +00:00
});
pniContact = await server.createPrimaryDevice({
2022-04-27 00:30:44 +00:00
profileName: 'My profile is a secret',
2022-04-20 19:35:53 +00:00
});
2022-08-15 21:53:33 +00:00
state = state.addContact(
pniContact,
{
identityState: Proto.ContactRecord.IdentityState.VERIFIED,
whitelisted: true,
2022-04-20 19:35:53 +00:00
2022-08-15 21:53:33 +00:00
identityKey: pniContact.getPublicKey(UUIDKind.PNI).serialize(),
2022-04-20 19:35:53 +00:00
2022-08-15 21:53:33 +00:00
givenName: 'PNI Contact',
},
UUIDKind.PNI
);
2022-04-20 19:35:53 +00:00
state = state.addRecord({
type: IdentifierType.STORY_DISTRIBUTION_LIST,
record: {
storyDistributionList: {
allowsReplies: true,
identifier: uuidToBytes(MY_STORIES_ID),
isBlockList: true,
name: MY_STORIES_ID,
recipientUuids: [],
},
},
});
2022-04-20 19:35:53 +00:00
await phone.setStorageState(state);
app = await bootstrap.link();
});
2022-07-08 20:46:25 +00:00
afterEach(async function after() {
if (this.currentTest?.state !== 'passed') {
2022-09-01 16:35:44 +00:00
await bootstrap.saveLogs(app);
2022-07-08 20:46:25 +00:00
}
2022-04-20 19:35:53 +00:00
await app.close();
await bootstrap.teardown();
});
it('should create group and modify it', async () => {
2022-04-27 00:30:44 +00:00
const { phone } = bootstrap;
2022-04-20 19:35:53 +00:00
let state = await phone.expectStorageState('initial state');
const window = await app.getWindow();
const leftPane = window.locator('.left-pane-wrapper');
const conversationStack = window.locator('.conversation-stack');
debug('clicking compose and "New group" buttons');
await leftPane.locator('.module-main-header__compose-icon').click();
await leftPane
.locator('_react=BaseConversationListItem[title = "New group"]')
.click();
debug('inviting ACI member');
await leftPane
2022-04-27 00:30:44 +00:00
.locator('.module-left-pane__compose-search-form__input')
.fill('ACI');
await leftPane
.locator('_react=BaseConversationListItem[title = "ACI Contact"]')
2022-04-20 19:35:53 +00:00
.click();
debug('inviting PNI member');
await leftPane
.locator('.module-left-pane__compose-search-form__input')
2022-04-27 00:30:44 +00:00
.fill('PNI');
2022-04-20 19:35:53 +00:00
await leftPane
.locator('_react=BaseConversationListItem[title = "PNI Contact"]')
.click();
await leftPane
.locator('.module-left-pane__footer button >> "Next"')
.click();
debug('entering group title');
await leftPane.type('My group');
await leftPane
.locator('.module-left-pane__footer button >> "Create"')
.click();
debug('waiting for invitation modal');
{
const modal = window.locator(
'.module-GroupDialog:has-text("Invitation sent")'
);
await modal.locator('button >> "Okay"').click();
}
debug('waiting for group data from storage service');
let group: Group;
{
state = await phone.waitForStorageState({ after: state });
const groups = await phone.getAllGroups(state);
assert.strictEqual(groups.length, 1);
[group] = groups;
assert.strictEqual(group.title, 'My group');
assert.strictEqual(group.revision, 0);
assert.strictEqual(group.state.members?.length, 2);
assert.strictEqual(group.state.membersPendingProfileKey?.length, 1);
}
debug('opening group settings');
await conversationStack
.locator('button.module-ConversationHeader__button--more')
.click();
await conversationStack
.locator('.react-contextmenu-item >> "Group settings"')
.click();
debug('editing group title');
{
const detailsHeader = conversationStack.locator(
'_react=ConversationDetailsHeader'
);
detailsHeader.locator('button >> "My group"').click();
const modal = window.locator('.module-Modal:has-text("Edit group")');
// Group title should be immediately focused.
await modal.type(' (v2)');
await modal.locator('button >> "Save"').click();
}
debug('waiting for the second group update');
group = await phone.waitForGroupUpdate(group);
assert.strictEqual(group.title, 'My group (v2)');
assert.strictEqual(group.revision, 1);
});
});