2023-08-29 21:51:24 +00:00
|
|
|
// Copyright 2023 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { assert } from 'chai';
|
|
|
|
import createDebug from 'debug';
|
|
|
|
import Long from 'long';
|
|
|
|
|
|
|
|
import type { App } from '../playwright';
|
|
|
|
import * as durations from '../../util/durations';
|
|
|
|
import { Bootstrap } from '../bootstrap';
|
|
|
|
|
|
|
|
export const debug = createDebug('mock:test:readSync');
|
|
|
|
|
|
|
|
describe('readSync', function needsName() {
|
|
|
|
this.timeout(durations.MINUTE);
|
2023-09-19 17:02:37 +00:00
|
|
|
this.retries(4);
|
2023-08-29 21:51:24 +00:00
|
|
|
|
|
|
|
let bootstrap: Bootstrap;
|
|
|
|
let app: App;
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
bootstrap = new Bootstrap();
|
|
|
|
await bootstrap.init();
|
|
|
|
app = await bootstrap.link();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(async function after() {
|
|
|
|
if (!bootstrap) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await bootstrap.maybeSaveLogs(this.currentTest, app);
|
|
|
|
await app.close();
|
|
|
|
await bootstrap.teardown();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('applies out of order read syncs', async () => {
|
|
|
|
const { contacts, desktop, phone } = bootstrap;
|
|
|
|
const [friend] = contacts;
|
|
|
|
|
|
|
|
const page = await app.getWindow();
|
|
|
|
|
|
|
|
debug('incoming messages');
|
|
|
|
const timestamp1 = bootstrap.getTimestamp();
|
|
|
|
const timestamp2 = bootstrap.getTimestamp();
|
|
|
|
const timestamp3 = bootstrap.getTimestamp();
|
|
|
|
await friend.sendText(desktop, '૮₍˶•⤙•˶₎ა', {
|
|
|
|
timestamp: timestamp1,
|
|
|
|
});
|
|
|
|
await friend.sendText(desktop, '(੭。╹▿╹。)੭', {
|
|
|
|
timestamp: timestamp2,
|
|
|
|
});
|
|
|
|
await friend.sendText(desktop, '<(˶ᵔᵕᵔ˶)>', {
|
|
|
|
timestamp: timestamp3,
|
|
|
|
});
|
|
|
|
|
|
|
|
debug('checking left pane for unread count');
|
|
|
|
const leftPane = page.locator('#LeftPane');
|
|
|
|
|
|
|
|
await leftPane
|
|
|
|
.locator('.module-conversation-list__item--contact-or-conversation')
|
|
|
|
.first()
|
|
|
|
.waitFor();
|
|
|
|
|
|
|
|
debug('checking left pane for 3 unread');
|
|
|
|
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'
|
|
|
|
)
|
|
|
|
.first()
|
|
|
|
.innerText();
|
|
|
|
assert.equal(unreadCount, '3', 'unread count');
|
|
|
|
|
|
|
|
debug('incoming out of order messages');
|
|
|
|
const massTimestamps = Array.from(Array(100)).map(() =>
|
|
|
|
bootstrap.getTimestamp()
|
|
|
|
);
|
|
|
|
const readTimestamp = bootstrap.getTimestamp();
|
|
|
|
const unreadTimestamp = bootstrap.getTimestamp();
|
|
|
|
|
|
|
|
async function sendReadMessage() {
|
|
|
|
debug('sending read message', { timestamp: readTimestamp });
|
|
|
|
await friend.sendText(desktop, 'read marker', {
|
|
|
|
timestamp: readTimestamp,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function sendUnreadMessage() {
|
|
|
|
debug('sending unread message', { timestamp: unreadTimestamp });
|
|
|
|
await friend.sendText(desktop, 'unread message', {
|
|
|
|
timestamp: unreadTimestamp,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
async function sendReadSyncs(timestamps: Array<number>) {
|
|
|
|
debug('sending read syncs', { timestamps });
|
|
|
|
|
|
|
|
const sendOptions = {
|
|
|
|
timestamp: bootstrap.getTimestamp(),
|
|
|
|
};
|
|
|
|
|
|
|
|
const longTimestamps = timestamps.map(timestamp =>
|
|
|
|
Long.fromNumber(timestamp)
|
|
|
|
);
|
|
|
|
|
|
|
|
const senderAci = friend.device.aci;
|
|
|
|
|
|
|
|
await phone.sendRaw(
|
|
|
|
desktop,
|
|
|
|
{
|
|
|
|
syncMessage: {
|
|
|
|
read: longTimestamps.map(timestamp => ({
|
|
|
|
senderAci,
|
|
|
|
timestamp,
|
|
|
|
})),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
sendOptions
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
await sendReadSyncs([timestamp2, timestamp3]);
|
|
|
|
await sendReadSyncs(massTimestamps);
|
|
|
|
await Promise.all(
|
|
|
|
massTimestamps.map(timestamp =>
|
|
|
|
friend.sendText(desktop, String(timestamp), {
|
|
|
|
timestamp,
|
|
|
|
})
|
|
|
|
)
|
|
|
|
);
|
|
|
|
await sendReadSyncs([readTimestamp, timestamp1]);
|
|
|
|
await sendReadMessage();
|
|
|
|
await sendUnreadMessage();
|
|
|
|
|
|
|
|
debug('checking left pane for 1 unread');
|
|
|
|
const newUnreadCount = await leftPane
|
|
|
|
.locator(
|
|
|
|
'.module-conversation-list__item--contact-or-conversation__unread-indicator.module-conversation-list__item--contact-or-conversation__unread-indicator--unread-messages'
|
|
|
|
)
|
|
|
|
.first()
|
|
|
|
.innerText();
|
|
|
|
|
|
|
|
assert.equal(newUnreadCount, '1', 'updated unread count');
|
|
|
|
debug({ newUnreadCount });
|
|
|
|
|
|
|
|
debug('opening conversation');
|
|
|
|
await leftPane
|
|
|
|
.locator('.module-conversation-list__item--contact-or-conversation')
|
|
|
|
.first()
|
|
|
|
.click();
|
|
|
|
|
|
|
|
debug('checking for latest message');
|
|
|
|
await page.locator('.module-message__text >> "unread message"').waitFor();
|
|
|
|
});
|
|
|
|
});
|