// 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 { 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(); } }