171 lines
4.5 KiB
TypeScript
171 lines
4.5 KiB
TypeScript
// Copyright 2020-2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
/* eslint-disable more/no-then */
|
|
/* eslint-disable @typescript-eslint/ban-types */
|
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
/* eslint-disable max-classes-per-file */
|
|
|
|
import EventTarget, { EventHandler } from './EventTarget';
|
|
import MessageReceiver from './MessageReceiver';
|
|
import { ContactSyncEvent, GroupSyncEvent } from './messageReceiverEvents';
|
|
import MessageSender from './SendMessage';
|
|
import { assert } from '../util/assert';
|
|
import { getSendOptions } from '../util/getSendOptions';
|
|
import { handleMessageSend } from '../util/handleMessageSend';
|
|
import * as log from '../logging/log';
|
|
|
|
class SyncRequestInner extends EventTarget {
|
|
private started = false;
|
|
|
|
contactSync?: boolean;
|
|
|
|
groupSync?: boolean;
|
|
|
|
timeout: any;
|
|
|
|
oncontact: (event: ContactSyncEvent) => void;
|
|
|
|
ongroup: (event: GroupSyncEvent) => void;
|
|
|
|
timeoutMillis: number;
|
|
|
|
constructor(
|
|
private sender: MessageSender,
|
|
private receiver: MessageReceiver,
|
|
timeoutMillis?: number
|
|
) {
|
|
super();
|
|
|
|
if (
|
|
!(sender instanceof MessageSender) ||
|
|
!(receiver instanceof MessageReceiver)
|
|
) {
|
|
throw new Error(
|
|
'Tried to construct a SyncRequest without MessageSender and MessageReceiver'
|
|
);
|
|
}
|
|
|
|
this.oncontact = this.onContactSyncComplete.bind(this);
|
|
receiver.addEventListener('contactSync', this.oncontact);
|
|
|
|
this.ongroup = this.onGroupSyncComplete.bind(this);
|
|
receiver.addEventListener('groupSync', this.ongroup);
|
|
|
|
this.timeoutMillis = timeoutMillis || 60000;
|
|
}
|
|
|
|
async start(): Promise<void> {
|
|
if (this.started) {
|
|
assert(false, 'SyncRequestInner: started more than once. Doing nothing');
|
|
return;
|
|
}
|
|
this.started = true;
|
|
|
|
const { sender } = this;
|
|
|
|
const ourConversation = window.ConversationController.getOurConversationOrThrow();
|
|
const sendOptions = await getSendOptions(ourConversation.attributes, {
|
|
syncMessage: true,
|
|
});
|
|
|
|
if (window.ConversationController.areWePrimaryDevice()) {
|
|
log.warn('SyncRequest.start: We are primary device; returning early');
|
|
return;
|
|
}
|
|
|
|
log.info('SyncRequest created. Sending config sync request...');
|
|
handleMessageSend(sender.sendRequestConfigurationSyncMessage(sendOptions), {
|
|
messageIds: [],
|
|
sendType: 'otherSync',
|
|
});
|
|
|
|
log.info('SyncRequest now sending block sync request...');
|
|
handleMessageSend(sender.sendRequestBlockSyncMessage(sendOptions), {
|
|
messageIds: [],
|
|
sendType: 'otherSync',
|
|
});
|
|
|
|
log.info('SyncRequest now sending contact sync message...');
|
|
handleMessageSend(sender.sendRequestContactSyncMessage(sendOptions), {
|
|
messageIds: [],
|
|
sendType: 'otherSync',
|
|
})
|
|
.then(() => {
|
|
log.info('SyncRequest now sending group sync message...');
|
|
return handleMessageSend(
|
|
sender.sendRequestGroupSyncMessage(sendOptions),
|
|
{ messageIds: [], sendType: 'otherSync' }
|
|
);
|
|
})
|
|
.catch((error: Error) => {
|
|
log.error(
|
|
'SyncRequest error:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
});
|
|
this.timeout = setTimeout(this.onTimeout.bind(this), this.timeoutMillis);
|
|
}
|
|
|
|
onContactSyncComplete() {
|
|
this.contactSync = true;
|
|
this.update();
|
|
}
|
|
|
|
onGroupSyncComplete() {
|
|
this.groupSync = true;
|
|
this.update();
|
|
}
|
|
|
|
update() {
|
|
if (this.contactSync && this.groupSync) {
|
|
this.dispatchEvent(new Event('success'));
|
|
this.cleanup();
|
|
}
|
|
}
|
|
|
|
onTimeout() {
|
|
if (this.contactSync || this.groupSync) {
|
|
this.dispatchEvent(new Event('success'));
|
|
} else {
|
|
this.dispatchEvent(new Event('timeout'));
|
|
}
|
|
this.cleanup();
|
|
}
|
|
|
|
cleanup() {
|
|
clearTimeout(this.timeout);
|
|
this.receiver.removeEventListener('contactsync', this.oncontact);
|
|
this.receiver.removeEventListener('groupSync', this.ongroup);
|
|
delete this.listeners;
|
|
}
|
|
}
|
|
|
|
export default class SyncRequest {
|
|
private inner: SyncRequestInner;
|
|
|
|
addEventListener: (
|
|
name: 'success' | 'timeout',
|
|
handler: EventHandler
|
|
) => void;
|
|
|
|
removeEventListener: (
|
|
name: 'success' | 'timeout',
|
|
handler: EventHandler
|
|
) => void;
|
|
|
|
constructor(
|
|
sender: MessageSender,
|
|
receiver: MessageReceiver,
|
|
timeoutMillis?: number
|
|
) {
|
|
const inner = new SyncRequestInner(sender, receiver, timeoutMillis);
|
|
this.inner = inner;
|
|
this.addEventListener = inner.addEventListener.bind(inner);
|
|
this.removeEventListener = inner.removeEventListener.bind(inner);
|
|
}
|
|
|
|
start(): void {
|
|
this.inner.start();
|
|
}
|
|
}
|