Merge pull request #834 from loki-project/multi-device-fix

Multi device fixes
pull/836/head
Mikunj Varsani 5 years ago committed by GitHub
commit 13634a42c6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -3,9 +3,10 @@
[![Build Status](https://travis-ci.org/loki-project/loki-messenger.svg?branch=development)](https://travis-ci.org/loki-project/loki-messenger) [![Build Status](https://travis-ci.org/loki-project/loki-messenger.svg?branch=development)](https://travis-ci.org/loki-project/loki-messenger)
Session allows for truly decentralized, end to end, and private encrypted chats. Session is built to handle both online and fully Asynchronous offline messages. Session implements the Signal protocol for message encryption. Our Client interface is a fork of [Signal Messenger](https://signal.org/). Session allows for truly decentralized, end to end, and private encrypted chats. Session is built to handle both online and fully Asynchronous offline messages. Session implements the Signal protocol for message encryption. Our Client interface is a fork of [Signal Messenger](https://signal.org/).
## Summary ## Summary
Session integrates directly with Loki [Service Nodes](https://lokidocs.com/ServiceNodes/SNOverview/), which are a set of distributed, decentralized and Sybil resistant nodes. Service Nodes act as servers which store messages offline, and a set of nodes which allow for onion routing functionality obfuscating users IP Addresses. For a full understanding of how Session works, read the [Loki whitepaper](https://loki.network/whitepaper). Session integrates directly with Loki [Service Nodes](https://lokidocs.com/ServiceNodes/SNOverview/), which are a set of distributed, decentralized and Sybil resistant nodes. Service Nodes act as servers which store messages offline, and a set of nodes which allow for onion routing functionality obfuscating users IP Addresses. For a full understanding of how Session works, read the [Loki whitepaper](https://loki.network/whitepaper).
**Offline messages** **Offline messages**

@ -763,9 +763,10 @@
const ev = new Event('group'); const ev = new Event('group');
const ourKey = textsecure.storage.user.getNumber(); const primaryDeviceKey =
window.storage.get('primaryDevicePubKey') ||
const allMembers = [ourKey, ...members]; textsecure.storage.user.getNumber();
const allMembers = [primaryDeviceKey, ...members];
ev.groupDetails = { ev.groupDetails = {
id: groupId, id: groupId,
@ -794,7 +795,7 @@
window.friends.friendRequestStatusEnum.friends window.friends.friendRequestStatusEnum.friends
); );
convo.updateGroupAdmins([ourKey]); convo.updateGroupAdmins([primaryDeviceKey]);
appView.openConversation(groupId, {}); appView.openConversation(groupId, {});
}; };

@ -198,10 +198,25 @@
isOnline() { isOnline() {
return this.isMe() || this.get('isOnline'); return this.isMe() || this.get('isOnline');
}, },
isMe() { isMe() {
return this.isOurLocalDevice() || this.isOurPrimaryDevice();
},
isOurPrimaryDevice() {
return this.id === window.storage.get('primaryDevicePubKey'); return this.id === window.storage.get('primaryDevicePubKey');
}, },
async isOurDevice() {
if (this.isMe()) {
return true;
}
const ourDevices = await window.libloki.storage.getPairedDevicesFor(
this.ourNumber
);
return ourDevices.includes(this.id);
},
isOurLocalDevice() {
return this.id === this.ourNumber;
},
isPublic() { isPublic() {
return !!(this.id && this.id.match(/^publicChat:/)); return !!(this.id && this.id.match(/^publicChat:/));
}, },
@ -886,9 +901,6 @@
throw new Error('Invalid friend request state'); throw new Error('Invalid friend request state');
} }
}, },
isOurConversation() {
return this.id === this.ourNumber;
},
isSecondaryDevice() { isSecondaryDevice() {
return !!this.get('secondaryStatus'); return !!this.get('secondaryStatus');
}, },

@ -205,7 +205,7 @@
}, },
getLokiNameForNumber(number) { getLokiNameForNumber(number) {
const conversation = ConversationController.get(number); const conversation = ConversationController.get(number);
if (!conversation) { if (!conversation || !conversation.getLokiProfile()) {
return number; return number;
} }
return conversation.getLokiProfile().displayName; return conversation.getLokiProfile().displayName;
@ -1898,6 +1898,8 @@
const authorisation = await libloki.storage.getGrantAuthorisationForSecondaryPubKey( const authorisation = await libloki.storage.getGrantAuthorisationForSecondaryPubKey(
source source
); );
const primarySource =
(authorisation && authorisation.primaryDevicePubKey) || source;
const isGroupMessage = !!initialMessage.group; const isGroupMessage = !!initialMessage.group;
if (isGroupMessage) { if (isGroupMessage) {
conversationId = initialMessage.group.id; conversationId = initialMessage.group.id;
@ -1916,10 +1918,12 @@
const knownMembers = conversation.get('members'); const knownMembers = conversation.get('members');
if (!newGroup && knownMembers) { if (!newGroup && knownMembers) {
const fromMember = knownMembers.includes(source); const fromMember = knownMembers.includes(primarySource);
if (!fromMember) { if (!fromMember) {
window.log.warn(`Ignoring group message from non-member: ${source}`); window.log.warn(
`Ignoring group message from non-member: ${primarySource}`
);
confirm(); confirm();
return null; return null;
} }
@ -1938,7 +1942,9 @@
); );
} }
const fromAdmin = conversation.get('groupAdmins').includes(source); const fromAdmin = conversation
.get('groupAdmins')
.includes(primarySource);
if (!fromAdmin) { if (!fromAdmin) {
// Make sure the message is not removing members / renaming the group // Make sure the message is not removing members / renaming the group
@ -2016,11 +2022,11 @@
.getConversations() .getConversations()
.models.filter(c => c.get('members')) .models.filter(c => c.get('members'))
.reduce((acc, x) => window.Lodash.concat(acc, x.get('members')), []) .reduce((acc, x) => window.Lodash.concat(acc, x.get('members')), [])
.includes(source); .includes(primarySource);
if (groupMember) { if (groupMember) {
window.log.info( window.log.info(
`Auto accepting a 'group' friend request for a known group member: ${groupMember}` `Auto accepting a 'group' friend request for a known group member: ${primarySource}`
); );
window.libloki.api.sendBackgroundMessage(message.get('source')); window.libloki.api.sendBackgroundMessage(message.get('source'));
@ -2355,6 +2361,12 @@
await sendingDeviceConversation.onFriendRequestAccepted(); await sendingDeviceConversation.onFriendRequestAccepted();
} }
} }
// We need to map the original message source to the primary device
if (source !== ourNumber) {
message.set({ source: primarySource });
}
const id = await window.Signal.Data.saveMessage(message.attributes, { const id = await window.Signal.Data.saveMessage(message.attributes, {
Message: Whisper.Message, Message: Whisper.Message,
}); });

@ -487,7 +487,10 @@ class LokiAppDotNetServerAPI {
try { try {
response = options.textResponse ? respStr : JSON.parse(respStr); response = options.textResponse ? respStr : JSON.parse(respStr);
} catch (e) { } catch (e) {
log.warn(`_sendToProxy Could not parse inner JSON [${respStr}]`, endpoint); log.warn(
`_sendToProxy Could not parse inner JSON [${respStr}]`,
endpoint
);
} }
} else { } else {
log.warn( log.warn(

@ -240,6 +240,10 @@
return secondaryPubKeys.concat(primaryDevicePubKey); return secondaryPubKeys.concat(primaryDevicePubKey);
} }
function getPairedDevicesFor(pubkey) {
return window.Signal.Data.getPairedDevicesFor(pubkey);
}
window.libloki.storage = { window.libloki.storage = {
getPreKeyBundleForContact, getPreKeyBundleForContact,
saveContactPreKeyBundle, saveContactPreKeyBundle,
@ -250,6 +254,7 @@
removePairingAuthorisationForSecondaryPubKey, removePairingAuthorisationForSecondaryPubKey,
getGrantAuthorisationForSecondaryPubKey, getGrantAuthorisationForSecondaryPubKey,
getAuthorisationForSecondaryPubKey, getAuthorisationForSecondaryPubKey,
getPairedDevicesFor,
getAllDevicePubKeysForPrimaryPubKey, getAllDevicePubKeysForPrimaryPubKey,
getSecondaryDevicesFor, getSecondaryDevicesFor,
getPrimaryDeviceMapping, getPrimaryDeviceMapping,

@ -1314,8 +1314,9 @@ MessageReceiver.prototype.extend({
primaryPubKey primaryPubKey
); );
// If we don't have a mapping on the primary then we have been unlinked
if (!primaryMapping) { if (!primaryMapping) {
return false; return true;
} }
// We expect the primary device to have updated its mapping // We expect the primary device to have updated its mapping
@ -1366,7 +1367,11 @@ MessageReceiver.prototype.extend({
} }
} }
if (friendRequest) { // If we got a friend request message or
// if we're not friends with the current user that sent this private message
// Check to see if we need to auto accept their friend request
const isGroupMessage = !!groupId;
if (friendRequest || (!isGroupMessage && !conversation.isFriend())) {
if (isMe) { if (isMe) {
window.log.info('refusing to add a friend request to ourselves'); window.log.info('refusing to add a friend request to ourselves');
throw new Error('Cannot add a friend request for ourselves!'); throw new Error('Cannot add a friend request for ourselves!');

@ -350,23 +350,33 @@ OutgoingMessage.prototype = {
} catch (e) { } catch (e) {
// do nothing // do nothing
} }
if ( if (conversation && !this.isGroup) {
conversation && const isOurDevice = await conversation.isOurDevice();
!conversation.isFriend() && const isFriends =
!conversation.hasReceivedFriendRequest() && conversation.isFriend() ||
!this.isGroup conversation.hasReceivedFriendRequest();
) { // We should only send a friend request to our device if we don't have keys
// We want to send an automated friend request if: const shouldSendAutomatedFR = isOurDevice ? !keysFound : !isFriends;
// - We aren't already friends if (shouldSendAutomatedFR) {
// - We haven't received a friend request from this device // We want to send an automated friend request if:
// - We haven't sent a friend request recently // - We aren't already friends
if (conversation.friendRequestTimerIsExpired()) { // - We haven't received a friend request from this device
isMultiDeviceRequest = true; // - We haven't sent a friend request recently
thisDeviceMessageType = 'friend-request'; if (conversation.friendRequestTimerIsExpired()) {
} else { isMultiDeviceRequest = true;
// Throttle automated friend requests thisDeviceMessageType = 'friend-request';
this.successfulNumbers.push(devicePubKey); } else {
return null; // Throttle automated friend requests
this.successfulNumbers.push(devicePubKey);
return null;
}
}
// If we're not friends with our own device then we should become friends
if (isOurDevice && keysFound && !isFriends) {
conversation.setFriendRequestStatus(
window.friends.friendRequestStatusEnum.friends
);
} }
} }
} }

@ -411,7 +411,11 @@ MessageSender.prototype = {
const ourNumber = textsecure.storage.user.getNumber(); const ourNumber = textsecure.storage.user.getNumber();
numbers.forEach(number => { // Note: Since we're just doing independant tasks,
// using `async` in the `forEach` loop should be fine.
// If however we want to use the results from forEach then
// we would need to convert this to a Promise.all(numbers.map(...))
numbers.forEach(async number => {
// Note: if we are sending a private group message, we do our best to // Note: if we are sending a private group message, we do our best to
// ensure we have signal protocol sessions with every member, but if we // ensure we have signal protocol sessions with every member, but if we
// fail, let's at least send messages to those members with which we do: // fail, let's at least send messages to those members with which we do:
@ -420,9 +424,17 @@ MessageSender.prototype = {
s => s.number === number s => s.number === number
); );
let keysFound = false;
// If we don't have a session but we already have prekeys to
// start communication then we should use them
if (!haveSession && !options.isPublic) {
keysFound = await outgoing.getKeysForNumber(number, []);
}
if ( if (
number === ourNumber || number === ourNumber ||
haveSession || haveSession ||
keysFound ||
options.isPublic || options.isPublic ||
options.messageType === 'friend-request' options.messageType === 'friend-request'
) { ) {
@ -873,7 +885,14 @@ MessageSender.prototype = {
}, },
sendGroupProto(providedNumbers, proto, timestamp = Date.now(), options = {}) { sendGroupProto(providedNumbers, proto, timestamp = Date.now(), options = {}) {
if (providedNumbers.length === 0) { // We always assume that only primary device is a member in the group
const primaryDeviceKey =
window.storage.get('primaryDevicePubKey') ||
textsecure.storage.user.getNumber();
const numbers = providedNumbers.filter(
number => number !== primaryDeviceKey
);
if (numbers.length === 0) {
return Promise.resolve({ return Promise.resolve({
successfulNumbers: [], successfulNumbers: [],
failoverNumbers: [], failoverNumbers: [],
@ -883,7 +902,7 @@ MessageSender.prototype = {
}); });
} }
return new Promise((resolve, reject) => { const sendPromise = new Promise((resolve, reject) => {
const silent = true; const silent = true;
const callback = res => { const callback = res => {
res.dataMessage = proto.toArrayBuffer(); res.dataMessage = proto.toArrayBuffer();
@ -896,13 +915,20 @@ MessageSender.prototype = {
this.sendMessageProto( this.sendMessageProto(
timestamp, timestamp,
providedNumbers, numbers,
proto, proto,
callback, callback,
silent, silent,
options options
); );
}); });
return sendPromise.then(result => {
// Sync the group message to our other devices
const encoded = textsecure.protobuf.DataMessage.encode(proto);
this.sendSyncMessage(encoded, timestamp, null, null, [], [], options);
return result;
});
}, },
async getMessageProto( async getMessageProto(
@ -1087,8 +1113,11 @@ MessageSender.prototype = {
profileKey, profileKey,
options options
) { ) {
const me = textsecure.storage.user.getNumber(); // We always assume that only primary device is a member in the group
let numbers = groupNumbers.filter(number => number !== me); const primaryDeviceKey =
window.storage.get('primaryDevicePubKey') ||
textsecure.storage.user.getNumber();
let numbers = groupNumbers.filter(number => number !== primaryDeviceKey);
if (options.isPublic) { if (options.isPublic) {
numbers = [groupId]; numbers = [groupId];
} }
@ -1132,8 +1161,10 @@ MessageSender.prototype = {
proto.group.name = name; proto.group.name = name;
proto.group.members = members; proto.group.members = members;
const ourPK = textsecure.storage.user.getNumber(); const primaryDeviceKey =
proto.group.admins = [ourPK]; window.storage.get('primaryDevicePubKey') ||
textsecure.storage.user.getNumber();
proto.group.admins = [primaryDeviceKey];
return this.makeAttachmentPointer(avatar).then(attachment => { return this.makeAttachmentPointer(avatar).then(attachment => {
proto.group.avatar = attachment; proto.group.avatar = attachment;

@ -1241,6 +1241,7 @@
margin: 10px auto; margin: 10px auto;
padding: 5px 20px; padding: 5px 20px;
border-radius: 4px; border-radius: 4px;
word-break: break-word;
} }
.module-group-notification__contact { .module-group-notification__contact {

@ -71,7 +71,12 @@ export class EditProfileDialog extends React.Component<Props, State> {
const viewDefault = this.state.mode === 'default'; const viewDefault = this.state.mode === 'default';
const viewEdit = this.state.mode === 'edit'; const viewEdit = this.state.mode === 'edit';
const viewQR = this.state.mode === 'qr'; const viewQR = this.state.mode === 'qr';
const sessionID = window.textsecure.storage.user.getNumber();
/* tslint:disable:no-backbone-get-set-outside-model */
const sessionID =
window.textsecure.storage.get('primaryDevicePubKey') ||
window.textsecure.storage.user.getNumber();
/* tslint:enable:no-backbone-get-set-outside-model */
const backButton = const backButton =
viewEdit || viewQR viewEdit || viewQR

@ -38,7 +38,7 @@ export class GroupNotification extends React.Component<Props> {
key={`external-${contact.phoneNumber}`} key={`external-${contact.phoneNumber}`}
className="module-group-notification__contact" className="module-group-notification__contact"
> >
{contact.profileName} {contact.profileName || contact.phoneNumber}
</span> </span>
); );

@ -56,7 +56,7 @@ export class SessionClosableOverlay extends React.Component<Props, State> {
const conversationList = conversations.filter((conversation: any) => { const conversationList = conversations.filter((conversation: any) => {
return ( return (
!conversation.isOurConversation() && !conversation.isMe() &&
conversation.isPrivate() && conversation.isPrivate() &&
!conversation.isSecondaryDevice() && !conversation.isSecondaryDevice() &&
conversation.isFriend() conversation.isFriend()

Loading…
Cancel
Save