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.
275 lines
7.8 KiB
TypeScript
275 lines
7.8 KiB
TypeScript
import _ from 'lodash';
|
|
|
|
import { MessageModel } from '../models/message';
|
|
import { saveMessage } from '../../ts/data/data';
|
|
import { fromBase64ToArrayBuffer } from '../session/utils/String';
|
|
import { AttachmentDownloads } from '../session/utils';
|
|
import { ConversationModel } from '../models/conversation';
|
|
import {
|
|
downloadFileOpenGroupV2,
|
|
downloadFileOpenGroupV2ByUrl,
|
|
} from '../opengroup/opengroupV2/OpenGroupAPIV2';
|
|
import { OpenGroupRequestCommonType } from '../opengroup/opengroupV2/ApiUtil';
|
|
import { FSv2 } from '../fileserver';
|
|
import { getUnpaddedAttachment } from '../session/crypto/BufferPadding';
|
|
|
|
export async function downloadAttachment(attachment: any) {
|
|
const asURL = new URL(attachment.url);
|
|
const serverUrl = asURL.origin;
|
|
|
|
// is it an attachment hosted on the file server v2 ?
|
|
const defaultFsOldV2 = _.startsWith(serverUrl, FSv2.oldFileServerV2URL);
|
|
const defaultFsV2 = _.startsWith(serverUrl, FSv2.fileServerV2URL);
|
|
|
|
let res: ArrayBuffer | null = null;
|
|
|
|
if (defaultFsV2 || defaultFsOldV2) {
|
|
let attachmentId = attachment.id;
|
|
if (!attachmentId) {
|
|
// try to get the fileId from the end of the URL
|
|
attachmentId = attachment.url;
|
|
}
|
|
window?.log?.info('Download v2 file server attachment');
|
|
res = await FSv2.downloadFileFromFSv2(attachmentId, defaultFsOldV2);
|
|
} else {
|
|
window.log.warn(
|
|
`downloadAttachment attachment is neither opengroup attachment nor fsv2... Dropping it ${asURL.href}`
|
|
);
|
|
throw new Error('Attachment url is not opengroupv2 nor fileserver v2. Unsupported');
|
|
}
|
|
|
|
if (!res?.byteLength) {
|
|
window?.log?.error('Failed to download attachment. Length is 0');
|
|
throw new Error(`Failed to download attachment. Length is 0 for ${attachment.url}`);
|
|
}
|
|
|
|
// The attachment id is actually just the absolute url of the attachment
|
|
let data = res;
|
|
if (!attachment.isRaw) {
|
|
const { key, digest, size } = attachment;
|
|
|
|
if (!key || !digest) {
|
|
throw new Error('Attachment is not raw but we do not have a key to decode it');
|
|
}
|
|
|
|
data = await window.textsecure.crypto.decryptAttachment(
|
|
data,
|
|
fromBase64ToArrayBuffer(key),
|
|
fromBase64ToArrayBuffer(digest)
|
|
);
|
|
|
|
if (!size || size !== data.byteLength) {
|
|
// we might have padding, check that all the remaining bytes are padding bytes
|
|
// otherwise we have an error.
|
|
const unpaddedData = getUnpaddedAttachment(data, size);
|
|
if (!unpaddedData) {
|
|
throw new Error(
|
|
`downloadAttachment: Size ${size} did not match downloaded attachment size ${data.byteLength}`
|
|
);
|
|
}
|
|
data = unpaddedData;
|
|
}
|
|
}
|
|
|
|
return {
|
|
..._.omit(attachment, 'digest', 'key'),
|
|
data,
|
|
};
|
|
}
|
|
|
|
/**
|
|
*
|
|
* @param attachment Either the details of the attachment to download (on a per room basis), or the pathName to the file you want to get
|
|
*/
|
|
export async function downloadAttachmentOpenGroupV2(
|
|
attachment:
|
|
| {
|
|
id: number;
|
|
url: string;
|
|
size: number;
|
|
}
|
|
| string,
|
|
roomInfos: OpenGroupRequestCommonType
|
|
) {
|
|
if (typeof attachment === 'string') {
|
|
const dataUintFromUrl = await downloadFileOpenGroupV2ByUrl(attachment, roomInfos);
|
|
|
|
if (!dataUintFromUrl?.length) {
|
|
window?.log?.error('Failed to download attachment. Length is 0');
|
|
throw new Error(`Failed to download attachment. Length is 0 for ${attachment}`);
|
|
}
|
|
return dataUintFromUrl;
|
|
}
|
|
const dataUint = await downloadFileOpenGroupV2(attachment.id, roomInfos);
|
|
|
|
if (!dataUint?.length) {
|
|
window?.log?.error('Failed to download attachment. Length is 0');
|
|
throw new Error(`Failed to download attachment. Length is 0 for ${attachment.url}`);
|
|
}
|
|
|
|
let data = dataUint;
|
|
if (attachment.size !== dataUint.length) {
|
|
// we might have padding, check that all the remaining bytes are padding bytes
|
|
// otherwise we have an error.
|
|
const unpaddedData = getUnpaddedAttachment(dataUint.buffer, attachment.size);
|
|
if (!unpaddedData) {
|
|
throw new Error(
|
|
`downloadAttachment: Size ${attachment.size} did not match downloaded attachment size ${data.byteLength}`
|
|
);
|
|
}
|
|
data = new Uint8Array(unpaddedData);
|
|
} else {
|
|
// nothing to do, the attachment has already the correct size.
|
|
// There is just no padding included, which is what we agreed on
|
|
window?.log?.info('Received opengroupv2 unpadded attachment');
|
|
}
|
|
|
|
return {
|
|
..._.omit(attachment, 'digest', 'key'),
|
|
data: data.buffer,
|
|
};
|
|
}
|
|
|
|
async function processNormalAttachments(
|
|
message: MessageModel,
|
|
normalAttachments: Array<any>,
|
|
convo: ConversationModel
|
|
): Promise<number> {
|
|
const isOpenGroupV2 = convo.isOpenGroupV2();
|
|
|
|
if (message.isTrustedForAttachmentDownload()) {
|
|
const openGroupV2Details = (isOpenGroupV2 && convo.toOpenGroupV2()) || undefined;
|
|
const attachments = await Promise.all(
|
|
normalAttachments.map(async (attachment: any, index: any) => {
|
|
return AttachmentDownloads.addJob(attachment, {
|
|
messageId: message.id,
|
|
type: 'attachment',
|
|
index,
|
|
isOpenGroupV2,
|
|
openGroupV2Details,
|
|
});
|
|
})
|
|
);
|
|
|
|
message.set({ attachments });
|
|
|
|
return attachments.length;
|
|
}
|
|
window.log.info('No downloading attachments yet as this user is not trusted for now.');
|
|
return 0;
|
|
}
|
|
|
|
async function processPreviews(message: MessageModel, convo: ConversationModel): Promise<number> {
|
|
let addedCount = 0;
|
|
const isOpenGroupV2 = convo.isOpenGroupV2();
|
|
|
|
const preview = await Promise.all(
|
|
(message.get('preview') || []).map(async (item: any, index: any) => {
|
|
if (!item.image) {
|
|
return item;
|
|
}
|
|
addedCount += 1;
|
|
|
|
const image = await AttachmentDownloads.addJob(item.image, {
|
|
messageId: message.id,
|
|
type: 'preview',
|
|
index,
|
|
isOpenGroupV2,
|
|
});
|
|
|
|
return { ...item, image };
|
|
})
|
|
);
|
|
|
|
message.set({ preview });
|
|
|
|
return addedCount;
|
|
}
|
|
|
|
async function processQuoteAttachments(
|
|
message: MessageModel,
|
|
convo: ConversationModel
|
|
): Promise<number> {
|
|
let addedCount = 0;
|
|
|
|
const quote = message.get('quote');
|
|
|
|
if (!quote || !quote.attachments || !quote.attachments.length) {
|
|
return 0;
|
|
}
|
|
const isOpenGroupV2 = convo.isOpenGroupV2();
|
|
|
|
quote.attachments = await Promise.all(
|
|
quote.attachments.map(async (item: any, index: any) => {
|
|
// If we already have a path, then we copied this image from the quoted
|
|
// message and we don't need to download the attachment.
|
|
if (!item.thumbnail || item.thumbnail.path) {
|
|
return item;
|
|
}
|
|
|
|
addedCount += 1;
|
|
|
|
const thumbnail = await AttachmentDownloads.addJob(item.thumbnail, {
|
|
messageId: message.id,
|
|
type: 'quote',
|
|
index,
|
|
isOpenGroupV2,
|
|
});
|
|
|
|
return { ...item, thumbnail };
|
|
})
|
|
);
|
|
|
|
message.set({ quote });
|
|
|
|
return addedCount;
|
|
}
|
|
|
|
async function processGroupAvatar(
|
|
message: MessageModel,
|
|
convo: ConversationModel
|
|
): Promise<boolean> {
|
|
let group = message.get('group');
|
|
|
|
if (!group || !group.avatar) {
|
|
return false;
|
|
}
|
|
const isOpenGroupV2 = convo.isOpenGroupV2();
|
|
|
|
group = {
|
|
...group,
|
|
avatar: await AttachmentDownloads.addJob(group.avatar, {
|
|
messageId: message.id,
|
|
type: 'group-avatar',
|
|
index: 0,
|
|
isOpenGroupV2,
|
|
}),
|
|
};
|
|
|
|
message.set({ group });
|
|
|
|
return true;
|
|
}
|
|
|
|
export async function queueAttachmentDownloads(
|
|
message: MessageModel,
|
|
conversation: ConversationModel
|
|
): Promise<void> {
|
|
let count = 0;
|
|
|
|
count += await processNormalAttachments(message, message.get('attachments') || [], conversation);
|
|
|
|
count += await processPreviews(message, conversation);
|
|
|
|
count += await processQuoteAttachments(message, conversation);
|
|
|
|
// I don 't think we rely on this for anything
|
|
if (await processGroupAvatar(message, conversation)) {
|
|
count += 1;
|
|
}
|
|
|
|
if (count > 0) {
|
|
await saveMessage(message.attributes);
|
|
}
|
|
}
|