417 lines
12 KiB
TypeScript
417 lines
12 KiB
TypeScript
|
// Copyright 2024 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
|
||
|
import { assert } from 'chai';
|
||
|
import {
|
||
|
type PrimaryDevice,
|
||
|
Proto,
|
||
|
StorageState,
|
||
|
} from '@signalapp/mock-server';
|
||
|
import createDebug from 'debug';
|
||
|
import Long from 'long';
|
||
|
|
||
|
import * as durations from '../../util/durations';
|
||
|
import { uuidToBytes } from '../../util/uuidToBytes';
|
||
|
import { MY_STORY_ID } from '../../types/Stories';
|
||
|
import { Bootstrap } from '../bootstrap';
|
||
|
import type { App } from '../bootstrap';
|
||
|
import { expectSystemMessages, typeIntoInput } from '../helpers';
|
||
|
|
||
|
export const debug = createDebug('mock:test:messaging');
|
||
|
|
||
|
const IdentifierType = Proto.ManifestRecord.Identifier.Type;
|
||
|
|
||
|
const DAY = 24 * 3600;
|
||
|
|
||
|
describe('messaging/expireTimerVersion', function (this: Mocha.Suite) {
|
||
|
this.timeout(durations.MINUTE);
|
||
|
|
||
|
let bootstrap: Bootstrap;
|
||
|
let app: App;
|
||
|
let stranger: PrimaryDevice;
|
||
|
const STRANGER_NAME = 'Stranger';
|
||
|
|
||
|
beforeEach(async () => {
|
||
|
bootstrap = new Bootstrap({ contactCount: 1 });
|
||
|
await bootstrap.init();
|
||
|
|
||
|
const {
|
||
|
server,
|
||
|
phone,
|
||
|
contacts: [contact],
|
||
|
} = bootstrap;
|
||
|
|
||
|
stranger = await server.createPrimaryDevice({
|
||
|
profileName: STRANGER_NAME,
|
||
|
});
|
||
|
|
||
|
let state = StorageState.getEmpty();
|
||
|
|
||
|
state = state.updateAccount({
|
||
|
profileKey: phone.profileKey.serialize(),
|
||
|
e164: phone.device.number,
|
||
|
});
|
||
|
|
||
|
state = state.addContact(stranger, {
|
||
|
identityState: Proto.ContactRecord.IdentityState.DEFAULT,
|
||
|
whitelisted: true,
|
||
|
serviceE164: undefined,
|
||
|
profileKey: stranger.profileKey.serialize(),
|
||
|
});
|
||
|
|
||
|
state = state.addContact(contact, {
|
||
|
identityState: Proto.ContactRecord.IdentityState.DEFAULT,
|
||
|
whitelisted: true,
|
||
|
serviceE164: undefined,
|
||
|
profileKey: contact.profileKey.serialize(),
|
||
|
});
|
||
|
contact.device.capabilities.versionedExpirationTimer = false;
|
||
|
|
||
|
// Put both contacts in left pane
|
||
|
state = state.pin(stranger);
|
||
|
state = state.pin(contact);
|
||
|
|
||
|
// Add my story
|
||
|
state = state.addRecord({
|
||
|
type: IdentifierType.STORY_DISTRIBUTION_LIST,
|
||
|
record: {
|
||
|
storyDistributionList: {
|
||
|
allowsReplies: true,
|
||
|
identifier: uuidToBytes(MY_STORY_ID),
|
||
|
isBlockList: true,
|
||
|
name: MY_STORY_ID,
|
||
|
recipientServiceIds: [],
|
||
|
},
|
||
|
},
|
||
|
});
|
||
|
|
||
|
await phone.setStorageState(state);
|
||
|
|
||
|
app = await bootstrap.link();
|
||
|
});
|
||
|
|
||
|
afterEach(async function (this: Mocha.Context) {
|
||
|
await bootstrap.maybeSaveLogs(this.currentTest, app);
|
||
|
await app.close();
|
||
|
await bootstrap.teardown();
|
||
|
});
|
||
|
|
||
|
const SCENARIOS = [
|
||
|
{
|
||
|
name: 'they win and we start',
|
||
|
theyFirst: false,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 3,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 4,
|
||
|
finalTimer: 90 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: [
|
||
|
'You set the disappearing message time to 60 days.',
|
||
|
`${STRANGER_NAME} set the disappearing message time to 90 days.`,
|
||
|
],
|
||
|
},
|
||
|
{
|
||
|
name: 'they win and start',
|
||
|
theyFirst: true,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 3,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 4,
|
||
|
finalTimer: 90 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: [
|
||
|
`${STRANGER_NAME} set the disappearing message time to 90 days.`,
|
||
|
],
|
||
|
},
|
||
|
{
|
||
|
name: 'we win and start',
|
||
|
theyFirst: false,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 4,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 3,
|
||
|
finalTimer: 60 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: ['You set the disappearing message time to 60 days.'],
|
||
|
},
|
||
|
{
|
||
|
name: 'we win and they start',
|
||
|
theyFirst: true,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 4,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 3,
|
||
|
finalTimer: 60 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: [
|
||
|
`${STRANGER_NAME} set the disappearing message time to 90 days.`,
|
||
|
'You set the disappearing message time to 60 days.',
|
||
|
],
|
||
|
},
|
||
|
{
|
||
|
name: 'race and we start',
|
||
|
theyFirst: false,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 4,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 4,
|
||
|
finalTimer: 90 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: [
|
||
|
'You set the disappearing message time to 60 days.',
|
||
|
`${STRANGER_NAME} set the disappearing message time to 90 days.`,
|
||
|
],
|
||
|
},
|
||
|
{
|
||
|
name: 'race and they start',
|
||
|
theyFirst: true,
|
||
|
ourTimer: 60 * DAY,
|
||
|
ourVersion: 4,
|
||
|
theirTimer: 90 * DAY,
|
||
|
theirVersion: 4,
|
||
|
finalTimer: 60 * DAY,
|
||
|
finalVersion: 4,
|
||
|
systemMessages: [
|
||
|
`${STRANGER_NAME} set the disappearing message time to 90 days.`,
|
||
|
'You set the disappearing message time to 60 days.',
|
||
|
],
|
||
|
},
|
||
|
];
|
||
|
|
||
|
for (const scenario of SCENARIOS) {
|
||
|
const testName =
|
||
|
`sets correct version after ${scenario.name}, ` +
|
||
|
`theyFirst=${scenario.theyFirst}`;
|
||
|
// eslint-disable-next-line no-loop-func
|
||
|
it(testName, async () => {
|
||
|
const { phone, desktop } = bootstrap;
|
||
|
|
||
|
const sendSync = async () => {
|
||
|
debug('Send a sync message');
|
||
|
const timestamp = bootstrap.getTimestamp();
|
||
|
const destinationServiceId = stranger.device.aci;
|
||
|
const content = {
|
||
|
syncMessage: {
|
||
|
sent: {
|
||
|
destinationServiceId,
|
||
|
timestamp: Long.fromNumber(timestamp),
|
||
|
message: {
|
||
|
body: 'request',
|
||
|
timestamp: Long.fromNumber(timestamp),
|
||
|
expireTimer: scenario.ourTimer,
|
||
|
expireTimerVersion: scenario.ourVersion,
|
||
|
},
|
||
|
unidentifiedStatus: [
|
||
|
{
|
||
|
destinationServiceId,
|
||
|
},
|
||
|
],
|
||
|
},
|
||
|
},
|
||
|
};
|
||
|
const sendOptions = {
|
||
|
timestamp,
|
||
|
};
|
||
|
await phone.sendRaw(desktop, content, sendOptions);
|
||
|
};
|
||
|
|
||
|
const sendResponse = async () => {
|
||
|
debug('Send a response message');
|
||
|
const timestamp = bootstrap.getTimestamp();
|
||
|
const content = {
|
||
|
dataMessage: {
|
||
|
body: 'response',
|
||
|
timestamp: Long.fromNumber(timestamp),
|
||
|
expireTimer: scenario.theirTimer,
|
||
|
expireTimerVersion: scenario.theirVersion,
|
||
|
},
|
||
|
};
|
||
|
const sendOptions = {
|
||
|
timestamp,
|
||
|
};
|
||
|
const key = await desktop.popSingleUseKey();
|
||
|
await stranger.addSingleUseKey(desktop, key);
|
||
|
await stranger.sendRaw(desktop, content, sendOptions);
|
||
|
};
|
||
|
|
||
|
if (scenario.theyFirst) {
|
||
|
await sendResponse();
|
||
|
await sendSync();
|
||
|
} else {
|
||
|
await sendSync();
|
||
|
await sendResponse();
|
||
|
}
|
||
|
|
||
|
const window = await app.getWindow();
|
||
|
const leftPane = window.locator('#LeftPane');
|
||
|
|
||
|
debug('opening conversation with the contact');
|
||
|
await leftPane
|
||
|
.locator(
|
||
|
`[data-testid="${stranger.device.aci}"] >> "${stranger.profileName}"`
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
await expectSystemMessages(window, scenario.systemMessages);
|
||
|
|
||
|
await window.locator('.module-conversation-hero').waitFor();
|
||
|
|
||
|
debug('Send message to merged contact');
|
||
|
{
|
||
|
const compositionInput = await app.waitForEnabledComposer();
|
||
|
|
||
|
await typeIntoInput(compositionInput, 'Hello');
|
||
|
await compositionInput.press('Enter');
|
||
|
}
|
||
|
|
||
|
debug('Getting message to contact');
|
||
|
const { body, dataMessage } = await stranger.waitForMessage();
|
||
|
|
||
|
assert.strictEqual(body, 'Hello');
|
||
|
assert.strictEqual(dataMessage.expireTimer, scenario.finalTimer);
|
||
|
assert.strictEqual(dataMessage.expireTimerVersion, scenario.finalVersion);
|
||
|
});
|
||
|
}
|
||
|
|
||
|
it('should not bump version for not capable recipient', async () => {
|
||
|
const {
|
||
|
contacts: [contact],
|
||
|
} = bootstrap;
|
||
|
|
||
|
const window = await app.getWindow();
|
||
|
const leftPane = window.locator('#LeftPane');
|
||
|
|
||
|
debug('opening conversation with the contact');
|
||
|
await leftPane
|
||
|
.locator(
|
||
|
`[data-testid="${contact.device.aci}"] >> "${contact.profileName}"`
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
await window.locator('.module-conversation-hero').waitFor();
|
||
|
|
||
|
const conversationStack = window.locator('.Inbox__conversation-stack');
|
||
|
|
||
|
debug('setting timer to 1 week');
|
||
|
await conversationStack
|
||
|
.locator('button.module-ConversationHeader__button--more')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator('.react-contextmenu-item >> "Disappearing messages"')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator(
|
||
|
'.module-ConversationHeader__disappearing-timer__item >> "1 week"'
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
debug('Getting first expiration update');
|
||
|
{
|
||
|
const { dataMessage } = await contact.waitForMessage();
|
||
|
assert.strictEqual(
|
||
|
dataMessage.flags,
|
||
|
Proto.DataMessage.Flags.EXPIRATION_TIMER_UPDATE
|
||
|
);
|
||
|
assert.strictEqual(dataMessage.expireTimer, 604800);
|
||
|
assert.strictEqual(dataMessage.expireTimerVersion, 1);
|
||
|
}
|
||
|
|
||
|
debug('setting timer to 4 weeks');
|
||
|
await conversationStack
|
||
|
.locator('button.module-ConversationHeader__button--more')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator('.react-contextmenu-item >> "Disappearing messages"')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator(
|
||
|
'.module-ConversationHeader__disappearing-timer__item >> "4 weeks"'
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
debug('Getting second expiration update');
|
||
|
{
|
||
|
const { dataMessage } = await contact.waitForMessage();
|
||
|
assert.strictEqual(
|
||
|
dataMessage.flags,
|
||
|
Proto.DataMessage.Flags.EXPIRATION_TIMER_UPDATE
|
||
|
);
|
||
|
assert.strictEqual(dataMessage.expireTimer, 2419200);
|
||
|
assert.strictEqual(dataMessage.expireTimerVersion, 1);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
it('should bump version for capable recipient', async () => {
|
||
|
const window = await app.getWindow();
|
||
|
const leftPane = window.locator('#LeftPane');
|
||
|
|
||
|
debug('opening conversation with the contact');
|
||
|
await leftPane
|
||
|
.locator(
|
||
|
`[data-testid="${stranger.device.aci}"] >> "${stranger.profileName}"`
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
await window.locator('.module-conversation-hero').waitFor();
|
||
|
|
||
|
const conversationStack = window.locator('.Inbox__conversation-stack');
|
||
|
|
||
|
debug('setting timer to 1 week');
|
||
|
await conversationStack
|
||
|
.locator('button.module-ConversationHeader__button--more')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator('.react-contextmenu-item >> "Disappearing messages"')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator(
|
||
|
'.module-ConversationHeader__disappearing-timer__item >> "1 week"'
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
debug('Getting first expiration update');
|
||
|
{
|
||
|
const { dataMessage } = await stranger.waitForMessage();
|
||
|
assert.strictEqual(
|
||
|
dataMessage.flags,
|
||
|
Proto.DataMessage.Flags.EXPIRATION_TIMER_UPDATE
|
||
|
);
|
||
|
assert.strictEqual(dataMessage.expireTimer, 604800);
|
||
|
assert.strictEqual(dataMessage.expireTimerVersion, 2);
|
||
|
}
|
||
|
|
||
|
debug('setting timer to 4 weeks');
|
||
|
await conversationStack
|
||
|
.locator('button.module-ConversationHeader__button--more')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator('.react-contextmenu-item >> "Disappearing messages"')
|
||
|
.click();
|
||
|
|
||
|
await window
|
||
|
.locator(
|
||
|
'.module-ConversationHeader__disappearing-timer__item >> "4 weeks"'
|
||
|
)
|
||
|
.click();
|
||
|
|
||
|
debug('Getting second expiration update');
|
||
|
{
|
||
|
const { dataMessage } = await stranger.waitForMessage();
|
||
|
assert.strictEqual(
|
||
|
dataMessage.flags,
|
||
|
Proto.DataMessage.Flags.EXPIRATION_TIMER_UPDATE
|
||
|
);
|
||
|
assert.strictEqual(dataMessage.expireTimer, 2419200);
|
||
|
assert.strictEqual(dataMessage.expireTimerVersion, 3);
|
||
|
}
|
||
|
});
|
||
|
});
|