136 lines
4 KiB
TypeScript
136 lines
4 KiB
TypeScript
|
// Copyright 2021 Signal Messenger, LLC
|
||
|
// SPDX-License-Identifier: AGPL-3.0-only
|
||
|
|
||
|
import { ConversationAttributesType } from '../model-types.d';
|
||
|
import { SendMetadataType, SendOptionsType } from '../textsecure/SendMessage';
|
||
|
import { arrayBufferToBase64, getRandomBytes } from '../Crypto';
|
||
|
import { getConversationMembers } from './getConversationMembers';
|
||
|
import { isDirectConversation, isMe } from './whatTypeOfConversation';
|
||
|
import { missingCaseError } from './missingCaseError';
|
||
|
import { senderCertificateService } from '../services/senderCertificate';
|
||
|
import {
|
||
|
PhoneNumberSharingMode,
|
||
|
parsePhoneNumberSharingMode,
|
||
|
} from './phoneNumberSharingMode';
|
||
|
import {
|
||
|
SenderCertificateMode,
|
||
|
SerializedCertificateType,
|
||
|
} from '../textsecure/OutgoingMessage';
|
||
|
|
||
|
const SEALED_SENDER = {
|
||
|
UNKNOWN: 0,
|
||
|
ENABLED: 1,
|
||
|
DISABLED: 2,
|
||
|
UNRESTRICTED: 3,
|
||
|
};
|
||
|
|
||
|
export async function getSendOptions(
|
||
|
conversationAttrs: ConversationAttributesType,
|
||
|
options: { syncMessage?: boolean } = {}
|
||
|
): Promise<SendOptionsType> {
|
||
|
const { syncMessage } = options;
|
||
|
|
||
|
if (!isDirectConversation(conversationAttrs)) {
|
||
|
const contactCollection = getConversationMembers(conversationAttrs);
|
||
|
const sendMetadata: SendMetadataType = {};
|
||
|
await Promise.all(
|
||
|
contactCollection.map(async contactAttrs => {
|
||
|
const conversation = window.ConversationController.get(contactAttrs.id);
|
||
|
if (!conversation) {
|
||
|
return;
|
||
|
}
|
||
|
const {
|
||
|
sendMetadata: conversationSendMetadata,
|
||
|
} = await conversation.getSendOptions(options);
|
||
|
Object.assign(sendMetadata, conversationSendMetadata || {});
|
||
|
})
|
||
|
);
|
||
|
return { sendMetadata };
|
||
|
}
|
||
|
|
||
|
const { accessKey, sealedSender } = conversationAttrs;
|
||
|
|
||
|
// We never send sync messages as sealed sender
|
||
|
if (syncMessage && isMe(conversationAttrs)) {
|
||
|
return {
|
||
|
sendMetadata: undefined,
|
||
|
};
|
||
|
}
|
||
|
|
||
|
const { e164, uuid } = conversationAttrs;
|
||
|
|
||
|
const senderCertificate = await getSenderCertificateForDirectConversation(
|
||
|
conversationAttrs
|
||
|
);
|
||
|
|
||
|
// If we've never fetched user's profile, we default to what we have
|
||
|
if (sealedSender === SEALED_SENDER.UNKNOWN) {
|
||
|
const identifierData = {
|
||
|
accessKey: accessKey || arrayBufferToBase64(getRandomBytes(16)),
|
||
|
senderCertificate,
|
||
|
};
|
||
|
return {
|
||
|
sendMetadata: {
|
||
|
...(e164 ? { [e164]: identifierData } : {}),
|
||
|
...(uuid ? { [uuid]: identifierData } : {}),
|
||
|
},
|
||
|
};
|
||
|
}
|
||
|
|
||
|
if (sealedSender === SEALED_SENDER.DISABLED) {
|
||
|
return {
|
||
|
sendMetadata: undefined,
|
||
|
};
|
||
|
}
|
||
|
|
||
|
const identifierData = {
|
||
|
accessKey:
|
||
|
accessKey && sealedSender === SEALED_SENDER.ENABLED
|
||
|
? accessKey
|
||
|
: arrayBufferToBase64(getRandomBytes(16)),
|
||
|
senderCertificate,
|
||
|
};
|
||
|
|
||
|
return {
|
||
|
sendMetadata: {
|
||
|
...(e164 ? { [e164]: identifierData } : {}),
|
||
|
...(uuid ? { [uuid]: identifierData } : {}),
|
||
|
},
|
||
|
};
|
||
|
}
|
||
|
|
||
|
function getSenderCertificateForDirectConversation(
|
||
|
conversationAttrs: ConversationAttributesType
|
||
|
): Promise<undefined | SerializedCertificateType> {
|
||
|
if (!isDirectConversation(conversationAttrs)) {
|
||
|
throw new Error(
|
||
|
'getSenderCertificateForDirectConversation should only be called for direct conversations'
|
||
|
);
|
||
|
}
|
||
|
|
||
|
const phoneNumberSharingMode = parsePhoneNumberSharingMode(
|
||
|
window.storage.get('phoneNumberSharingMode')
|
||
|
);
|
||
|
|
||
|
let certificateMode: SenderCertificateMode;
|
||
|
switch (phoneNumberSharingMode) {
|
||
|
case PhoneNumberSharingMode.Everybody:
|
||
|
certificateMode = SenderCertificateMode.WithE164;
|
||
|
break;
|
||
|
case PhoneNumberSharingMode.ContactsOnly: {
|
||
|
const isInSystemContacts = Boolean(conversationAttrs.name);
|
||
|
certificateMode = isInSystemContacts
|
||
|
? SenderCertificateMode.WithE164
|
||
|
: SenderCertificateMode.WithoutE164;
|
||
|
break;
|
||
|
}
|
||
|
case PhoneNumberSharingMode.Nobody:
|
||
|
certificateMode = SenderCertificateMode.WithoutE164;
|
||
|
break;
|
||
|
default:
|
||
|
throw missingCaseError(phoneNumberSharingMode);
|
||
|
}
|
||
|
|
||
|
return senderCertificateService.get(certificateMode);
|
||
|
}
|