2024-03-15 14:20:33 +00:00
|
|
|
// Copyright 2023 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2024-10-18 17:15:03 +00:00
|
|
|
import { randomBytes } from 'node:crypto';
|
2024-10-11 19:14:28 +00:00
|
|
|
import { join } from 'node:path';
|
|
|
|
import { readFile } from 'node:fs/promises';
|
2024-03-15 14:20:33 +00:00
|
|
|
import createDebug from 'debug';
|
|
|
|
import Long from 'long';
|
2024-05-07 16:47:46 +00:00
|
|
|
import { Proto, StorageState } from '@signalapp/mock-server';
|
2024-08-28 21:20:33 +00:00
|
|
|
import { expect } from 'playwright/test';
|
2024-03-15 14:20:33 +00:00
|
|
|
|
2024-05-07 16:47:46 +00:00
|
|
|
import { generateStoryDistributionId } from '../../types/StoryDistributionId';
|
|
|
|
import { MY_STORY_ID } from '../../types/Stories';
|
2024-10-18 17:15:03 +00:00
|
|
|
import { generateAci } from '../../types/ServiceId';
|
|
|
|
import { generateBackup } from '../../test-both/helpers/generateBackup';
|
2024-10-11 19:14:28 +00:00
|
|
|
import { IMAGE_JPEG } from '../../types/MIME';
|
2024-05-07 16:47:46 +00:00
|
|
|
import { uuidToBytes } from '../../util/uuidToBytes';
|
2024-03-15 14:20:33 +00:00
|
|
|
import * as durations from '../../util/durations';
|
|
|
|
import type { App } from '../playwright';
|
|
|
|
import { Bootstrap } from '../bootstrap';
|
2024-10-11 19:14:28 +00:00
|
|
|
import {
|
|
|
|
getMessageInTimelineByTimestamp,
|
|
|
|
sendTextMessage,
|
|
|
|
sendReaction,
|
|
|
|
} from '../helpers';
|
2024-03-15 14:20:33 +00:00
|
|
|
|
|
|
|
export const debug = createDebug('mock:test:backups');
|
|
|
|
|
2024-05-07 16:47:46 +00:00
|
|
|
const IdentifierType = Proto.ManifestRecord.Identifier.Type;
|
|
|
|
|
|
|
|
const DISTRIBUTION1 = generateStoryDistributionId();
|
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
const CAT_PATH = join(
|
|
|
|
__dirname,
|
|
|
|
'..',
|
|
|
|
'..',
|
|
|
|
'..',
|
|
|
|
'fixtures',
|
|
|
|
'cat-screenshot.png'
|
|
|
|
);
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
describe('backups', function (this: Mocha.Suite) {
|
|
|
|
this.timeout(100 * durations.MINUTE);
|
|
|
|
|
|
|
|
let bootstrap: Bootstrap;
|
|
|
|
let app: App;
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
bootstrap = new Bootstrap();
|
|
|
|
await bootstrap.init();
|
2024-10-18 17:15:03 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async function (this: Mocha.Context) {
|
|
|
|
if (!bootstrap) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await bootstrap.maybeSaveLogs(this.currentTest, app);
|
|
|
|
await app.close();
|
|
|
|
await bootstrap.teardown();
|
|
|
|
});
|
2024-04-15 20:54:21 +00:00
|
|
|
|
2024-10-18 17:15:03 +00:00
|
|
|
it('exports and imports regular backup', async function () {
|
2024-04-15 20:54:21 +00:00
|
|
|
let state = StorageState.getEmpty();
|
|
|
|
|
|
|
|
const { phone, contacts } = bootstrap;
|
|
|
|
const [friend, pinned] = contacts;
|
|
|
|
|
|
|
|
state = state.updateAccount({
|
|
|
|
profileKey: phone.profileKey.serialize(),
|
|
|
|
e164: phone.device.number,
|
|
|
|
givenName: phone.profileName,
|
|
|
|
readReceipts: true,
|
|
|
|
hasCompletedUsernameOnboarding: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
state = state.addContact(friend, {
|
|
|
|
identityKey: friend.publicKey.serialize(),
|
|
|
|
profileKey: friend.profileKey.serialize(),
|
|
|
|
});
|
|
|
|
|
|
|
|
state = state.addContact(pinned, {
|
|
|
|
identityKey: pinned.publicKey.serialize(),
|
|
|
|
profileKey: pinned.profileKey.serialize(),
|
2024-07-15 20:58:55 +00:00
|
|
|
whitelisted: true,
|
2024-04-15 20:54:21 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
state = state.pin(pinned);
|
|
|
|
|
2024-05-07 16:47:46 +00:00
|
|
|
// Create empty 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: [pinned.device.aci],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
state = state.addRecord({
|
|
|
|
type: IdentifierType.STORY_DISTRIBUTION_LIST,
|
|
|
|
record: {
|
|
|
|
storyDistributionList: {
|
|
|
|
allowsReplies: true,
|
|
|
|
identifier: uuidToBytes(DISTRIBUTION1),
|
|
|
|
isBlockList: false,
|
|
|
|
name: 'friend',
|
|
|
|
recipientServiceIds: [friend.device.aci],
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2024-04-15 20:54:21 +00:00
|
|
|
await phone.setStorageState(state);
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
app = await bootstrap.link();
|
|
|
|
|
2024-10-18 17:15:03 +00:00
|
|
|
const { desktop, server } = bootstrap;
|
2024-04-15 20:54:21 +00:00
|
|
|
|
|
|
|
{
|
|
|
|
const window = await app.getWindow();
|
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
debug('wait for storage service sync to finish');
|
|
|
|
|
2024-04-15 20:54:21 +00:00
|
|
|
const leftPane = window.locator('#LeftPane');
|
2024-07-15 20:58:55 +00:00
|
|
|
const contact = leftPane.locator(
|
|
|
|
`[data-testid="${pinned.device.aci}"] >> "${pinned.profileName}"`
|
|
|
|
);
|
|
|
|
await contact.click();
|
|
|
|
|
|
|
|
debug('setting bubble color');
|
|
|
|
const conversationStack = window.locator('.Inbox__conversation-stack');
|
|
|
|
await conversationStack
|
|
|
|
.locator('button.module-ConversationHeader__button--more')
|
|
|
|
.click();
|
|
|
|
|
|
|
|
await window
|
|
|
|
.locator('.react-contextmenu-item >> "Chat settings"')
|
|
|
|
.click();
|
|
|
|
|
|
|
|
await conversationStack
|
|
|
|
.locator('.ConversationDetails__chat-color')
|
|
|
|
.click();
|
|
|
|
await conversationStack
|
|
|
|
.locator('.ChatColorPicker__bubble--infrared')
|
|
|
|
.click();
|
|
|
|
|
|
|
|
const backButton = conversationStack.locator(
|
|
|
|
'.ConversationPanel__header__back-button'
|
|
|
|
);
|
|
|
|
// Go back from colors
|
|
|
|
await backButton.first().click();
|
|
|
|
// Go back from settings
|
|
|
|
await backButton.last().click();
|
2024-04-15 20:54:21 +00:00
|
|
|
}
|
2024-03-15 14:20:33 +00:00
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
const sends = new Array<Promise<void>>();
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
for (let i = 0; i < 5; i += 1) {
|
2024-10-11 19:14:28 +00:00
|
|
|
sends.push(
|
|
|
|
sendTextMessage({
|
|
|
|
from: phone,
|
|
|
|
to: pinned,
|
|
|
|
text: `to pinned ${i}`,
|
|
|
|
desktop,
|
2024-07-15 20:58:55 +00:00
|
|
|
timestamp: bootstrap.getTimestamp(),
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
const theirTimestamp = bootstrap.getTimestamp();
|
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
sends.push(
|
|
|
|
sendTextMessage({
|
|
|
|
from: friend,
|
|
|
|
to: desktop,
|
|
|
|
text: `msg ${i}`,
|
|
|
|
desktop,
|
|
|
|
timestamp: theirTimestamp,
|
|
|
|
})
|
|
|
|
);
|
2024-03-15 14:20:33 +00:00
|
|
|
|
|
|
|
const ourTimestamp = bootstrap.getTimestamp();
|
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
sends.push(
|
|
|
|
sendTextMessage({
|
|
|
|
from: phone,
|
|
|
|
to: friend,
|
|
|
|
text: `respond ${i}`,
|
|
|
|
desktop,
|
2024-03-15 14:20:33 +00:00
|
|
|
timestamp: ourTimestamp,
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
const reactionTimestamp = bootstrap.getTimestamp();
|
2024-10-11 19:14:28 +00:00
|
|
|
sends.push(
|
|
|
|
sendReaction({
|
|
|
|
from: friend,
|
|
|
|
to: desktop,
|
|
|
|
targetAuthor: desktop,
|
|
|
|
targetMessageTimestamp: ourTimestamp,
|
|
|
|
reactionTimestamp,
|
|
|
|
desktop,
|
|
|
|
emoji: '👍',
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
2024-03-15 14:20:33 +00:00
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
const catTimestamp = bootstrap.getTimestamp();
|
|
|
|
const plaintextCat = await readFile(CAT_PATH);
|
|
|
|
const ciphertextCat = await bootstrap.storeAttachmentOnCDN(
|
|
|
|
plaintextCat,
|
|
|
|
IMAGE_JPEG
|
|
|
|
);
|
|
|
|
sends.push(
|
|
|
|
pinned.sendRaw(
|
2024-03-15 14:20:33 +00:00
|
|
|
desktop,
|
|
|
|
{
|
|
|
|
dataMessage: {
|
2024-10-11 19:14:28 +00:00
|
|
|
timestamp: Long.fromNumber(catTimestamp),
|
|
|
|
attachments: [ciphertextCat],
|
2024-03-15 14:20:33 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
2024-10-11 19:14:28 +00:00
|
|
|
timestamp: catTimestamp,
|
2024-03-15 14:20:33 +00:00
|
|
|
}
|
2024-10-11 19:14:28 +00:00
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
await Promise.all(sends);
|
|
|
|
|
|
|
|
{
|
|
|
|
const window = await app.getWindow();
|
|
|
|
await getMessageInTimelineByTimestamp(window, catTimestamp)
|
|
|
|
.locator('img')
|
|
|
|
.waitFor();
|
2024-03-15 14:20:33 +00:00
|
|
|
}
|
|
|
|
|
2024-10-07 19:58:59 +00:00
|
|
|
await app.uploadBackup();
|
2024-05-30 18:53:30 +00:00
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
const comparator = await bootstrap.createScreenshotComparator(
|
|
|
|
app,
|
|
|
|
async (window, snapshot) => {
|
|
|
|
const leftPane = window.locator('#LeftPane');
|
2024-07-15 20:58:55 +00:00
|
|
|
const pinnedElem = leftPane.locator(
|
2024-10-11 19:14:28 +00:00
|
|
|
`[data-testid="${pinned.toContact().aci}"] >> "Photo"`
|
2024-07-15 20:58:55 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
debug('Waiting for messages to pinned contact to come through');
|
|
|
|
await pinnedElem.click();
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
const contactElem = leftPane.locator(
|
|
|
|
`[data-testid="${friend.toContact().aci}"] >> "respond 4"`
|
|
|
|
);
|
|
|
|
|
2024-07-15 20:58:55 +00:00
|
|
|
debug('Waiting for messages to regular contact to come through');
|
2024-03-15 14:20:33 +00:00
|
|
|
await contactElem.waitFor();
|
|
|
|
|
2024-07-15 20:58:55 +00:00
|
|
|
await snapshot('styled bubbles');
|
2024-03-15 14:20:33 +00:00
|
|
|
|
2024-09-03 16:41:37 +00:00
|
|
|
debug('Waiting for unread count');
|
|
|
|
const unreadCount = await leftPane
|
|
|
|
.locator(
|
|
|
|
'.module-conversation-list__item--contact-or-conversation__unread-indicator.module-conversation-list__item--contact-or-conversation__unread-indicator--unread-messages'
|
|
|
|
)
|
|
|
|
.last();
|
|
|
|
await unreadCount.waitFor();
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
debug('Going into the conversation');
|
|
|
|
await contactElem.click();
|
|
|
|
await window
|
|
|
|
.locator('.ConversationView .module-message >> "respond 4"')
|
|
|
|
.waitFor();
|
|
|
|
|
2024-09-03 16:41:37 +00:00
|
|
|
debug('Waiting for conversation to be marked read');
|
|
|
|
await unreadCount.waitFor({ state: 'hidden' });
|
2024-05-07 16:47:46 +00:00
|
|
|
|
|
|
|
debug('Switching to stories nav tab');
|
|
|
|
await window.getByTestId('NavTabsItem--Stories').click();
|
|
|
|
|
|
|
|
debug('Opening story privacy');
|
|
|
|
await window.locator('.StoriesTab__MoreActionsIcon').click();
|
|
|
|
await window.getByRole('button', { name: 'Story Privacy' }).click();
|
2024-08-28 21:20:33 +00:00
|
|
|
await expect(
|
|
|
|
window.locator('.StoriesSettingsModal__overlay')
|
|
|
|
).toHaveCSS('opacity', '1');
|
2024-05-07 16:47:46 +00:00
|
|
|
|
|
|
|
await snapshot('story privacy');
|
2024-03-15 14:20:33 +00:00
|
|
|
},
|
|
|
|
this.test
|
|
|
|
);
|
|
|
|
|
|
|
|
await app.close();
|
|
|
|
|
|
|
|
// Restart
|
2024-05-31 14:15:43 +00:00
|
|
|
await bootstrap.eraseStorage();
|
2024-10-11 19:14:28 +00:00
|
|
|
await server.removeAllCDNAttachments();
|
2024-10-07 19:58:59 +00:00
|
|
|
app = await bootstrap.link();
|
|
|
|
await app.waitForBackupImportComplete();
|
|
|
|
|
|
|
|
// Make sure that contact sync happens after backup import, otherwise the
|
|
|
|
// app won't show contacts as "system"
|
|
|
|
await app.waitForContactSync();
|
2024-03-15 14:20:33 +00:00
|
|
|
|
2024-10-11 19:14:28 +00:00
|
|
|
debug('Waiting for attachments to be downloaded');
|
|
|
|
{
|
|
|
|
const window = await app.getWindow();
|
|
|
|
await window
|
|
|
|
.locator('.BackupMediaDownloadProgress__button-close')
|
|
|
|
.click();
|
|
|
|
}
|
|
|
|
|
2024-03-15 14:20:33 +00:00
|
|
|
await comparator(app);
|
|
|
|
});
|
2024-10-18 17:15:03 +00:00
|
|
|
|
|
|
|
it('imports ephemeral backup', async function () {
|
|
|
|
const ephemeralBackupKey = randomBytes(32);
|
|
|
|
const cdnKey = randomBytes(16).toString('hex');
|
|
|
|
|
|
|
|
const { phone, server } = bootstrap;
|
|
|
|
|
|
|
|
const contact1 = generateAci();
|
|
|
|
const contact2 = generateAci();
|
|
|
|
|
|
|
|
phone.ephemeralBackupKey = ephemeralBackupKey;
|
|
|
|
|
|
|
|
// Store backup attachment in transit tier
|
|
|
|
const { stream: backupStream } = generateBackup({
|
|
|
|
aci: phone.device.aci,
|
|
|
|
profileKey: phone.profileKey.serialize(),
|
|
|
|
backupKey: ephemeralBackupKey,
|
|
|
|
conversations: 2,
|
|
|
|
conversationAcis: [contact1, contact2],
|
|
|
|
messages: 50,
|
|
|
|
});
|
|
|
|
|
|
|
|
await server.storeAttachmentOnCdn(3, cdnKey, backupStream);
|
|
|
|
|
|
|
|
app = await bootstrap.link({
|
|
|
|
ephemeralBackup: {
|
|
|
|
cdn: 3,
|
|
|
|
key: cdnKey,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
await app.waitForBackupImportComplete();
|
|
|
|
|
|
|
|
const window = await app.getWindow();
|
|
|
|
|
|
|
|
const leftPane = window.locator('#LeftPane');
|
|
|
|
|
|
|
|
const contact1Elem = leftPane.locator(
|
|
|
|
`[data-testid="${contact1}"] >> "Message 48"`
|
|
|
|
);
|
|
|
|
const contact2Elem = leftPane.locator(
|
|
|
|
`[data-testid="${contact2}"] >> "Message 49"`
|
|
|
|
);
|
|
|
|
await contact1Elem.waitFor();
|
|
|
|
|
|
|
|
await contact2Elem.click();
|
|
|
|
await window.locator('.module-message >> "Message 33"').waitFor();
|
|
|
|
});
|
2024-03-15 14:20:33 +00:00
|
|
|
});
|