|  |  | @ -5,14 +5,11 @@ import filesize from 'filesize'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { |  |  |  | import { | 
			
		
	
		
		
			
				
					
					|  |  |  |   cloneDeep, |  |  |  |   cloneDeep, | 
			
		
	
		
		
			
				
					
					|  |  |  |   debounce, |  |  |  |   debounce, | 
			
		
	
		
		
			
				
					
					|  |  |  |   groupBy, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   isEmpty, |  |  |  |   isEmpty, | 
			
		
	
		
		
			
				
					
					|  |  |  |   size as lodashSize, |  |  |  |   size as lodashSize, | 
			
		
	
		
		
			
				
					
					|  |  |  |   map, |  |  |  |   map, | 
			
		
	
		
		
			
				
					
					|  |  |  |   partition, |  |  |  |   partition, | 
			
		
	
		
		
			
				
					
					|  |  |  |   pick, |  |  |  |   pick, | 
			
		
	
		
		
			
				
					
					|  |  |  |   reject, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   sortBy, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |   uniq, |  |  |  |   uniq, | 
			
		
	
		
		
			
				
					
					|  |  |  | } from 'lodash'; |  |  |  | } from 'lodash'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { SignalService } from '../protobuf'; |  |  |  | import { SignalService } from '../protobuf'; | 
			
		
	
	
		
		
			
				
					|  |  | @ -80,6 +77,7 @@ import { | 
			
		
	
		
		
			
				
					
					|  |  |  |   messagesChanged, |  |  |  |   messagesChanged, | 
			
		
	
		
		
			
				
					
					|  |  |  | } from '../state/ducks/conversations'; |  |  |  | } from '../state/ducks/conversations'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { AttachmentTypeWithPath, isVoiceMessage } from '../types/Attachment'; |  |  |  | import { AttachmentTypeWithPath, isVoiceMessage } from '../types/Attachment'; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | import { isAudio } from '../types/MIME'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { |  |  |  | import { | 
			
		
	
		
		
			
				
					
					|  |  |  |   deleteExternalMessageFiles, |  |  |  |   deleteExternalMessageFiles, | 
			
		
	
		
		
			
				
					
					|  |  |  |   getAbsoluteAttachmentPath, |  |  |  |   getAbsoluteAttachmentPath, | 
			
		
	
	
		
		
			
				
					|  |  | @ -88,8 +86,10 @@ import { | 
			
		
	
		
		
			
				
					
					|  |  |  |   loadQuoteData, |  |  |  |   loadQuoteData, | 
			
		
	
		
		
			
				
					
					|  |  |  | } from '../types/MessageAttachment'; |  |  |  | } from '../types/MessageAttachment'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { ReactionList } from '../types/Reaction'; |  |  |  | import { ReactionList } from '../types/Reaction'; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | import { getAudioDuration, getVideoDuration } from '../types/attachments/VisualAttachment'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { getAttachmentMetadata } from '../types/message/initializeAttachmentMetadata'; |  |  |  | import { getAttachmentMetadata } from '../types/message/initializeAttachmentMetadata'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { roomHasBlindEnabled } from '../types/sqlSharedTypes'; |  |  |  | import { roomHasBlindEnabled } from '../types/sqlSharedTypes'; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  | import { GoogleChrome } from '../util'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { LinkPreviews } from '../util/linkPreviews'; |  |  |  | import { LinkPreviews } from '../util/linkPreviews'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { Notifications } from '../util/notifications'; |  |  |  | import { Notifications } from '../util/notifications'; | 
			
		
	
		
		
			
				
					
					|  |  |  | import { Storage } from '../util/storage'; |  |  |  | import { Storage } from '../util/storage'; | 
			
		
	
	
		
		
			
				
					|  |  | @ -680,39 +680,45 @@ export class MessageModel extends Backbone.Model<MessageAttributes> { | 
			
		
	
		
		
			
				
					
					|  |  |  |   } |  |  |  |   } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |   public async getPropsForMessageDetail(): Promise<MessagePropsDetails> { |  |  |  |   public async getPropsForMessageDetail(): Promise<MessagePropsDetails> { | 
			
		
	
		
		
			
				
					
					|  |  |  |     // We include numbers we didn't successfully send to so we can display errors.
 |  |  |  |     // process attachments so we have the fileSize, url and screenshots
 | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     // Older messages don't have the recipients included on the message, so we fall
 |  |  |  |     const attachments = this.get('attachments') || []; | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     //   back to the conversation's current recipients
 |  |  |  |     for (let i = 0; i < attachments.length; i++) { | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |     const contacts: Array<string> = this.isIncoming() |  |  |  |       let props = this.getPropsForAttachment(attachments[i]); | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |       ? [this.get('source')] |  |  |  |       if ( | 
			
				
				
			
		
	
		
		
			
				
					
					|  |  |  |       : this.get('sent_to') || []; |  |  |  |         props?.contentType && | 
			
				
				
			
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         GoogleChrome.isVideoTypeSupported(props?.contentType) && | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         !props.duration && | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         props.url | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       ) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         // eslint-disable-next-line no-await-in-loop
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         const duration = await getVideoDuration({ | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           objectUrl: props.url, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           contentType: props.contentType, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         }); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         props = { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           ...props, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           duration, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         }; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       if (props?.contentType && isAudio(props?.contentType) && !props.duration && props.url) { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         // eslint-disable-next-line no-await-in-loop
 | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         const duration = await getAudioDuration({ | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           objectUrl: props.url, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           contentType: props.contentType, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         }); | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         props = { | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           ...props, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |           duration, | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |         }; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       } | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |       attachments[i] = props; | 
			
		
	
		
		
			
				
					
					|  |  |  |  |  |  |  |     } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     // This will make the error message for outgoing key errors a bit nicer
 |  |  |  |     // This will make the error message for outgoing key errors a bit nicer
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     const allErrors = (this.get('errors') || []).map((error: any) => { |  |  |  |     const errors = (this.get('errors') || []).map((error: any) => { | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |       return error; |  |  |  |       return error; | 
			
		
	
		
		
			
				
					
					|  |  |  |     }); |  |  |  |     }); | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     // If an error has a specific number it's associated with, we'll show it next to
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     //   that contact. Otherwise, it will be a standalone entry.
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     const errors = reject(allErrors, error => Boolean(error.number)); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     const errorsGroupedById = groupBy(allErrors, 'number'); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     const finalContacts = await Promise.all( |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |       (contacts || []).map(async id => { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         const errorsForContact = errorsGroupedById[id]; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         const contact = findAndFormatContact(id); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         return { |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |           ...contact, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |           status: this.getMessagePropStatus(), |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |           errors: errorsForContact, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |           profileName: contact.profileName, |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |         }; |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |       }) |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     ); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     // sort by pubkey
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     const sortedContacts = sortBy(finalContacts, contact => contact.pubkey); |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     const toRet: MessagePropsDetails = { |  |  |  |     const toRet: MessagePropsDetails = { | 
			
		
	
		
		
			
				
					
					|  |  |  |       sentAt: this.get('sent_at') || 0, |  |  |  |       sentAt: this.get('sent_at') || 0, | 
			
		
	
		
		
			
				
					
					|  |  |  |       receivedAt: this.get('received_at') || 0, |  |  |  |       receivedAt: this.get('received_at') || 0, | 
			
		
	
	
		
		
			
				
					|  |  | @ -724,7 +730,6 @@ export class MessageModel extends Backbone.Model<MessageAttributes> { | 
			
		
	
		
		
			
				
					
					|  |  |  |       attachments, |  |  |  |       attachments, | 
			
		
	
		
		
			
				
					
					|  |  |  |       timestamp: this.get('timestamp'), |  |  |  |       timestamp: this.get('timestamp'), | 
			
		
	
		
		
			
				
					
					|  |  |  |       serverTimestamp: this.get('serverTimestamp'), |  |  |  |       serverTimestamp: this.get('serverTimestamp'), | 
			
		
	
		
		
			
				
					
					|  |  |  |       contacts: sortedContacts || [], |  |  |  |  | 
			
		
	
		
		
			
				
					
					|  |  |  |     }; |  |  |  |     }; | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |     return toRet; |  |  |  |     return toRet; | 
			
		
	
	
		
		
			
				
					|  |  | @ -843,7 +848,7 @@ export class MessageModel extends Backbone.Model<MessageAttributes> { | 
			
		
	
		
		
			
				
					
					|  |  |  |       const conversation: ConversationModel | undefined = this.getConversation(); |  |  |  |       const conversation: ConversationModel | undefined = this.getConversation(); | 
			
		
	
		
		
			
				
					
					|  |  |  |       if (!conversation) { |  |  |  |       if (!conversation) { | 
			
		
	
		
		
			
				
					
					|  |  |  |         window?.log?.info( |  |  |  |         window?.log?.info( | 
			
		
	
		
		
			
				
					
					|  |  |  |           'cannot retry send message, the corresponding conversation was not found.' |  |  |  |           '[retrySend] Cannot retry send message, the corresponding conversation was not found.' | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         ); |  |  |  |         ); | 
			
		
	
		
		
			
				
					
					|  |  |  |         return null; |  |  |  |         return null; | 
			
		
	
		
		
			
				
					
					|  |  |  |       } |  |  |  |       } | 
			
		
	
	
		
		
			
				
					|  |  | @ -861,7 +866,7 @@ export class MessageModel extends Backbone.Model<MessageAttributes> { | 
			
		
	
		
		
			
				
					
					|  |  |  |         }; |  |  |  |         }; | 
			
		
	
		
		
			
				
					
					|  |  |  |         const roomInfos = OpenGroupData.getV2OpenGroupRoom(conversation.id); |  |  |  |         const roomInfos = OpenGroupData.getV2OpenGroupRoom(conversation.id); | 
			
		
	
		
		
			
				
					
					|  |  |  |         if (!roomInfos) { |  |  |  |         if (!roomInfos) { | 
			
		
	
		
		
			
				
					
					|  |  |  |           throw new Error('Could not find roomInfos for this conversation'); |  |  |  |           throw new Error('[retrySend] Could not find roomInfos for this conversation'); | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         } |  |  |  |         } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
		
		
			
				
					
					|  |  |  |         const openGroupMessage = new OpenGroupVisibleMessage(openGroupParams); |  |  |  |         const openGroupMessage = new OpenGroupVisibleMessage(openGroupParams); | 
			
		
	
	
		
		
			
				
					|  |  | @ -912,7 +917,7 @@ export class MessageModel extends Backbone.Model<MessageAttributes> { | 
			
		
	
		
		
			
				
					
					|  |  |  |       // as they are all polling from the same group swarm pubkey
 |  |  |  |       // as they are all polling from the same group swarm pubkey
 | 
			
		
	
		
		
			
				
					
					|  |  |  |       if (!conversation.isClosedGroup()) { |  |  |  |       if (!conversation.isClosedGroup()) { | 
			
		
	
		
		
			
				
					
					|  |  |  |         throw new Error( |  |  |  |         throw new Error( | 
			
		
	
		
		
			
				
					
					|  |  |  |           'We should only end up with a closed group here. Anything else is an error' |  |  |  |           '[retrySend] We should only end up with a closed group here. Anything else is an error' | 
			
				
				
			
		
	
		
		
	
		
		
			
				
					
					|  |  |  |         ); |  |  |  |         ); | 
			
		
	
		
		
			
				
					
					|  |  |  |       } |  |  |  |       } | 
			
		
	
		
		
			
				
					
					|  |  |  | 
 |  |  |  | 
 | 
			
		
	
	
		
		
			
				
					|  |  | 
 |