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.
643 lines
18 KiB
TypeScript
643 lines
18 KiB
TypeScript
import { SignalService } from './../protobuf';
|
|
import { removeFromCache } from './cache';
|
|
import { EnvelopePlus } from './types';
|
|
import { getEnvelopeId } from './common';
|
|
|
|
import { PubKey } from '../session/types';
|
|
import { handleMessageJob } from './queuedJob';
|
|
import { downloadAttachment } from './attachments';
|
|
import _ from 'lodash';
|
|
import { StringUtils, UserUtils } from '../session/utils';
|
|
import { getMessageQueue } from '../session';
|
|
import { ConversationController } from '../session/conversations';
|
|
import { handleClosedGroupControlMessage } from './closedGroups';
|
|
import { MessageModel } from '../models/message';
|
|
import { MessageModelType } from '../models/messageType';
|
|
import { getMessageBySender } from '../../ts/data/data';
|
|
import { ConversationModel, ConversationType } from '../models/conversation';
|
|
import { DeliveryReceiptMessage } from '../session/messages/outgoing/controlMessage/receipt/DeliveryReceiptMessage';
|
|
|
|
export async function updateProfile(
|
|
conversation: ConversationModel,
|
|
profile: SignalService.DataMessage.ILokiProfile,
|
|
profileKey: any
|
|
) {
|
|
const { dcodeIO, textsecure, Signal } = window;
|
|
|
|
// Retain old values unless changed:
|
|
const newProfile = conversation.get('profile') || {};
|
|
|
|
newProfile.displayName = profile.displayName;
|
|
|
|
if (profile.profilePicture) {
|
|
const prevPointer = conversation.get('avatarPointer');
|
|
const needsUpdate = !prevPointer || !_.isEqual(prevPointer, profile.profilePicture);
|
|
|
|
if (needsUpdate) {
|
|
const downloaded = await downloadAttachment({
|
|
url: profile.profilePicture,
|
|
isRaw: true,
|
|
});
|
|
|
|
// null => use placeholder with color and first letter
|
|
let path = null;
|
|
if (profileKey) {
|
|
// Convert profileKey to ArrayBuffer, if needed
|
|
const encoding = typeof profileKey === 'string' ? 'base64' : null;
|
|
try {
|
|
const profileKeyArrayBuffer = dcodeIO.ByteBuffer.wrap(
|
|
profileKey,
|
|
encoding
|
|
).toArrayBuffer();
|
|
const decryptedData = await textsecure.crypto.decryptProfile(
|
|
downloaded.data,
|
|
profileKeyArrayBuffer
|
|
);
|
|
const upgraded = await Signal.Migrations.processNewAttachment({
|
|
...downloaded,
|
|
data: decryptedData,
|
|
});
|
|
// Only update the convo if the download and decrypt is a success
|
|
conversation.set('avatarPointer', profile.profilePicture);
|
|
conversation.set('profileKey', profileKey);
|
|
({ path } = upgraded);
|
|
} catch (e) {
|
|
window.log.error(`Could not decrypt profile image: ${e}`);
|
|
}
|
|
}
|
|
newProfile.avatar = path;
|
|
}
|
|
} else {
|
|
newProfile.avatar = null;
|
|
}
|
|
|
|
const conv = await ConversationController.getInstance().getOrCreateAndWait(
|
|
conversation.id,
|
|
ConversationType.PRIVATE
|
|
);
|
|
await conv.setLokiProfile(newProfile);
|
|
}
|
|
|
|
function cleanAttachment(attachment: any) {
|
|
return {
|
|
..._.omit(attachment, 'thumbnail'),
|
|
id: attachment.id.toString(),
|
|
key: attachment.key ? StringUtils.decode(attachment.key, 'base64') : null,
|
|
digest:
|
|
attachment.digest && attachment.digest.length > 0
|
|
? StringUtils.decode(attachment.digest, 'base64')
|
|
: null,
|
|
};
|
|
}
|
|
|
|
function cleanAttachments(decrypted: any) {
|
|
const { quote, group } = decrypted;
|
|
|
|
// Here we go from binary to string/base64 in all AttachmentPointer digest/key fields
|
|
|
|
if (group && group.type === SignalService.GroupContext.Type.UPDATE) {
|
|
if (group.avatar !== null) {
|
|
group.avatar = cleanAttachment(group.avatar);
|
|
}
|
|
}
|
|
|
|
decrypted.attachments = (decrypted.attachments || []).map(cleanAttachment);
|
|
decrypted.preview = (decrypted.preview || []).map((item: any) => {
|
|
const { image } = item;
|
|
|
|
if (!image) {
|
|
return item;
|
|
}
|
|
|
|
return {
|
|
...item,
|
|
image: cleanAttachment(image),
|
|
};
|
|
});
|
|
|
|
decrypted.contact = (decrypted.contact || []).map((item: any) => {
|
|
const { avatar } = item;
|
|
|
|
if (!avatar || !avatar.avatar) {
|
|
return item;
|
|
}
|
|
|
|
return {
|
|
...item,
|
|
avatar: {
|
|
...item.avatar,
|
|
avatar: cleanAttachment(item.avatar.avatar),
|
|
},
|
|
};
|
|
});
|
|
|
|
if (quote) {
|
|
if (quote.id) {
|
|
quote.id = _.toNumber(quote.id);
|
|
}
|
|
|
|
quote.attachments = (quote.attachments || []).map((item: any) => {
|
|
const { thumbnail } = item;
|
|
|
|
if (!thumbnail || thumbnail.length === 0) {
|
|
return item;
|
|
}
|
|
|
|
return {
|
|
...item,
|
|
thumbnail: cleanAttachment(item.thumbnail),
|
|
};
|
|
});
|
|
}
|
|
}
|
|
|
|
export async function processDecrypted(
|
|
envelope: EnvelopePlus,
|
|
decrypted: SignalService.IDataMessage
|
|
) {
|
|
/* tslint:disable:no-bitwise */
|
|
const FLAGS = SignalService.DataMessage.Flags;
|
|
|
|
// Now that its decrypted, validate the message and clean it up for consumer
|
|
// processing
|
|
// Note that messages may (generally) only perform one action and we ignore remaining
|
|
// fields after the first action.
|
|
|
|
if (decrypted.flags == null) {
|
|
decrypted.flags = 0;
|
|
}
|
|
if (decrypted.expireTimer == null) {
|
|
decrypted.expireTimer = 0;
|
|
}
|
|
if (decrypted.flags & FLAGS.EXPIRATION_TIMER_UPDATE) {
|
|
decrypted.body = '';
|
|
decrypted.attachments = [];
|
|
} else if (decrypted.flags !== 0) {
|
|
throw new Error('Unknown flags in message');
|
|
}
|
|
|
|
if (decrypted.group) {
|
|
// decrypted.group.id = new TextDecoder('utf-8').decode(decrypted.group.id);
|
|
|
|
switch (decrypted.group.type) {
|
|
case SignalService.GroupContext.Type.UPDATE:
|
|
decrypted.body = '';
|
|
decrypted.attachments = [];
|
|
break;
|
|
case SignalService.GroupContext.Type.QUIT:
|
|
decrypted.body = '';
|
|
decrypted.attachments = [];
|
|
break;
|
|
case SignalService.GroupContext.Type.DELIVER:
|
|
decrypted.group.name = null;
|
|
decrypted.group.members = [];
|
|
decrypted.group.avatar = null;
|
|
break;
|
|
case SignalService.GroupContext.Type.REQUEST_INFO:
|
|
decrypted.body = '';
|
|
decrypted.attachments = [];
|
|
break;
|
|
default:
|
|
await removeFromCache(envelope);
|
|
throw new Error('Unknown group message type');
|
|
}
|
|
}
|
|
|
|
const attachmentCount = decrypted?.attachments?.length || 0;
|
|
const ATTACHMENT_MAX = 32;
|
|
if (attachmentCount > ATTACHMENT_MAX) {
|
|
await removeFromCache(envelope);
|
|
throw new Error(
|
|
`Too many attachments: ${attachmentCount} included in one message, max is ${ATTACHMENT_MAX}`
|
|
);
|
|
}
|
|
|
|
cleanAttachments(decrypted);
|
|
|
|
return decrypted as SignalService.DataMessage;
|
|
/* tslint:disable:no-bitwise */
|
|
}
|
|
|
|
export function isMessageEmpty(message: SignalService.DataMessage) {
|
|
const { flags, body, attachments, group, quote, contact, preview, groupInvitation } = message;
|
|
|
|
return (
|
|
!flags &&
|
|
// FIXME remove this hack to drop auto friend requests messages in a few weeks 15/07/2020
|
|
isBodyEmpty(body) &&
|
|
_.isEmpty(attachments) &&
|
|
_.isEmpty(group) &&
|
|
_.isEmpty(quote) &&
|
|
_.isEmpty(contact) &&
|
|
_.isEmpty(preview) &&
|
|
_.isEmpty(groupInvitation)
|
|
);
|
|
}
|
|
|
|
function isBodyEmpty(body: string) {
|
|
return _.isEmpty(body);
|
|
}
|
|
|
|
/**
|
|
* We have a few origins possible
|
|
* - if the message is from a private conversation with a friend and he wrote to us,
|
|
* the conversation to add the message to is our friend pubkey, so envelope.source
|
|
* - if the message is from a medium group conversation
|
|
* * envelope.source is the medium group pubkey
|
|
* * envelope.senderIdentity is the author pubkey (the one who sent the message)
|
|
* - at last, if the message is a syncMessage,
|
|
* * envelope.source is our pubkey (our other device has the same pubkey as us)
|
|
* * dataMessage.syncTarget is either the group public key OR the private conversation this message is about.
|
|
*/
|
|
export async function handleDataMessage(
|
|
envelope: EnvelopePlus,
|
|
dataMessage: SignalService.IDataMessage
|
|
): Promise<void> {
|
|
// we handle group updates from our other devices in handleClosedGroupControlMessage()
|
|
if (dataMessage.closedGroupControlMessage) {
|
|
await handleClosedGroupControlMessage(
|
|
envelope,
|
|
dataMessage.closedGroupControlMessage as SignalService.DataMessage.ClosedGroupControlMessage
|
|
);
|
|
return;
|
|
}
|
|
|
|
const message = await processDecrypted(envelope, dataMessage);
|
|
const source = dataMessage.syncTarget || envelope.source;
|
|
const senderPubKey = envelope.senderIdentity || envelope.source;
|
|
const isMe = UserUtils.isUsFromCache(senderPubKey);
|
|
const isSyncMessage = Boolean(dataMessage.syncTarget?.length);
|
|
|
|
window.log.info(`Handle dataMessage from ${source} `);
|
|
|
|
if (isSyncMessage && !isMe) {
|
|
window.log.warn('Got a sync message from someone else than me. Dropping it.');
|
|
return removeFromCache(envelope);
|
|
} else if (isSyncMessage && dataMessage.syncTarget) {
|
|
// override the envelope source
|
|
envelope.source = dataMessage.syncTarget;
|
|
}
|
|
|
|
const senderConversation = await ConversationController.getInstance().getOrCreateAndWait(
|
|
senderPubKey,
|
|
ConversationType.PRIVATE
|
|
);
|
|
|
|
// Check if we need to update any profile names
|
|
if (!isMe && senderConversation && message.profile) {
|
|
await updateProfile(senderConversation, message.profile, message.profileKey);
|
|
}
|
|
if (isMessageEmpty(message)) {
|
|
window.log.warn(`Message ${getEnvelopeId(envelope)} ignored; it was empty`);
|
|
return removeFromCache(envelope);
|
|
}
|
|
|
|
const ev: any = {};
|
|
if (isMe) {
|
|
// Data messages for medium groups don't arrive as sync messages. Instead,
|
|
// linked devices poll for group messages independently, thus they need
|
|
// to recognise some of those messages at their own.
|
|
ev.type = 'sent';
|
|
} else {
|
|
ev.type = 'message';
|
|
}
|
|
|
|
if (envelope.senderIdentity) {
|
|
message.group = {
|
|
id: envelope.source as any, // FIXME Uint8Array vs string
|
|
};
|
|
}
|
|
|
|
ev.confirm = () => removeFromCache(envelope);
|
|
ev.data = {
|
|
source: senderPubKey,
|
|
destination: isMe ? message.syncTarget : undefined,
|
|
sourceDevice: 1,
|
|
timestamp: _.toNumber(envelope.timestamp),
|
|
receivedAt: envelope.receivedAt,
|
|
message,
|
|
};
|
|
|
|
await handleMessageEvent(ev); // dataMessage
|
|
}
|
|
|
|
type MessageDuplicateSearchType = {
|
|
body: string;
|
|
id: string;
|
|
timestamp: number;
|
|
serverId?: number;
|
|
};
|
|
|
|
export type MessageId = {
|
|
source: string;
|
|
serverId: number;
|
|
sourceDevice: number;
|
|
timestamp: number;
|
|
message: MessageDuplicateSearchType;
|
|
};
|
|
const PUBLICCHAT_MIN_TIME_BETWEEN_DUPLICATE_MESSAGES = 10 * 1000; // 10s
|
|
|
|
export async function isMessageDuplicate({
|
|
source,
|
|
sourceDevice,
|
|
timestamp,
|
|
message,
|
|
serverId,
|
|
}: MessageId) {
|
|
const { Errors } = window.Signal.Types;
|
|
// serverId is only used for opengroupv2
|
|
try {
|
|
const result = await getMessageBySender({
|
|
source,
|
|
sourceDevice,
|
|
sent_at: timestamp,
|
|
});
|
|
|
|
if (!result) {
|
|
return false;
|
|
}
|
|
const filteredResult = [result].filter((m: any) => m.attributes.body === message.body);
|
|
if (serverId) {
|
|
return filteredResult.some(m => isDuplicateServerId(m, { ...message, serverId }, source));
|
|
}
|
|
return filteredResult.some(m => isDuplicate(m, message, source));
|
|
} catch (error) {
|
|
window.log.error('isMessageDuplicate error:', Errors.toLogFormat(error));
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This function is to be used to check for duplicates for open group v2 messages.
|
|
* It just check that the sender and the serverId of a received and an already saved messages are the same
|
|
*/
|
|
export const isDuplicateServerId = (
|
|
m: MessageModel,
|
|
testedMessage: MessageDuplicateSearchType,
|
|
source: string
|
|
) => {
|
|
// The username in this case is the users pubKey
|
|
const sameUsername = m.attributes.source === source;
|
|
// testedMessage.id is needed as long as we support opengroupv1
|
|
const sameServerId =
|
|
m.attributes.serverId !== undefined &&
|
|
(testedMessage.serverId || testedMessage.id) === m.attributes.serverId;
|
|
|
|
return sameUsername && sameServerId;
|
|
};
|
|
|
|
export const isDuplicate = (
|
|
m: MessageModel,
|
|
testedMessage: MessageDuplicateSearchType,
|
|
source: string
|
|
) => {
|
|
// The username in this case is the users pubKey
|
|
const sameUsername = m.attributes.source === source;
|
|
// testedMessage.id is needed as long as we support opengroupv1
|
|
const sameServerId =
|
|
m.attributes.serverId !== undefined &&
|
|
(testedMessage.serverId || testedMessage.id) === m.attributes.serverId;
|
|
const sameText = m.attributes.body === testedMessage.body;
|
|
// Don't filter out messages that are too far apart from each other
|
|
const timestampsSimilar =
|
|
Math.abs(m.attributes.sent_at - testedMessage.timestamp) <=
|
|
PUBLICCHAT_MIN_TIME_BETWEEN_DUPLICATE_MESSAGES;
|
|
|
|
return sameUsername && sameText && (timestampsSimilar || sameServerId);
|
|
};
|
|
|
|
async function handleProfileUpdate(
|
|
profileKeyBuffer: Uint8Array,
|
|
convoId: string,
|
|
convoType: ConversationType,
|
|
isIncoming: boolean
|
|
) {
|
|
const profileKey = StringUtils.decode(profileKeyBuffer, 'base64');
|
|
|
|
if (!isIncoming) {
|
|
// We update our own profileKey if it's different from what we have
|
|
const ourNumber = UserUtils.getOurPubKeyStrFromCache();
|
|
const me = ConversationController.getInstance().getOrCreate(
|
|
ourNumber,
|
|
ConversationType.PRIVATE
|
|
);
|
|
|
|
// Will do the save for us if needed
|
|
await me.setProfileKey(profileKey);
|
|
} else {
|
|
const sender = await ConversationController.getInstance().getOrCreateAndWait(
|
|
convoId,
|
|
ConversationType.PRIVATE
|
|
);
|
|
|
|
// Will do the save for us
|
|
await sender.setProfileKey(profileKey);
|
|
}
|
|
}
|
|
|
|
export interface MessageCreationData {
|
|
timestamp: number;
|
|
isPublic: boolean;
|
|
receivedAt: number;
|
|
sourceDevice: number; // always 1 isn't it?
|
|
source: string;
|
|
serverId: number;
|
|
message: any;
|
|
serverTimestamp: any;
|
|
|
|
// Needed for synced outgoing messages
|
|
unidentifiedStatus: any; // ???
|
|
expirationStartTimestamp: any; // ???
|
|
destination: string;
|
|
}
|
|
|
|
export function initIncomingMessage(data: MessageCreationData): MessageModel {
|
|
const {
|
|
timestamp,
|
|
isPublic,
|
|
receivedAt,
|
|
sourceDevice,
|
|
source,
|
|
serverId,
|
|
message,
|
|
serverTimestamp,
|
|
} = data;
|
|
|
|
const messageGroupId = message?.group?.id;
|
|
let groupId = messageGroupId && messageGroupId.length > 0 ? messageGroupId : null;
|
|
|
|
if (groupId) {
|
|
groupId = PubKey.removeTextSecurePrefixIfNeeded(groupId);
|
|
}
|
|
|
|
const messageData: any = {
|
|
source,
|
|
sourceDevice,
|
|
serverId, // + (not present below in `createSentMessage`)
|
|
sent_at: timestamp,
|
|
serverTimestamp,
|
|
received_at: receivedAt || Date.now(),
|
|
conversationId: groupId ?? source,
|
|
type: 'incoming',
|
|
direction: 'incoming', // +
|
|
unread: 1, // +
|
|
isPublic, // +
|
|
};
|
|
|
|
return new MessageModel(messageData);
|
|
}
|
|
|
|
function createSentMessage(data: MessageCreationData): MessageModel {
|
|
const now = Date.now();
|
|
|
|
const {
|
|
timestamp,
|
|
serverTimestamp,
|
|
serverId,
|
|
isPublic,
|
|
receivedAt,
|
|
sourceDevice,
|
|
expirationStartTimestamp,
|
|
destination,
|
|
message,
|
|
} = data;
|
|
|
|
const sentSpecificFields = {
|
|
sent_to: [],
|
|
sent: true,
|
|
expirationStartTimestamp: Math.min(expirationStartTimestamp || data.timestamp || now, now),
|
|
};
|
|
|
|
const messageGroupId = message?.group?.id;
|
|
let groupId = messageGroupId && messageGroupId.length > 0 ? messageGroupId : null;
|
|
|
|
if (groupId) {
|
|
groupId = PubKey.removeTextSecurePrefixIfNeeded(groupId);
|
|
}
|
|
|
|
const messageData = {
|
|
source: UserUtils.getOurPubKeyStrFromCache(),
|
|
sourceDevice,
|
|
serverTimestamp,
|
|
serverId,
|
|
sent_at: timestamp,
|
|
received_at: isPublic ? receivedAt : now,
|
|
isPublic,
|
|
conversationId: groupId ?? destination,
|
|
type: 'outgoing' as MessageModelType,
|
|
...sentSpecificFields,
|
|
};
|
|
|
|
return new MessageModel(messageData);
|
|
}
|
|
|
|
export function createMessage(data: MessageCreationData, isIncoming: boolean): MessageModel {
|
|
if (isIncoming) {
|
|
return initIncomingMessage(data);
|
|
} else {
|
|
return createSentMessage(data);
|
|
}
|
|
}
|
|
|
|
function sendDeliveryReceipt(source: string, timestamp: any) {
|
|
const receiptMessage = new DeliveryReceiptMessage({
|
|
timestamp: Date.now(),
|
|
timestamps: [timestamp],
|
|
});
|
|
const device = new PubKey(source);
|
|
void getMessageQueue().sendToPubKey(device, receiptMessage);
|
|
}
|
|
|
|
export interface MessageEvent {
|
|
data: any;
|
|
type: string;
|
|
confirm: () => void;
|
|
}
|
|
|
|
// tslint:disable:cyclomatic-complexity max-func-body-length */
|
|
export async function handleMessageEvent(event: MessageEvent): Promise<void> {
|
|
const { data, confirm } = event;
|
|
|
|
const isIncoming = event.type === 'message';
|
|
|
|
if (!data || !data.message) {
|
|
window.log.warn('Invalid data passed to handleMessageEvent.', event);
|
|
confirm();
|
|
return;
|
|
}
|
|
|
|
const { message, destination } = data;
|
|
|
|
let { source } = data;
|
|
|
|
const isGroupMessage = Boolean(message.group);
|
|
|
|
const type = isGroupMessage ? ConversationType.GROUP : ConversationType.PRIVATE;
|
|
|
|
let conversationId = isIncoming ? source : destination || source; // for synced message
|
|
if (!conversationId) {
|
|
window.log.error('We cannot handle a message without a conversationId');
|
|
confirm();
|
|
return;
|
|
}
|
|
if (message.profileKey?.length) {
|
|
await handleProfileUpdate(message.profileKey, conversationId, type, isIncoming);
|
|
}
|
|
|
|
const msg = createMessage(data, isIncoming);
|
|
|
|
// if the message is `sent` (from secondary device) we have to set the sender manually... (at least for now)
|
|
source = source || msg.get('source');
|
|
|
|
if (await isMessageDuplicate(data)) {
|
|
window.log.info('Received duplicate message. Dropping it.');
|
|
confirm();
|
|
return;
|
|
}
|
|
|
|
const isOurDevice = UserUtils.isUsFromCache(source);
|
|
|
|
const shouldSendReceipt = isIncoming && !isGroupMessage && !isOurDevice;
|
|
|
|
if (shouldSendReceipt) {
|
|
sendDeliveryReceipt(source, data.timestamp);
|
|
}
|
|
|
|
// Conversation Id is:
|
|
// - primarySource if it is an incoming DM message,
|
|
// - destination if it is an outgoing message,
|
|
// - group.id if it is a group message
|
|
if (isGroupMessage) {
|
|
// remove the prefix from the source object so this is correct for all other
|
|
message.group.id = PubKey.removeTextSecurePrefixIfNeeded(message.group.id);
|
|
|
|
conversationId = message.group.id;
|
|
}
|
|
|
|
if (!conversationId) {
|
|
window.log.warn('Invalid conversation id for incoming message', conversationId);
|
|
}
|
|
const ourNumber = UserUtils.getOurPubKeyStrFromCache();
|
|
|
|
// =========================================
|
|
|
|
if (!isGroupMessage && source !== ourNumber) {
|
|
// Ignore auth from our devices
|
|
conversationId = source;
|
|
}
|
|
|
|
const conversation = await ConversationController.getInstance().getOrCreateAndWait(
|
|
conversationId,
|
|
type
|
|
);
|
|
|
|
if (!conversation) {
|
|
window.log.warn('Skipping handleJob for unknown convo: ', conversationId);
|
|
return;
|
|
}
|
|
|
|
conversation.queueJob(async () => {
|
|
await handleMessageJob(msg, conversation, message, ourNumber, confirm, source);
|
|
});
|
|
}
|