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.
286 lines
9.2 KiB
JavaScript
286 lines
9.2 KiB
JavaScript
/* global window, textsecure, dcodeIO, StringView, ConversationController */
|
|
|
|
// eslint-disable-next-line func-names
|
|
(function() {
|
|
window.libloki = window.libloki || {};
|
|
|
|
async function sendBackgroundMessage(pubKey) {
|
|
return sendOnlineBroadcastMessage(pubKey);
|
|
}
|
|
|
|
async function sendOnlineBroadcastMessage(pubKey, isPing = false) {
|
|
const authorisation = await window.libloki.storage.getGrantAuthorisationForSecondaryPubKey(
|
|
pubKey
|
|
);
|
|
if (authorisation && authorisation.primaryDevicePubKey !== pubKey) {
|
|
sendOnlineBroadcastMessage(authorisation.primaryDevicePubKey);
|
|
return;
|
|
}
|
|
const p2pAddress = null;
|
|
const p2pPort = null;
|
|
// We result loki address message for sending "background" messages
|
|
const type = textsecure.protobuf.LokiAddressMessage.Type.HOST_UNREACHABLE;
|
|
|
|
const lokiAddressMessage = new textsecure.protobuf.LokiAddressMessage({
|
|
p2pAddress,
|
|
p2pPort,
|
|
type,
|
|
});
|
|
const content = new textsecure.protobuf.Content({
|
|
lokiAddressMessage,
|
|
});
|
|
|
|
const options = { messageType: 'onlineBroadcast', isPing };
|
|
// Send a empty message with information about how to contact us directly
|
|
const outgoingMessage = new textsecure.OutgoingMessage(
|
|
null, // server
|
|
Date.now(), // timestamp,
|
|
[pubKey], // numbers
|
|
content, // message
|
|
true, // silent
|
|
() => null, // callback
|
|
options
|
|
);
|
|
await outgoingMessage.sendToNumber(pubKey);
|
|
}
|
|
|
|
function createPairingAuthorisationProtoMessage({
|
|
primaryDevicePubKey,
|
|
secondaryDevicePubKey,
|
|
requestSignature,
|
|
grantSignature,
|
|
}) {
|
|
if (!primaryDevicePubKey || !secondaryDevicePubKey || !requestSignature) {
|
|
throw new Error(
|
|
'createPairingAuthorisationProtoMessage: pubkeys missing'
|
|
);
|
|
}
|
|
if (requestSignature.constructor !== ArrayBuffer) {
|
|
throw new Error(
|
|
'createPairingAuthorisationProtoMessage expects a signature as ArrayBuffer'
|
|
);
|
|
}
|
|
if (grantSignature && grantSignature.constructor !== ArrayBuffer) {
|
|
throw new Error(
|
|
'createPairingAuthorisationProtoMessage expects a signature as ArrayBuffer'
|
|
);
|
|
}
|
|
return new textsecure.protobuf.PairingAuthorisationMessage({
|
|
requestSignature: new Uint8Array(requestSignature),
|
|
grantSignature: grantSignature ? new Uint8Array(grantSignature) : null,
|
|
primaryDevicePubKey,
|
|
secondaryDevicePubKey,
|
|
});
|
|
}
|
|
|
|
function sendUnpairingMessageToSecondary(pubKey) {
|
|
const flags = textsecure.protobuf.DataMessage.Flags.UNPAIRING_REQUEST;
|
|
const dataMessage = new textsecure.protobuf.DataMessage({
|
|
flags,
|
|
});
|
|
const content = new textsecure.protobuf.Content({
|
|
dataMessage,
|
|
});
|
|
const options = { messageType: 'device-unpairing' };
|
|
const outgoingMessage = new textsecure.OutgoingMessage(
|
|
null, // server
|
|
Date.now(), // timestamp,
|
|
[pubKey], // numbers
|
|
content, // message
|
|
true, // silent
|
|
() => null, // callback
|
|
options
|
|
);
|
|
return outgoingMessage.sendToNumber(pubKey);
|
|
}
|
|
// Serialise as <Element0.length><Element0><Element1.length><Element1>...
|
|
// This is an implementation of the reciprocal of contacts_parser.js
|
|
function serialiseByteBuffers(buffers) {
|
|
const result = new dcodeIO.ByteBuffer();
|
|
buffers.forEach(buffer => {
|
|
// bytebuffer container expands and increments
|
|
// offset automatically
|
|
result.writeInt32(buffer.limit);
|
|
result.append(buffer);
|
|
});
|
|
result.limit = result.offset;
|
|
result.reset();
|
|
return result;
|
|
}
|
|
async function createContactSyncProtoMessage(conversations) {
|
|
// Extract required contacts information out of conversations
|
|
const sessionContacts = conversations.filter(
|
|
c => c.isPrivate() && !c.isSecondaryDevice()
|
|
);
|
|
|
|
if (sessionContacts.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const rawContacts = await Promise.all(
|
|
sessionContacts.map(async conversation => {
|
|
const profile = conversation.getLokiProfile();
|
|
const number = conversation.getNumber();
|
|
const name = profile
|
|
? profile.displayName
|
|
: conversation.getProfileName();
|
|
const status = await conversation.safeGetVerified();
|
|
const protoState = textsecure.storage.protocol.convertVerifiedStatusToProtoState(
|
|
status
|
|
);
|
|
const verified = new textsecure.protobuf.Verified({
|
|
state: protoState,
|
|
destination: number,
|
|
identityKey: StringView.hexToArrayBuffer(number),
|
|
});
|
|
return {
|
|
name,
|
|
verified,
|
|
number,
|
|
nickname: conversation.getNickname(),
|
|
blocked: conversation.isBlocked(),
|
|
expireTimer: conversation.get('expireTimer'),
|
|
};
|
|
})
|
|
);
|
|
// Convert raw contacts to an array of buffers
|
|
const contactDetails = rawContacts
|
|
.filter(x => x.number !== textsecure.storage.user.getNumber())
|
|
.map(x => new textsecure.protobuf.ContactDetails(x))
|
|
.map(x => x.encode());
|
|
// Serialise array of byteBuffers into 1 byteBuffer
|
|
const byteBuffer = serialiseByteBuffers(contactDetails);
|
|
const data = new Uint8Array(byteBuffer.toArrayBuffer());
|
|
const contacts = new textsecure.protobuf.SyncMessage.Contacts({
|
|
data,
|
|
});
|
|
const syncMessage = new textsecure.protobuf.SyncMessage({
|
|
contacts,
|
|
});
|
|
return syncMessage;
|
|
}
|
|
function createGroupSyncProtoMessage(conversations) {
|
|
// We only want to sync across closed groups that we haven't left
|
|
const sessionGroups = conversations.filter(
|
|
c => c.isClosedGroup() && !c.get('left') && c.isFriend()
|
|
);
|
|
|
|
if (sessionGroups.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const rawGroups = sessionGroups.map(conversation => ({
|
|
id: window.Signal.Crypto.bytesFromString(conversation.id),
|
|
name: conversation.get('name'),
|
|
members: conversation.get('members') || [],
|
|
blocked: conversation.isBlocked(),
|
|
expireTimer: conversation.get('expireTimer'),
|
|
admins: conversation.get('groupAdmins') || [],
|
|
}));
|
|
|
|
// Convert raw groups to an array of buffers
|
|
const groupDetails = rawGroups
|
|
.map(x => new textsecure.protobuf.GroupDetails(x))
|
|
.map(x => x.encode());
|
|
// Serialise array of byteBuffers into 1 byteBuffer
|
|
const byteBuffer = serialiseByteBuffers(groupDetails);
|
|
const data = new Uint8Array(byteBuffer.toArrayBuffer());
|
|
const groups = new textsecure.protobuf.SyncMessage.Groups({
|
|
data,
|
|
});
|
|
const syncMessage = new textsecure.protobuf.SyncMessage({
|
|
groups,
|
|
});
|
|
return syncMessage;
|
|
}
|
|
function createOpenGroupsSyncProtoMessage(conversations) {
|
|
// We only want to sync across open groups that we haven't left
|
|
const sessionOpenGroups = conversations.filter(
|
|
c => c.isPublic() && !c.isRss() && !c.get('left')
|
|
);
|
|
|
|
if (sessionOpenGroups.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
const openGroups = sessionOpenGroups.map(
|
|
conversation =>
|
|
new textsecure.protobuf.SyncMessage.OpenGroupDetails({
|
|
url: conversation.id.split('@').pop(),
|
|
channelId: conversation.get('channelId'),
|
|
})
|
|
);
|
|
|
|
const syncMessage = new textsecure.protobuf.SyncMessage({
|
|
openGroups,
|
|
});
|
|
return syncMessage;
|
|
}
|
|
async function sendPairingAuthorisation(authorisation, recipientPubKey) {
|
|
const pairingAuthorisation = createPairingAuthorisationProtoMessage(
|
|
authorisation
|
|
);
|
|
const ourNumber = textsecure.storage.user.getNumber();
|
|
const ourConversation = await ConversationController.getOrCreateAndWait(
|
|
ourNumber,
|
|
'private'
|
|
);
|
|
const content = new textsecure.protobuf.Content({
|
|
pairingAuthorisation,
|
|
});
|
|
const isGrant = authorisation.primaryDevicePubKey === ourNumber;
|
|
if (isGrant) {
|
|
// Send profile name to secondary device
|
|
const lokiProfile = ourConversation.getLokiProfile();
|
|
// profile.avatar is the path to the local image
|
|
// replace with the avatar URL
|
|
const avatarPointer = ourConversation.get('avatarPointer');
|
|
lokiProfile.avatar = avatarPointer;
|
|
const profile = new textsecure.protobuf.DataMessage.LokiProfile(
|
|
lokiProfile
|
|
);
|
|
const profileKey = window.storage.get('profileKey');
|
|
const dataMessage = new textsecure.protobuf.DataMessage({
|
|
profile,
|
|
profileKey,
|
|
});
|
|
content.dataMessage = dataMessage;
|
|
}
|
|
// Send
|
|
const options = { messageType: 'pairing-request' };
|
|
const p = new Promise((resolve, reject) => {
|
|
const timestamp = Date.now();
|
|
|
|
const outgoingMessage = new textsecure.OutgoingMessage(
|
|
null, // server
|
|
timestamp,
|
|
[recipientPubKey], // numbers
|
|
content, // message
|
|
true, // silent
|
|
result => {
|
|
// callback
|
|
if (result.errors.length > 0) {
|
|
reject(result.errors[0]);
|
|
} else {
|
|
resolve();
|
|
}
|
|
},
|
|
options
|
|
);
|
|
outgoingMessage.sendToNumber(recipientPubKey);
|
|
});
|
|
return p;
|
|
}
|
|
|
|
window.libloki.api = {
|
|
sendBackgroundMessage,
|
|
sendOnlineBroadcastMessage,
|
|
sendPairingAuthorisation,
|
|
createPairingAuthorisationProtoMessage,
|
|
sendUnpairingMessageToSecondary,
|
|
createContactSyncProtoMessage,
|
|
createGroupSyncProtoMessage,
|
|
createOpenGroupsSyncProtoMessage,
|
|
};
|
|
})();
|