From 064f3dd0e008074b237d41dbba168f232ff5a968 Mon Sep 17 00:00:00 2001 From: Scott Nonnenberg Date: Tue, 5 Jul 2022 17:28:00 -0700 Subject: [PATCH] updateSchema: Be resilient to invalid images --- ts/types/Attachment.ts | 50 ++++++++++++++++++++++-------------- ts/types/Message2.ts | 10 +++++--- ts/util/scaleImageToLevel.ts | 7 ++++- 3 files changed, 44 insertions(+), 23 deletions(-) diff --git a/ts/types/Attachment.ts b/ts/types/Attachment.ts index 6a8de05ce65a..1087833823dd 100644 --- a/ts/types/Attachment.ts +++ b/ts/types/Attachment.ts @@ -28,6 +28,7 @@ import { scaleImageToLevel } from '../util/scaleImageToLevel'; import * as GoogleChrome from '../util/GoogleChrome'; import { parseIntOrThrow } from '../util/parseIntOrThrow'; import { getValue } from '../RemoteConfig'; +import { isRecord } from '../util/isRecord'; const MAX_WIDTH = 300; const MAX_HEIGHT = MAX_WIDTH * 1.5; @@ -251,7 +252,7 @@ export function isValid( // part of re-encoding the image: export async function autoOrientJPEG( attachment: AttachmentType, - _: unknown, + { logger }: { logger: LoggerType }, { sendHQImages = false, isIncoming = false, @@ -280,26 +281,37 @@ export async function autoOrientJPEG( const dataBlob = new Blob([attachment.data], { type: attachment.contentType, }); - const { blob: xcodedDataBlob } = await scaleImageToLevel( - dataBlob, - attachment.contentType, - isIncoming - ); - const xcodedDataArrayBuffer = await blobToArrayBuffer(xcodedDataBlob); + try { + const { blob: xcodedDataBlob } = await scaleImageToLevel( + dataBlob, + attachment.contentType, + isIncoming + ); + const xcodedDataArrayBuffer = await blobToArrayBuffer(xcodedDataBlob); - // IMPORTANT: We overwrite the existing `data` `Uint8Array` losing the original - // image data. Ideally, we’d preserve the original image data for users who want to - // retain it but due to reports of data loss, we don’t want to overburden IndexedDB - // by potentially doubling stored image data. - // See: https://github.com/signalapp/Signal-Desktop/issues/1589 - const xcodedAttachment = { - // `digest` is no longer valid for auto-oriented image data, so we discard it: - ...omit(attachment, 'digest'), - data: new Uint8Array(xcodedDataArrayBuffer), - size: xcodedDataArrayBuffer.byteLength, - }; + // IMPORTANT: We overwrite the existing `data` `Uint8Array` losing the original + // image data. Ideally, we’d preserve the original image data for users who want to + // retain it but due to reports of data loss, we don’t want to overburden IndexedDB + // by potentially doubling stored image data. + // See: https://github.com/signalapp/Signal-Desktop/issues/1589 + const xcodedAttachment = { + // `digest` is no longer valid for auto-oriented image data, so we discard it: + ...omit(attachment, 'digest'), + data: new Uint8Array(xcodedDataArrayBuffer), + size: xcodedDataArrayBuffer.byteLength, + }; - return xcodedAttachment; + return xcodedAttachment; + } catch (error: unknown) { + const errorString = + isRecord(error) && 'stack' in error ? error.stack : error; + logger.error( + 'autoOrientJPEG: Failed to rotate/scale attachment', + errorString + ); + + return attachment; + } } const UNICODE_LEFT_TO_RIGHT_OVERRIDE = '\u202D'; diff --git a/ts/types/Message2.ts b/ts/types/Message2.ts index 6f2117f6eb9f..18fc2edbb697 100644 --- a/ts/types/Message2.ts +++ b/ts/types/Message2.ts @@ -580,9 +580,13 @@ export const processNewAttachment = async ( throw new TypeError('context.logger is required'); } - const rotatedAttachment = await autoOrientJPEG(attachment, undefined, { - isIncoming: true, - }); + const rotatedAttachment = await autoOrientJPEG( + attachment, + { logger }, + { + isIncoming: true, + } + ); const onDiskAttachment = await migrateDataToFileSystem(rotatedAttachment, { writeNewAttachmentData, }); diff --git a/ts/util/scaleImageToLevel.ts b/ts/util/scaleImageToLevel.ts index a086893b6404..d885c9008ff0 100644 --- a/ts/util/scaleImageToLevel.ts +++ b/ts/util/scaleImageToLevel.ts @@ -8,6 +8,7 @@ import { IMAGE_JPEG } from '../types/MIME'; import { canvasToBlob } from './canvasToBlob'; import { getValue } from '../RemoteConfig'; import { parseNumber } from './libphonenumberUtil'; +import { isRecord } from './isRecord'; enum MediaQualityLevels { One = 1, @@ -126,7 +127,11 @@ export async function scaleImageToLevel( } ({ image } = data); } catch (err) { - const error = new Error('scaleImageToLevel: Failed to process image'); + const errorString = isRecord(err) && 'stack' in err ? err.stack : err; + const error = new Error( + 'scaleImageToLevel: Failed to process image', + errorString + ); error.originalError = err; throw error; }