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.
446 lines
13 KiB
TypeScript
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));
|
|
}
|
|
}
|