You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-desktop/ts/receiver/multidevice.ts

446 lines
13 KiB
TypeScript

import { removeFromCache } from './cache';
import { EnvelopePlus } from './types';
import * as Data from '../../js/modules/data';
import { SignalService } from '../protobuf';
import { updateProfile } from './receiver';
import { onVerified } from './syncMessages';
import { StringUtils } from '../session/utils';
import { MultiDeviceProtocol, SessionProtocol } from '../session/protocols';
import { PubKey } from '../session/types';
import ByteBuffer from 'bytebuffer';
import { BlockedNumberController } from '../util';
async function unpairingRequestIsLegit(source: string, ourPubKey: string) {
const { textsecure, storage, lokiFileServerAPI } = window;
const isSecondary = textsecure.storage.get('isSecondaryDevice');
if (!isSecondary) {
return false;
}
const primaryPubKey = storage.get('primaryDevicePubKey');
// TODO: allow unpairing from any paired device?
if (source !== primaryPubKey) {
return false;
}
const primaryMapping = await lokiFileServerAPI.getUserDeviceMapping(
primaryPubKey
);
// If we don't have a mapping on the primary then we have been unlinked
if (!primaryMapping) {
return true;
}
// We expect the primary device to have updated its mapping
// before sending the unpairing request
const found = primaryMapping.authorisations.find(
(authorisation: any) => authorisation.secondaryDevicePubKey === ourPubKey
);
// our pubkey should NOT be in the primary device mapping
return !found;
}
async function clearAppAndRestart() {
// remove our device mapping annotations from file server
await window.lokiFileServerAPI.clearOurDeviceMappingAnnotations();
// Delete the account and restart
try {
await window.Signal.Logs.deleteAll();
await Data.removeAll();
await Data.close();
await Data.removeDB();
await Data.removeOtherData();
// TODO generate an empty db with a flag
// to display a message about the unpairing
// after the app restarts
} catch (error) {
window.log.error(
'Something went wrong deleting all data:',
error && error.stack ? error.stack : error
);
}
window.restart();
}
export async function handleUnpairRequest(
envelope: EnvelopePlus,
ourPubKey: string
) {
// TODO: move high-level pairing logic to libloki.multidevice.xx
const legit = await unpairingRequestIsLegit(envelope.source, ourPubKey);
await removeFromCache(envelope);
if (legit) {
await clearAppAndRestart();
}
}
export async function handlePairingAuthorisationMessage(
envelope: EnvelopePlus,
pairingAuthorisation: SignalService.IPairingAuthorisationMessage,
dataMessage: SignalService.IDataMessage | undefined | null
): Promise<void> {
const { secondaryDevicePubKey, grantSignature } = pairingAuthorisation;
const isGrant =
grantSignature &&
grantSignature.length > 0 &&
secondaryDevicePubKey === window.textsecure.storage.user.getNumber();
if (isGrant) {
await handleAuthorisationForSelf(
envelope,
pairingAuthorisation,
dataMessage
);
} else {
await handlePairingRequest(envelope, pairingAuthorisation);
}
}
async function handlePairingRequest(
envelope: EnvelopePlus,
pairingRequest: SignalService.IPairingAuthorisationMessage
) {
const { libloki, Whisper } = window;
const valid = await libloki.crypto.validateAuthorisation(pairingRequest);
if (valid) {
// Pairing dialog is open and is listening
if (Whisper.events.isListenedTo('devicePairingRequestReceived')) {
await MultiDeviceProtocol.savePairingAuthorisation(
pairingRequest as Data.PairingAuthorisation
);
Whisper.events.trigger(
'devicePairingRequestReceived',
pairingRequest.secondaryDevicePubKey
);
} else {
Whisper.events.trigger(
'devicePairingRequestReceivedNoListener',
pairingRequest.secondaryDevicePubKey
);
}
// Ignore requests if the dialog is closed
}
await removeFromCache(envelope);
}
async function handleAuthorisationForSelf(
envelope: EnvelopePlus,
pairingAuthorisation: SignalService.IPairingAuthorisationMessage,
dataMessage: SignalService.IDataMessage | undefined | null
) {
const { ConversationController, libloki, Whisper } = window;
const valid = await libloki.crypto.validateAuthorisation(
pairingAuthorisation
);
const alreadySecondaryDevice = !!window.storage.get('isSecondaryDevice');
if (alreadySecondaryDevice) {
window.log.warn(
'Received an unexpected pairing authorisation (device is already paired as secondary device). Ignoring.'
);
} else if (!valid) {
window.log.warn(
'Received invalid pairing authorisation for self. Could not verify signature. Ignoring.'
);
} else {
const { primaryDevicePubKey, grantSignature } = pairingAuthorisation;
if (grantSignature && grantSignature.length > 0) {
// Authorisation received to become a secondary device
window.log.info(
`Received pairing authorisation from ${primaryDevicePubKey}`
);
// Set current device as secondary.
// This will ensure the authorisation is sent
// along with each session request.
window.storage.remove('secondaryDeviceStatus');
window.storage.put('isSecondaryDevice', true);
window.storage.put('primaryDevicePubKey', primaryDevicePubKey);
await MultiDeviceProtocol.savePairingAuthorisation(
pairingAuthorisation as Data.PairingAuthorisation
);
const primaryConversation = await ConversationController.getOrCreateAndWait(
primaryDevicePubKey,
'private'
);
primaryConversation.trigger('change');
Whisper.events.trigger('secondaryDeviceRegistration');
// Update profile
if (dataMessage) {
const { profile, profileKey } = dataMessage;
if (profile && profileKey) {
const ourNumber = window.storage.get('primaryDevicePubKey');
const me = window.ConversationController.get(ourNumber);
if (me) {
await updateProfile(me, profile, profileKey);
}
} else {
window.log.warn('profile or profileKey are missing in DataMessage');
}
}
} else {
window.log.warn('Unimplemented pairing authorisation message type');
}
}
await removeFromCache(envelope);
}
function parseContacts(arrbuf: ArrayBuffer): Array<any> {
const buffer = new ByteBuffer();
buffer.append(arrbuf);
buffer.offset = 0;
buffer.limit = arrbuf.byteLength;
const next = () => {
try {
if (buffer.limit === buffer.offset) {
return undefined; // eof
}
const len = buffer.readInt32();
const nextBuffer = buffer
// tslint:disable-next-line restrict-plus-operands
.slice(buffer.offset, buffer.offset + len)
.toArrayBuffer();
// TODO: de-dupe ByteBuffer.js includes in libaxo/libts
// then remove this toArrayBuffer call.
const proto: any = SignalService.ContactDetails.decode(
new Uint8Array(nextBuffer)
);
if (proto.profileKey && proto.profileKey.length === 0) {
proto.profileKey = null;
}
buffer.skip(len);
if (proto.avatar) {
const attachmentLen = proto.avatar.length;
proto.avatar.data = buffer
// tslint:disable-next-line restrict-plus-operands
.slice(buffer.offset, buffer.offset + attachmentLen)
.toArrayBuffer();
buffer.skip(attachmentLen);
}
if (proto.profileKey) {
proto.profileKey = proto.profileKey.buffer;
}
return proto;
} catch (error) {
window.log.error(
'ProtoParser.next error:',
error && error.stack ? error.stack : error
);
}
return null;
};
const results = [];
let contactDetails = next();
while (contactDetails) {
results.push(contactDetails);
contactDetails = next();
}
return results;
}
export async function handleContacts(
envelope: EnvelopePlus,
contacts: SignalService.SyncMessage.IContacts
) {
window.log.info('contact sync');
// const { blob } = contacts;
if (!contacts.data || contacts.data.length === 0) {
window.log.error('Contacts without data');
return;
}
const attachmentPointer = {
contacts,
data: ByteBuffer.wrap(contacts.data).toArrayBuffer(), // ByteBuffer to ArrayBuffer
};
const contactDetails = parseContacts(attachmentPointer.data);
await Promise.all(
contactDetails.map(async (cd: any) => onContactReceived(cd))
);
// Not sure it `contactsync` even does anything at the moment
// const ev = new Event('contactsync');
// results.push(this.dispatchAndWait(ev));
window.log.info('handleContacts: finished');
await removeFromCache(envelope);
}
// tslint:disable-next-line: max-func-body-length
async function onContactReceived(details: any) {
const {
ConversationController,
storage,
textsecure,
libloki,
Whisper,
} = window;
const { Errors } = window.Signal.Types;
const id = details.number;
libloki.api.debug.logContactSync(
'Got sync contact message with',
id,
' details:',
details
);
if (id === textsecure.storage.user.getNumber()) {
// special case for syncing details about ourselves
if (details.profileKey) {
window.log.info('Got sync message with our own profile key');
storage.put('profileKey', details.profileKey);
}
}
const c = new Whisper.Conversation({ id });
const validationError = c.validateNumber();
if (validationError) {
window.log.error(
'Invalid contact received:',
Errors.toLogFormat(validationError)
);
return;
}
try {
const conversation = await ConversationController.getOrCreateAndWait(
id,
'private'
);
let activeAt = conversation.get('active_at');
// The idea is to make any new contact show up in the left pane. If
// activeAt is null, then this contact has been purposefully hidden.
if (activeAt !== null) {
activeAt = activeAt || Date.now();
conversation.set('active_at', activeAt);
}
const primaryDevice = await MultiDeviceProtocol.getPrimaryDevice(id);
const secondaryDevices = await MultiDeviceProtocol.getSecondaryDevices(id);
const primaryConversation = await ConversationController.getOrCreateAndWait(
primaryDevice.key,
'private'
);
const secondaryConversations = await Promise.all(
secondaryDevices.map(async d => {
const secondaryConv = await ConversationController.getOrCreateAndWait(
d.key,
'private'
);
await secondaryConv.setSecondaryStatus(true, primaryDevice.key);
return conversation;
})
);
const deviceConversations = [
primaryConversation,
...secondaryConversations,
];
// triger session request with every devices of that user
// when we do not have a session with it already
deviceConversations.forEach(device => {
// tslint:disable-next-line: no-floating-promises
SessionProtocol.sendSessionRequestIfNeeded(new PubKey(device.id));
});
if (details.profileKey) {
const profileKey = StringUtils.decode(details.profileKey, 'base64');
conversation.setProfileKey(profileKey);
}
// Do not set name to allow working with lokiProfile and nicknames
conversation.set({
// name: details.name,
color: details.color,
});
if (details.name && details.name.length) {
await conversation.setLokiProfile({ displayName: details.name });
}
if (details.nickname && details.nickname.length) {
await conversation.setNickname(details.nickname);
}
// Update the conversation avatar only if new avatar exists and hash differs
const { avatar } = details;
if (avatar && avatar.data) {
const newAttributes = await window.Signal.Types.Conversation.maybeUpdateAvatar(
conversation.attributes,
avatar.data,
{
// This is some crazy inderection...
writeNewAttachmentData: window.Signal.writeNewAttachmentData,
deleteAttachmentData: window.Signal.deleteAttachmentData,
}
);
conversation.set(newAttributes);
}
await window.Signal.Data.updateConversation(id, conversation.attributes, {
Conversation: Whisper.Conversation,
});
const { expireTimer } = details;
const isValidExpireTimer = typeof expireTimer === 'number';
if (isValidExpireTimer) {
const source = textsecure.storage.user.getNumber();
const receivedAt = Date.now();
await conversation.updateExpirationTimer(
expireTimer,
source,
receivedAt,
{ fromSync: true }
);
}
if (details.verified) {
const { verified } = details;
const verifiedEvent: any = {};
verifiedEvent.verified = {
state: verified.state,
destination: verified.destination,
identityKey: verified.identityKey.buffer,
};
verifiedEvent.viaContactSync = true;
await onVerified(verifiedEvent);
}
const isBlocked = details.blocked || false;
if (conversation.isPrivate()) {
await BlockedNumberController.setBlocked(conversation.id, isBlocked);
}
conversation.updateTextInputState();
await conversation.trigger('change', conversation);
} catch (error) {
window.log.error('onContactReceived error:', Errors.toLogFormat(error));
}
}