|
|
|
@ -4,7 +4,7 @@ import {
|
|
|
|
|
openGroupV2ConversationIdRegex,
|
|
|
|
|
} from '../opengroup/utils/OpenGroupUtils';
|
|
|
|
|
import { getV2OpenGroupRoom } from '../data/opengroups';
|
|
|
|
|
import { ToastUtils, UserUtils } from '../session/utils';
|
|
|
|
|
import { SyncUtils, ToastUtils, UserUtils } from '../session/utils';
|
|
|
|
|
import {
|
|
|
|
|
ConversationModel,
|
|
|
|
|
ConversationNotificationSettingType,
|
|
|
|
@ -26,8 +26,16 @@ import {
|
|
|
|
|
updateInviteContactModal,
|
|
|
|
|
updateRemoveModeratorsModal,
|
|
|
|
|
} from '../state/ducks/modalDialog';
|
|
|
|
|
import { removeAllMessagesInConversation } from '../data/data';
|
|
|
|
|
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)) {
|
|
|
|
@ -325,3 +333,98 @@ export async function setDisappearingMessagesByConvoId(
|
|
|
|
|
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 = ConversationController.getInstance().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}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|