| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | // Copyright 2025 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  | import { groupBy } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | import type { ReadStatus } from '../messages/MessageReadStatus'; | 
					
						
							|  |  |  | import type { SeenStatus } from '../MessageSeenStatus'; | 
					
						
							|  |  |  | import type { ServiceIdString } from '../types/ServiceId'; | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  | import { dropNull, shallowDropNull } from '../util/dropNull'; | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* eslint-disable camelcase */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import type { | 
					
						
							|  |  |  |   MessageTypeUnhydrated, | 
					
						
							|  |  |  |   MessageType, | 
					
						
							|  |  |  |   MESSAGE_COLUMNS, | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  |   ReadableDB, | 
					
						
							|  |  |  |   MessageAttachmentDBType, | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | } from './Interface'; | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  | import { | 
					
						
							|  |  |  |   batchMultiVarQuery, | 
					
						
							|  |  |  |   convertOptionalIntegerToBoolean, | 
					
						
							|  |  |  |   jsonToObject, | 
					
						
							|  |  |  |   sql, | 
					
						
							|  |  |  |   sqlJoin, | 
					
						
							|  |  |  | } from './util'; | 
					
						
							|  |  |  | import type { AttachmentType } from '../types/Attachment'; | 
					
						
							|  |  |  | import { IMAGE_JPEG, stringToMIMEType } from '../types/MIME'; | 
					
						
							|  |  |  | import { strictAssert } from '../util/assert'; | 
					
						
							|  |  |  | import type { MessageAttributesType } from '../model-types'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const ROOT_MESSAGE_ATTACHMENT_EDIT_HISTORY_INDEX = -1; | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | function toBoolean(value: number | null): boolean | undefined { | 
					
						
							|  |  |  |   if (value == null) { | 
					
						
							|  |  |  |     return undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return value === 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  | export function hydrateMessage( | 
					
						
							|  |  |  |   db: ReadableDB, | 
					
						
							|  |  |  |   row: MessageTypeUnhydrated | 
					
						
							|  |  |  | ): MessageType { | 
					
						
							|  |  |  |   return hydrateMessages(db, [row])[0]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function hydrateMessages( | 
					
						
							|  |  |  |   db: ReadableDB, | 
					
						
							|  |  |  |   unhydratedMessages: Array<MessageTypeUnhydrated> | 
					
						
							|  |  |  | ): Array<MessageType> { | 
					
						
							|  |  |  |   const messagesWithColumnsHydrated = unhydratedMessages.map( | 
					
						
							|  |  |  |     hydrateMessageTableColumns | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return hydrateMessagesWithAttachments(db, messagesWithColumnsHydrated); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function hydrateMessageTableColumns( | 
					
						
							|  |  |  |   row: MessageTypeUnhydrated | 
					
						
							|  |  |  | ): MessageType { | 
					
						
							| 
									
										
										
										
											2025-01-16 13:34:35 -08:00
										 |  |  |   const { | 
					
						
							|  |  |  |     json, | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     body, | 
					
						
							|  |  |  |     conversationId, | 
					
						
							|  |  |  |     expirationStartTimestamp, | 
					
						
							|  |  |  |     expireTimer, | 
					
						
							|  |  |  |     isErased, | 
					
						
							|  |  |  |     isViewOnce, | 
					
						
							|  |  |  |     mentionsMe, | 
					
						
							|  |  |  |     received_at, | 
					
						
							|  |  |  |     received_at_ms, | 
					
						
							|  |  |  |     schemaVersion, | 
					
						
							|  |  |  |     serverGuid, | 
					
						
							|  |  |  |     sent_at, | 
					
						
							|  |  |  |     source, | 
					
						
							|  |  |  |     sourceServiceId, | 
					
						
							|  |  |  |     sourceDevice, | 
					
						
							|  |  |  |     storyId, | 
					
						
							|  |  |  |     type, | 
					
						
							|  |  |  |     readStatus, | 
					
						
							|  |  |  |     seenStatus, | 
					
						
							|  |  |  |     timestamp, | 
					
						
							|  |  |  |     serverTimestamp, | 
					
						
							|  |  |  |     unidentifiedDeliveryReceived, | 
					
						
							|  |  |  |   } = row; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     ...(JSON.parse(json) as Omit< | 
					
						
							|  |  |  |       MessageType, | 
					
						
							|  |  |  |       (typeof MESSAGE_COLUMNS)[number] | 
					
						
							|  |  |  |     >), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     body: dropNull(body), | 
					
						
							|  |  |  |     conversationId: conversationId || '', | 
					
						
							|  |  |  |     expirationStartTimestamp: dropNull(expirationStartTimestamp), | 
					
						
							|  |  |  |     expireTimer: dropNull(expireTimer) as MessageType['expireTimer'], | 
					
						
							|  |  |  |     isErased: toBoolean(isErased), | 
					
						
							|  |  |  |     isViewOnce: toBoolean(isViewOnce), | 
					
						
							|  |  |  |     mentionsMe: toBoolean(mentionsMe), | 
					
						
							|  |  |  |     received_at: received_at || 0, | 
					
						
							|  |  |  |     received_at_ms: dropNull(received_at_ms), | 
					
						
							|  |  |  |     schemaVersion: dropNull(schemaVersion), | 
					
						
							|  |  |  |     serverGuid: dropNull(serverGuid), | 
					
						
							|  |  |  |     sent_at: sent_at || 0, | 
					
						
							|  |  |  |     source: dropNull(source), | 
					
						
							|  |  |  |     sourceServiceId: dropNull(sourceServiceId) as ServiceIdString | undefined, | 
					
						
							|  |  |  |     sourceDevice: dropNull(sourceDevice), | 
					
						
							|  |  |  |     storyId: dropNull(storyId), | 
					
						
							|  |  |  |     type: type as MessageType['type'], | 
					
						
							|  |  |  |     readStatus: readStatus == null ? undefined : (readStatus as ReadStatus), | 
					
						
							|  |  |  |     seenStatus: seenStatus == null ? undefined : (seenStatus as SeenStatus), | 
					
						
							|  |  |  |     timestamp: timestamp || 0, | 
					
						
							|  |  |  |     serverTimestamp: dropNull(serverTimestamp), | 
					
						
							|  |  |  |     unidentifiedDeliveryReceived: toBoolean(unidentifiedDeliveryReceived), | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | export function getAttachmentReferencesForMessages( | 
					
						
							|  |  |  |   db: ReadableDB, | 
					
						
							|  |  |  |   messageIds: Array<string> | 
					
						
							|  |  |  | ): Array<MessageAttachmentDBType> { | 
					
						
							|  |  |  |   return batchMultiVarQuery( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     messageIds, | 
					
						
							|  |  |  |     ( | 
					
						
							|  |  |  |       messageIdBatch: ReadonlyArray<string>, | 
					
						
							|  |  |  |       persistent: boolean | 
					
						
							|  |  |  |     ): Array<MessageAttachmentDBType> => { | 
					
						
							|  |  |  |       const [query, params] = sql`
 | 
					
						
							|  |  |  |       SELECT * FROM message_attachments  | 
					
						
							|  |  |  |       WHERE messageId IN (${sqlJoin(messageIdBatch)}); | 
					
						
							|  |  |  |     `;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return db | 
					
						
							|  |  |  |         .prepare(query, { persistent }) | 
					
						
							|  |  |  |         .all<MessageAttachmentDBType>(params); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function hydrateMessagesWithAttachments( | 
					
						
							|  |  |  |   db: ReadableDB, | 
					
						
							|  |  |  |   messagesWithoutAttachments: Array<MessageType> | 
					
						
							|  |  |  | ): Array<MessageType> { | 
					
						
							|  |  |  |   const attachmentReferencesForAllMessages = getAttachmentReferencesForMessages( | 
					
						
							|  |  |  |     db, | 
					
						
							|  |  |  |     messagesWithoutAttachments.map(msg => msg.id) | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  |   const attachmentReferencesByMessage = groupBy( | 
					
						
							|  |  |  |     attachmentReferencesForAllMessages, | 
					
						
							|  |  |  |     'messageId' | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return messagesWithoutAttachments.map(msg => { | 
					
						
							|  |  |  |     const attachmentReferences = attachmentReferencesByMessage[msg.id] ?? []; | 
					
						
							|  |  |  |     if (!attachmentReferences.length) { | 
					
						
							|  |  |  |       return msg; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const attachmentsByEditHistoryIndex = groupBy( | 
					
						
							|  |  |  |       attachmentReferences, | 
					
						
							|  |  |  |       'editHistoryIndex' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const message = hydrateMessageRootOrRevisionWithAttachments( | 
					
						
							|  |  |  |       msg, | 
					
						
							|  |  |  |       attachmentsByEditHistoryIndex[ | 
					
						
							|  |  |  |         ROOT_MESSAGE_ATTACHMENT_EDIT_HISTORY_INDEX | 
					
						
							|  |  |  |       ] ?? [] | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (message.editHistory) { | 
					
						
							|  |  |  |       message.editHistory = message.editHistory.map((editHistory, idx) => { | 
					
						
							|  |  |  |         return hydrateMessageRootOrRevisionWithAttachments( | 
					
						
							|  |  |  |           editHistory, | 
					
						
							|  |  |  |           attachmentsByEditHistoryIndex[idx] ?? [] | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return message; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function hydrateMessageRootOrRevisionWithAttachments< | 
					
						
							|  |  |  |   T extends Pick< | 
					
						
							|  |  |  |     MessageAttributesType, | 
					
						
							|  |  |  |     | 'attachments' | 
					
						
							|  |  |  |     | 'bodyAttachment' | 
					
						
							|  |  |  |     | 'contact' | 
					
						
							|  |  |  |     | 'preview' | 
					
						
							|  |  |  |     | 'quote' | 
					
						
							|  |  |  |     | 'sticker' | 
					
						
							|  |  |  |   >, | 
					
						
							|  |  |  | >(message: T, messageAttachments: Array<MessageAttachmentDBType>): T { | 
					
						
							|  |  |  |   const attachmentsByType = groupBy( | 
					
						
							|  |  |  |     messageAttachments, | 
					
						
							|  |  |  |     'attachmentType' | 
					
						
							|  |  |  |   ) as Record< | 
					
						
							|  |  |  |     MessageAttachmentDBType['attachmentType'], | 
					
						
							|  |  |  |     Array<MessageAttachmentDBType> | 
					
						
							|  |  |  |   >; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const standardAttachments = attachmentsByType.attachment ?? []; | 
					
						
							|  |  |  |   const bodyAttachments = attachmentsByType['long-message'] ?? []; | 
					
						
							|  |  |  |   const quoteAttachments = attachmentsByType.quote ?? []; | 
					
						
							|  |  |  |   const previewAttachments = attachmentsByType.preview ?? []; | 
					
						
							|  |  |  |   const contactAttachments = attachmentsByType.contact ?? []; | 
					
						
							|  |  |  |   const stickerAttachment = (attachmentsByType.sticker ?? []).find( | 
					
						
							|  |  |  |     sticker => sticker.orderInMessage === 0 | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const hydratedMessage = structuredClone(message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (standardAttachments.length) { | 
					
						
							|  |  |  |     hydratedMessage.attachments = standardAttachments | 
					
						
							|  |  |  |       .sort((a, b) => a.orderInMessage - b.orderInMessage) | 
					
						
							|  |  |  |       .map(convertAttachmentDBFieldsToAttachmentType); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (bodyAttachments[0]) { | 
					
						
							|  |  |  |     hydratedMessage.bodyAttachment = convertAttachmentDBFieldsToAttachmentType( | 
					
						
							|  |  |  |       bodyAttachments[0] | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hydratedMessage.quote?.attachments.forEach((quoteAttachment, idx) => { | 
					
						
							|  |  |  |     const quoteThumbnail = quoteAttachments.find( | 
					
						
							|  |  |  |       attachment => attachment.orderInMessage === idx | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     if (quoteThumbnail) { | 
					
						
							|  |  |  |       // eslint-disable-next-line no-param-reassign
 | 
					
						
							|  |  |  |       quoteAttachment.thumbnail = | 
					
						
							|  |  |  |         convertAttachmentDBFieldsToAttachmentType(quoteThumbnail); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hydratedMessage.preview?.forEach((preview, idx) => { | 
					
						
							|  |  |  |     const previewAttachment = previewAttachments.find( | 
					
						
							|  |  |  |       attachment => attachment.orderInMessage === idx | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (previewAttachment) { | 
					
						
							|  |  |  |       // eslint-disable-next-line no-param-reassign
 | 
					
						
							|  |  |  |       preview.image = | 
					
						
							|  |  |  |         convertAttachmentDBFieldsToAttachmentType(previewAttachment); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   hydratedMessage.contact?.forEach((contact, idx) => { | 
					
						
							|  |  |  |     const contactAttachment = contactAttachments.find( | 
					
						
							|  |  |  |       attachment => attachment.orderInMessage === idx | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |     if (contactAttachment && contact.avatar) { | 
					
						
							|  |  |  |       // eslint-disable-next-line no-param-reassign
 | 
					
						
							|  |  |  |       contact.avatar.avatar = | 
					
						
							|  |  |  |         convertAttachmentDBFieldsToAttachmentType(contactAttachment); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (hydratedMessage.sticker && stickerAttachment) { | 
					
						
							|  |  |  |     hydratedMessage.sticker.data = | 
					
						
							|  |  |  |       convertAttachmentDBFieldsToAttachmentType(stickerAttachment); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return hydratedMessage; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function convertAttachmentDBFieldsToAttachmentType( | 
					
						
							|  |  |  |   dbFields: MessageAttachmentDBType | 
					
						
							|  |  |  | ): AttachmentType { | 
					
						
							|  |  |  |   const messageAttachment = shallowDropNull(dbFields); | 
					
						
							|  |  |  |   strictAssert(messageAttachment != null, 'must exist'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const { | 
					
						
							|  |  |  |     clientUuid, | 
					
						
							|  |  |  |     size, | 
					
						
							|  |  |  |     contentType, | 
					
						
							|  |  |  |     plaintextHash, | 
					
						
							|  |  |  |     path, | 
					
						
							|  |  |  |     localKey, | 
					
						
							|  |  |  |     caption, | 
					
						
							|  |  |  |     blurHash, | 
					
						
							|  |  |  |     height, | 
					
						
							|  |  |  |     width, | 
					
						
							|  |  |  |     digest, | 
					
						
							|  |  |  |     key, | 
					
						
							|  |  |  |     downloadPath, | 
					
						
							|  |  |  |     flags, | 
					
						
							|  |  |  |     fileName, | 
					
						
							|  |  |  |     version, | 
					
						
							|  |  |  |     incrementalMac, | 
					
						
							|  |  |  |     incrementalMacChunkSize: chunkSize, | 
					
						
							|  |  |  |     transitCdnKey: cdnKey, | 
					
						
							|  |  |  |     transitCdnNumber: cdnNumber, | 
					
						
							|  |  |  |     transitCdnUploadTimestamp: uploadTimestamp, | 
					
						
							|  |  |  |     error, | 
					
						
							|  |  |  |     pending, | 
					
						
							|  |  |  |     wasTooBig, | 
					
						
							|  |  |  |     isCorrupted, | 
					
						
							|  |  |  |     backfillError, | 
					
						
							|  |  |  |     storyTextAttachmentJson, | 
					
						
							|  |  |  |     copiedFromQuotedAttachment, | 
					
						
							|  |  |  |     localBackupPath, | 
					
						
							|  |  |  |   } = messageAttachment; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const result: AttachmentType = { | 
					
						
							|  |  |  |     clientUuid, | 
					
						
							|  |  |  |     size, | 
					
						
							|  |  |  |     contentType: stringToMIMEType(contentType), | 
					
						
							|  |  |  |     plaintextHash, | 
					
						
							|  |  |  |     path, | 
					
						
							|  |  |  |     localKey, | 
					
						
							|  |  |  |     caption, | 
					
						
							|  |  |  |     blurHash, | 
					
						
							|  |  |  |     height, | 
					
						
							|  |  |  |     width, | 
					
						
							|  |  |  |     digest, | 
					
						
							|  |  |  |     key, | 
					
						
							|  |  |  |     downloadPath, | 
					
						
							|  |  |  |     localBackupPath, | 
					
						
							|  |  |  |     flags, | 
					
						
							|  |  |  |     fileName, | 
					
						
							|  |  |  |     version, | 
					
						
							|  |  |  |     incrementalMac, | 
					
						
							|  |  |  |     chunkSize, | 
					
						
							|  |  |  |     cdnKey, | 
					
						
							|  |  |  |     cdnNumber, | 
					
						
							|  |  |  |     uploadTimestamp, | 
					
						
							|  |  |  |     pending: convertOptionalIntegerToBoolean(pending), | 
					
						
							|  |  |  |     error: convertOptionalIntegerToBoolean(error), | 
					
						
							|  |  |  |     wasTooBig: convertOptionalIntegerToBoolean(wasTooBig), | 
					
						
							|  |  |  |     copied: convertOptionalIntegerToBoolean(copiedFromQuotedAttachment), | 
					
						
							|  |  |  |     isCorrupted: convertOptionalIntegerToBoolean(isCorrupted), | 
					
						
							|  |  |  |     backfillError: convertOptionalIntegerToBoolean(backfillError), | 
					
						
							|  |  |  |     textAttachment: storyTextAttachmentJson | 
					
						
							|  |  |  |       ? jsonToObject(storyTextAttachmentJson) | 
					
						
							|  |  |  |       : undefined, | 
					
						
							| 
									
										
										
										
											2025-06-18 13:16:29 -04:00
										 |  |  |     backupCdnNumber: messageAttachment.backupCdnNumber, | 
					
						
							| 
									
										
										
										
											2025-05-22 21:09:54 -04:00
										 |  |  |     ...(messageAttachment.thumbnailPath | 
					
						
							|  |  |  |       ? { | 
					
						
							|  |  |  |           thumbnail: { | 
					
						
							|  |  |  |             path: messageAttachment.thumbnailPath, | 
					
						
							|  |  |  |             size: messageAttachment.thumbnailSize ?? 0, | 
					
						
							|  |  |  |             contentType: messageAttachment.thumbnailContentType | 
					
						
							|  |  |  |               ? stringToMIMEType(messageAttachment.thumbnailContentType) | 
					
						
							|  |  |  |               : IMAGE_JPEG, | 
					
						
							|  |  |  |             localKey: messageAttachment.thumbnailLocalKey, | 
					
						
							|  |  |  |             version: messageAttachment.thumbnailVersion, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       : {}), | 
					
						
							|  |  |  |     ...(messageAttachment.screenshotPath | 
					
						
							|  |  |  |       ? { | 
					
						
							|  |  |  |           screenshot: { | 
					
						
							|  |  |  |             path: messageAttachment.screenshotPath, | 
					
						
							|  |  |  |             size: messageAttachment.screenshotSize ?? 0, | 
					
						
							|  |  |  |             contentType: messageAttachment.screenshotContentType | 
					
						
							|  |  |  |               ? stringToMIMEType(messageAttachment.screenshotContentType) | 
					
						
							|  |  |  |               : IMAGE_JPEG, | 
					
						
							|  |  |  |             localKey: messageAttachment.screenshotLocalKey, | 
					
						
							|  |  |  |             version: messageAttachment.screenshotVersion, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       : {}), | 
					
						
							|  |  |  |     ...(messageAttachment.backupThumbnailPath | 
					
						
							|  |  |  |       ? { | 
					
						
							|  |  |  |           thumbnailFromBackup: { | 
					
						
							|  |  |  |             path: messageAttachment.backupThumbnailPath, | 
					
						
							|  |  |  |             size: messageAttachment.backupThumbnailSize ?? 0, | 
					
						
							|  |  |  |             contentType: messageAttachment.backupThumbnailContentType | 
					
						
							|  |  |  |               ? stringToMIMEType(messageAttachment.backupThumbnailContentType) | 
					
						
							|  |  |  |               : IMAGE_JPEG, | 
					
						
							|  |  |  |             localKey: messageAttachment.backupThumbnailLocalKey, | 
					
						
							|  |  |  |             version: messageAttachment.backupThumbnailVersion, | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       : {}), | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } |