import { getCompleteUrlFromRoom, openGroupPrefixRegex, openGroupV2ConversationIdRegex, } from '../opengroup/utils/OpenGroupUtils'; import { getV2OpenGroupRoom } from '../data/opengroups'; import { SyncUtils, ToastUtils, UserUtils } from '../session/utils'; import { ConversationModel, ConversationNotificationSettingType, ConversationTypeEnum, } from '../models/conversation'; import { MessageModel } from '../models/message'; import { ApiV2 } from '../opengroup/opengroupV2'; import _ from 'lodash'; import { getConversationController } from '../session/conversations'; import { BlockedNumberController } from '../util/blockedNumberController'; import { adminLeaveClosedGroup, changeNickNameModal, updateAddModeratorsModal, updateConfirmModal, updateGroupMembersModal, updateGroupNameModal, updateInviteContactModal, updateRemoveModeratorsModal, } from '../state/ducks/modalDialog'; import { createOrUpdateItem, lastAvatarUploadTimestamp, removeAllMessagesInConversation, } from '../data/data'; import { conversationReset } from '../state/ducks/conversations'; import { getDecryptedMediaUrl } from '../session/crypto/DecryptedAttachmentsManager'; import { IMAGE_JPEG } from '../types/MIME'; import { FSv2 } from '../fileserver'; import { fromBase64ToArray, toHex } from '../session/utils/String'; export const getCompleteUrlForV2ConvoId = async (convoId: string) => { if (convoId.match(openGroupV2ConversationIdRegex)) { // this is a v2 group, just build the url const roomInfos = await getV2OpenGroupRoom(convoId); if (roomInfos) { const fullUrl = getCompleteUrlFromRoom(roomInfos); return fullUrl; } } return undefined; }; export async function copyPublicKeyByConvoId(convoId: string) { if (convoId.match(openGroupPrefixRegex)) { // open group v1 or v2 if (convoId.match(openGroupV2ConversationIdRegex)) { // this is a v2 group, just build the url const completeUrl = await getCompleteUrlForV2ConvoId(convoId); if (completeUrl) { window.clipboard.writeText(completeUrl); ToastUtils.pushCopiedToClipBoard(); return; } window?.log?.warn('copy to pubkey no roomInfo'); return; } // this is a v1 const atIndex = convoId.indexOf('@'); const openGroupUrl = convoId.substr(atIndex + 1); window.clipboard.writeText(openGroupUrl); ToastUtils.pushCopiedToClipBoard(); return; } window.clipboard.writeText(convoId); ToastUtils.pushCopiedToClipBoard(); } /** * * @param messages the list of MessageModel to delete * @param convo the conversation to delete from (only v2 opengroups are supported) */ export async function deleteOpenGroupMessages( messages: Array, convo: ConversationModel ): Promise> { if (!convo.isPublic()) { throw new Error('cannot delete public message on a non public groups'); } if (convo.isOpenGroupV2()) { const roomInfos = convo.toOpenGroupV2(); // on v2 servers we can only remove a single message per request.. // so logic here is to delete each messages and get which one where not removed const validServerIdsToRemove = _.compact( messages.map(msg => { const serverId = msg.get('serverId'); return serverId; }) ); const validMessageModelsToRemove = _.compact( messages.map(msg => { const serverId = msg.get('serverId'); if (serverId) { return msg; } return undefined; }) ); let allMessagesAreDeleted: boolean = false; if (validServerIdsToRemove.length) { allMessagesAreDeleted = await ApiV2.deleteMessageByServerIds( validServerIdsToRemove, roomInfos ); } // remove only the messages we managed to remove on the server if (allMessagesAreDeleted) { window?.log?.info('Removed all those serverIds messages successfully'); return validMessageModelsToRemove; } else { window?.log?.info( 'failed to remove all those serverIds message. not removing them locally neither' ); return []; } } else { throw new Error('Opengroupv1 are not supported anymore'); } } export async function blockConvoById(conversationId: string) { const conversation = getConversationController().get(conversationId); if (!conversation.id || conversation.isPublic()) { return; } const promise = conversation.isPrivate() ? BlockedNumberController.block(conversation.id) : BlockedNumberController.blockGroup(conversation.id); await promise; await conversation.commit(); ToastUtils.pushToastSuccess('blocked', window.i18n('blocked')); } export async function unblockConvoById(conversationId: string) { const conversation = getConversationController().get(conversationId); if (!conversation) { // we assume it's a block contact and not group. // this is to be able to unlock a contact we don't have a conversation with. await BlockedNumberController.unblock(conversationId); ToastUtils.pushToastSuccess('unblocked', window.i18n('unblocked')); return; } if (!conversation.id || conversation.isPublic()) { return; } const promise = conversation.isPrivate() ? BlockedNumberController.unblock(conversationId) : BlockedNumberController.unblockGroup(conversationId); await promise; ToastUtils.pushToastSuccess('unblocked', window.i18n('unblocked')); await conversation.commit(); } export async function showUpdateGroupNameByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); if (conversation.isMediumGroup()) { // make sure all the members' convo exists so we can add or remove them await Promise.all( conversation .get('members') .map(m => getConversationController().getOrCreateAndWait(m, ConversationTypeEnum.PRIVATE)) ); } window.inboxStore?.dispatch(updateGroupNameModal({ conversationId })); } export async function showUpdateGroupMembersByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); if (conversation.isMediumGroup()) { // make sure all the members' convo exists so we can add or remove them await Promise.all( conversation .get('members') .map(m => getConversationController().getOrCreateAndWait(m, ConversationTypeEnum.PRIVATE)) ); } window.inboxStore?.dispatch(updateGroupMembersModal({ conversationId })); } export function showLeaveGroupByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); if (!conversation.isGroup()) { throw new Error('showLeaveGroupDialog() called with a non group convo.'); } const title = window.i18n('leaveGroup'); const message = window.i18n('leaveGroupConfirmation'); const ourPK = UserUtils.getOurPubKeyStrFromCache(); const isAdmin = (conversation.get('groupAdmins') || []).includes(ourPK); const isClosedGroup = conversation.get('is_medium_group') || false; // if this is not a closed group, or we are not admin, we can just show a confirmation dialog if (!isClosedGroup || (isClosedGroup && !isAdmin)) { const onClickClose = () => { window.inboxStore?.dispatch(updateConfirmModal(null)); }; window.inboxStore?.dispatch( updateConfirmModal({ title, message, onClickOk: () => { void conversation.leaveClosedGroup(); onClickClose(); }, onClickClose, }) ); } else { window.inboxStore?.dispatch( adminLeaveClosedGroup({ conversationId, }) ); } } export function showInviteContactByConvoId(conversationId: string) { window.inboxStore?.dispatch(updateInviteContactModal({ conversationId })); } export async function onMarkAllReadByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); await conversation.markReadBouncy(Date.now()); } export function showAddModeratorsByConvoId(conversationId: string) { window.inboxStore?.dispatch(updateAddModeratorsModal({ conversationId })); } export function showRemoveModeratorsByConvoId(conversationId: string) { window.inboxStore?.dispatch(updateRemoveModeratorsModal({ conversationId })); } export async function markAllReadByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); await conversation.markReadBouncy(Date.now()); } export async function setNotificationForConvoId( conversationId: string, selected: ConversationNotificationSettingType ) { const conversation = getConversationController().get(conversationId); const existingSettings = conversation.get('triggerNotificationsFor'); if (existingSettings !== selected) { conversation.set({ triggerNotificationsFor: selected }); await conversation.commit(); } } export async function clearNickNameByConvoId(conversationId: string) { const conversation = getConversationController().get(conversationId); await conversation.setNickname(''); } export function showChangeNickNameByConvoId(conversationId: string) { window.inboxStore?.dispatch(changeNickNameModal({ conversationId })); } export async function deleteMessagesByConvoIdNoConfirmation(conversationId: string) { const conversation = getConversationController().get(conversationId); await removeAllMessagesInConversation(conversationId); window.inboxStore?.dispatch( conversationReset({ conversationKey: conversationId, }) ); // destroy message keeps the active timestamp set so the // conversation still appears on the conversation list but is empty conversation.set({ lastMessage: null, unreadCount: 0, mentionedUs: false, }); await conversation.commit(); } export function deleteMessagesByConvoIdWithConfirmation(conversationId: string) { const onClickClose = () => { window?.inboxStore?.dispatch(updateConfirmModal(null)); }; const onClickOk = () => { void deleteMessagesByConvoIdNoConfirmation(conversationId); onClickClose(); }; window?.inboxStore?.dispatch( updateConfirmModal({ title: window.i18n('deleteMessages'), message: window.i18n('deleteConversationConfirmation'), onClickOk, onClickClose, }) ); } export async function setDisappearingMessagesByConvoId( conversationId: string, seconds: number | undefined ) { const conversation = getConversationController().get(conversationId); if (!seconds || seconds <= 0) { await conversation.updateExpirationTimer(null); } else { await conversation.updateExpirationTimer(seconds); } } /** * This function can be used for reupload our avatar to the fsv2 or upload a new avatar. * * If this is a reupload, the old profileKey is used, otherwise a new one is generated */ export async function uploadOurAvatar(newAvatarDecrypted?: ArrayBuffer) { const ourConvo = getConversationController().get(UserUtils.getOurPubKeyStrFromCache()); if (!ourConvo) { window.log.warn('ourConvo not found... This is not a valid case'); return; } let profileKey; let decryptedAvatarData; if (newAvatarDecrypted) { // Encrypt with a new key every time profileKey = window.libsignal.crypto.getRandomBytes(32); decryptedAvatarData = newAvatarDecrypted; } else { // this is a reupload. no need to generate a new profileKey profileKey = window.textsecure.storage.get('profileKey'); if (!profileKey) { window.log.warn('our profileKey not found'); return; } const currentAttachmentPath = ourConvo.getAvatarPath(); if (!currentAttachmentPath) { window.log.warn('No attachment currently set for our convo.. Nothing to do.'); return; } const decryptedAvatarUrl = await getDecryptedMediaUrl(currentAttachmentPath, IMAGE_JPEG); if (!decryptedAvatarUrl) { window.log.warn('Could not decrypt avatar stored locally..'); return; } const response = await fetch(decryptedAvatarUrl); const blob = await response.blob(); decryptedAvatarData = await blob.arrayBuffer(); } if (!decryptedAvatarData?.byteLength) { window.log.warn('Could not read content of avatar ...'); return; } const encryptedData = await window.textsecure.crypto.encryptProfile( decryptedAvatarData, profileKey ); const avatarPointer = await FSv2.uploadFileToFsV2(encryptedData); let fileUrl; if (!avatarPointer) { window.log.warn('failed to upload avatar to fsv2'); return; } ({ fileUrl } = avatarPointer); ourConvo.set('avatarPointer', fileUrl); // this encrypts and save the new avatar and returns a new attachment path const upgraded = await window.Signal.Migrations.processNewAttachment({ isRaw: true, data: decryptedAvatarData, url: fileUrl, }); // Replace our temporary image with the attachment pointer from the server: ourConvo.set('avatar', null); const displayName = ourConvo.get('profileName'); // write the profileKey even if it did not change window.storage.put('profileKey', profileKey); ourConvo.set({ profileKey: toHex(profileKey) }); // Replace our temporary image with the attachment pointer from the server: // this commits already await ourConvo.setLokiProfile({ avatar: upgraded.path, displayName, }); const newTimestampReupload = Date.now(); await createOrUpdateItem({ id: lastAvatarUploadTimestamp, value: newTimestampReupload }); if (newAvatarDecrypted) { UserUtils.setLastProfileUpdateTimestamp(Date.now()); await SyncUtils.forceSyncConfigurationNowIfNeeded(true); } else { window.log.info( `Reuploading avatar finished at ${newTimestampReupload}, newAttachmentPointer ${fileUrl}` ); } }