2021-09-24 20:02:30 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import path from 'path';
|
|
|
|
|
|
|
|
import * as log from '../logging/log';
|
2021-11-15 21:54:33 +00:00
|
|
|
import type {
|
|
|
|
AttachmentDraftType,
|
|
|
|
InMemoryAttachmentDraftType,
|
|
|
|
} from '../types/Attachment';
|
2022-10-24 20:46:36 +00:00
|
|
|
import { getMaximumAttachmentSize } from './attachments';
|
2021-09-24 20:02:30 +00:00
|
|
|
import { AttachmentToastType } from '../types/AttachmentToastType';
|
|
|
|
import { fileToBytes } from './fileToBytes';
|
|
|
|
import { handleImageAttachment } from './handleImageAttachment';
|
|
|
|
import { handleVideoAttachment } from './handleVideoAttachment';
|
|
|
|
import { isAttachmentSizeOkay } from './isAttachmentSizeOkay';
|
|
|
|
import { isFileDangerous } from './isFileDangerous';
|
2022-09-15 21:40:48 +00:00
|
|
|
import { isHeic, isImage, isVideo, stringToMIMEType } from '../types/MIME';
|
2021-09-24 20:02:30 +00:00
|
|
|
import { isImageTypeSupported, isVideoTypeSupported } from './GoogleChrome';
|
|
|
|
|
2021-11-15 21:54:33 +00:00
|
|
|
export function getPendingAttachment(
|
|
|
|
file: File
|
|
|
|
): AttachmentDraftType | undefined {
|
2021-09-24 20:02:30 +00:00
|
|
|
if (!file) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const fileType = stringToMIMEType(file.type);
|
|
|
|
const { name: fileName } = path.parse(file.name);
|
|
|
|
|
|
|
|
return {
|
|
|
|
contentType: fileType,
|
|
|
|
fileName,
|
2021-10-05 22:10:08 +00:00
|
|
|
size: file.size,
|
2021-09-24 20:02:30 +00:00
|
|
|
path: file.name,
|
|
|
|
pending: true,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function preProcessAttachment(
|
|
|
|
file: File,
|
2021-11-15 21:54:33 +00:00
|
|
|
draftAttachments: Array<AttachmentDraftType>
|
2021-09-24 20:02:30 +00:00
|
|
|
): AttachmentToastType | undefined {
|
|
|
|
if (!file) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-04-13 17:47:39 +00:00
|
|
|
if (file.size > getMaximumAttachmentSize()) {
|
2021-09-24 20:02:30 +00:00
|
|
|
return AttachmentToastType.ToastFileSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isFileDangerous(file.name)) {
|
|
|
|
return AttachmentToastType.ToastDangerousFileType;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (draftAttachments.length >= 32) {
|
|
|
|
return AttachmentToastType.ToastMaxAttachments;
|
|
|
|
}
|
|
|
|
|
2022-09-15 21:40:48 +00:00
|
|
|
const haveNonImageOrVideo = draftAttachments.some(
|
|
|
|
(attachment: AttachmentDraftType) => {
|
|
|
|
return (
|
|
|
|
!isImage(attachment.contentType) && !isVideo(attachment.contentType)
|
|
|
|
);
|
|
|
|
}
|
2021-09-24 20:02:30 +00:00
|
|
|
);
|
|
|
|
// You can't add another attachment if you already have a non-image staged
|
2022-09-15 21:40:48 +00:00
|
|
|
if (haveNonImageOrVideo) {
|
|
|
|
return AttachmentToastType.ToastUnsupportedMultiAttachment;
|
2021-09-24 20:02:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const fileType = stringToMIMEType(file.type);
|
2022-09-15 21:40:48 +00:00
|
|
|
const imageOrVideo = isImage(fileType) || isVideo(fileType);
|
2021-09-24 20:02:30 +00:00
|
|
|
|
|
|
|
// You can't add a non-image attachment if you already have attachments staged
|
2022-09-15 21:40:48 +00:00
|
|
|
if (!imageOrVideo && draftAttachments.length > 0) {
|
|
|
|
return AttachmentToastType.ToastCannotMixMultiAndNonMultiAttachments;
|
2021-09-24 20:02:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function processAttachment(
|
|
|
|
file: File
|
2021-11-15 21:54:33 +00:00
|
|
|
): Promise<InMemoryAttachmentDraftType | void> {
|
2021-09-24 20:02:30 +00:00
|
|
|
const fileType = stringToMIMEType(file.type);
|
|
|
|
|
2021-11-15 21:54:33 +00:00
|
|
|
let attachment: InMemoryAttachmentDraftType;
|
2021-09-24 20:02:30 +00:00
|
|
|
try {
|
2021-12-06 17:20:27 +00:00
|
|
|
if (isImageTypeSupported(fileType) || isHeic(fileType, file.name)) {
|
2021-09-24 20:02:30 +00:00
|
|
|
attachment = await handleImageAttachment(file);
|
|
|
|
} else if (isVideoTypeSupported(fileType)) {
|
|
|
|
attachment = await handleVideoAttachment(file);
|
|
|
|
} else {
|
|
|
|
const data = await fileToBytes(file);
|
|
|
|
attachment = {
|
|
|
|
contentType: fileType,
|
|
|
|
data,
|
|
|
|
fileName: file.name,
|
|
|
|
path: file.name,
|
|
|
|
pending: false,
|
|
|
|
size: data.byteLength,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
log.error(
|
|
|
|
`Was unable to generate thumbnail for fileType ${fileType}`,
|
|
|
|
e && e.stack ? e.stack : e
|
|
|
|
);
|
|
|
|
const data = await fileToBytes(file);
|
|
|
|
attachment = {
|
|
|
|
contentType: fileType,
|
|
|
|
data,
|
|
|
|
fileName: file.name,
|
|
|
|
path: file.name,
|
|
|
|
pending: false,
|
|
|
|
size: data.byteLength,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (isAttachmentSizeOkay(attachment)) {
|
|
|
|
return attachment;
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
log.error(
|
|
|
|
'Error ensuring that image is properly sized:',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|