// Copyright 2021-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import path from 'path'; import { debounce } from 'lodash'; import type { ThunkAction } from 'redux-thunk'; import type { AddLinkPreviewActionType, RemoveLinkPreviewActionType, } from './linkPreviews'; import type { AttachmentType, AttachmentDraftType, InMemoryAttachmentDraftType, } from '../../types/Attachment'; import type { DraftBodyRangesType, ReplacementValuesType, } from '../../types/Util'; import type { LinkPreviewType } from '../../types/message/LinkPreviews'; import type { MessageAttributesType } from '../../model-types.d'; import type { NoopActionType } from './noop'; import type { ShowToastActionType } from './toast'; import type { StateType as RootStateType } from '../reducer'; import type { UUIDStringType } from '../../types/UUID'; import * as log from '../../logging/log'; import * as Errors from '../../types/errors'; import { ADD_PREVIEW as ADD_LINK_PREVIEW, REMOVE_PREVIEW as REMOVE_LINK_PREVIEW, } from './linkPreviews'; import { LinkPreviewSourceType } from '../../types/LinkPreview'; import { RecordingState } from './audioRecorder'; import { SHOW_TOAST, ToastType } from './toast'; import { SafetyNumberChangeSource } from '../../components/SafetyNumberChangeDialog'; import { UUID } from '../../types/UUID'; import { assignWithNoUnnecessaryAllocation } from '../../util/assignWithNoUnnecessaryAllocation'; import { blockSendUntilConversationsAreVerified } from '../../util/blockSendUntilConversationsAreVerified'; import { clearConversationDraftAttachments } from '../../util/clearConversationDraftAttachments'; import { deleteDraftAttachment } from '../../util/deleteDraftAttachment'; import { getLinkPreviewForSend, hasLinkPreviewLoaded, maybeGrabLinkPreview, resetLinkPreview, } from '../../services/LinkPreview'; import { getMaximumAttachmentSize } from '../../util/attachments'; import { getRecipientsByConversation } from '../../util/getRecipientsByConversation'; import { getRenderDetailsForLimit, processAttachment, } from '../../util/processAttachment'; import { hasDraftAttachments } from '../../util/hasDraftAttachments'; import { isFileDangerous } from '../../util/isFileDangerous'; import { isImage, isVideo, stringToMIMEType } from '../../types/MIME'; import { isNotNil } from '../../util/isNotNil'; import { replaceIndex } from '../../util/replaceIndex'; import { resolveAttachmentDraftData } from '../../util/resolveAttachmentDraftData'; import { resolveDraftAttachmentOnDisk } from '../../util/resolveDraftAttachmentOnDisk'; import { shouldShowInvalidMessageToast } from '../../util/shouldShowInvalidMessageToast'; import { writeDraftAttachment } from '../../util/writeDraftAttachment'; // State export type ComposerStateType = { attachments: ReadonlyArray; focusCounter: number; isDisabled: boolean; linkPreviewLoading: boolean; linkPreviewResult?: LinkPreviewType; messageCompositionId: UUIDStringType; quotedMessage?: Pick; shouldSendHighQualityAttachments?: boolean; }; // Actions const ADD_PENDING_ATTACHMENT = 'composer/ADD_PENDING_ATTACHMENT'; const REPLACE_ATTACHMENTS = 'composer/REPLACE_ATTACHMENTS'; const RESET_COMPOSER = 'composer/RESET_COMPOSER'; const SET_FOCUS = 'composer/SET_FOCUS'; const SET_HIGH_QUALITY_SETTING = 'composer/SET_HIGH_QUALITY_SETTING'; const SET_QUOTED_MESSAGE = 'composer/SET_QUOTED_MESSAGE'; const SET_COMPOSER_DISABLED = 'composer/SET_COMPOSER_DISABLED'; type AddPendingAttachmentActionType = { type: typeof ADD_PENDING_ATTACHMENT; payload: AttachmentDraftType; }; type ReplaceAttachmentsActionType = { type: typeof REPLACE_ATTACHMENTS; payload: ReadonlyArray; }; type ResetComposerActionType = { type: typeof RESET_COMPOSER; }; type SetComposerDisabledStateActionType = { type: typeof SET_COMPOSER_DISABLED; payload: boolean; }; type SetFocusActionType = { type: typeof SET_FOCUS; }; type SetHighQualitySettingActionType = { type: typeof SET_HIGH_QUALITY_SETTING; payload: boolean; }; type SetQuotedMessageActionType = { type: typeof SET_QUOTED_MESSAGE; payload?: Pick; }; type ComposerActionType = | AddLinkPreviewActionType | AddPendingAttachmentActionType | RemoveLinkPreviewActionType | ReplaceAttachmentsActionType | ResetComposerActionType | SetComposerDisabledStateActionType | SetFocusActionType | SetHighQualitySettingActionType | SetQuotedMessageActionType; // Action Creators export const actions = { addAttachment, addPendingAttachment, onEditorStateChange, processAttachments, removeAttachment, replaceAttachments, resetComposer, sendMultiMediaMessage, sendStickerMessage, setComposerDisabledState, setComposerFocus, setMediaQualitySetting, setQuotedMessage, }; function sendMultiMediaMessage( conversationId: string, options: { draftAttachments?: ReadonlyArray; mentions?: DraftBodyRangesType; message?: string; timestamp?: number; voiceNoteAttachment?: InMemoryAttachmentDraftType; } ): ThunkAction< void, RootStateType, unknown, | NoopActionType | ResetComposerActionType | SetComposerDisabledStateActionType | SetQuotedMessageActionType | ShowToastActionType > { return async (dispatch, getState) => { const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error('sendMultiMediaMessage: No conversation found'); } const { draftAttachments, message = '', mentions, timestamp = Date.now(), voiceNoteAttachment, } = options; const state = getState(); const sendStart = Date.now(); const recipientsByConversation = getRecipientsByConversation([ conversation.attributes, ]); try { dispatch(setComposerDisabledState(true)); const sendAnyway = await blockSendUntilConversationsAreVerified( recipientsByConversation, SafetyNumberChangeSource.MessageSend ); if (!sendAnyway) { dispatch(setComposerDisabledState(false)); return; } } catch (error) { dispatch(setComposerDisabledState(false)); log.error('sendMessage error:', Errors.toLogFormat(error)); return; } conversation.clearTypingTimers(); const toastType = shouldShowInvalidMessageToast(conversation.attributes); if (toastType) { dispatch({ type: SHOW_TOAST, payload: { toastType, }, }); return; } try { if ( !message.length && !hasDraftAttachments(conversation.attributes.draftAttachments, { includePending: false, }) && !voiceNoteAttachment ) { return; } let attachments: Array = []; if (voiceNoteAttachment) { attachments = [voiceNoteAttachment]; } else if (draftAttachments) { attachments = ( await Promise.all(draftAttachments.map(resolveAttachmentDraftData)) ).filter(isNotNil); } const quote = state.composer.quotedMessage?.quote; const shouldSendHighQualityAttachments = window.reduxStore ? state.composer.shouldSendHighQualityAttachments : undefined; const sendHQImages = shouldSendHighQualityAttachments !== undefined ? shouldSendHighQualityAttachments : state.items['sent-media-quality'] === 'high'; const sendDelta = Date.now() - sendStart; log.info('Send pre-checks took', sendDelta, 'milliseconds'); await conversation.enqueueMessageForSend( { body: message, attachments, quote, preview: getLinkPreviewForSend(message), mentions, }, { sendHQImages, timestamp, extraReduxActions: () => { conversation.setMarkedUnread(false); resetLinkPreview(); clearConversationDraftAttachments(conversationId, draftAttachments); dispatch(setQuotedMessage(undefined)); dispatch(resetComposer()); }, } ); } catch (error) { log.error( 'Error pulling attached files before send', Errors.toLogFormat(error) ); } finally { dispatch(setComposerDisabledState(false)); } }; } function sendStickerMessage( conversationId: string, options: { packId: string; stickerId: number; } ): ThunkAction< void, RootStateType, unknown, NoopActionType | ShowToastActionType > { return async dispatch => { const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error('sendStickerMessage: No conversation found'); } const recipientsByConversation = getRecipientsByConversation([ conversation.attributes, ]); try { const sendAnyway = await blockSendUntilConversationsAreVerified( recipientsByConversation, SafetyNumberChangeSource.MessageSend ); if (!sendAnyway) { return; } const toastType = shouldShowInvalidMessageToast(conversation.attributes); if (toastType) { dispatch({ type: SHOW_TOAST, payload: { toastType, }, }); return; } const { packId, stickerId } = options; conversation.sendStickerMessage(packId, stickerId); } catch (error) { log.error('clickSend error:', Errors.toLogFormat(error)); } dispatch({ type: 'NOOP', payload: null, }); }; } // Not cool that we have to pull from ConversationModel here // but if the current selected conversation isn't the one that we're operating // on then we won't be able to grab attachments from state so we resort to the // next in-memory store. function getAttachmentsFromConversationModel( conversationId: string ): Array { const conversation = window.ConversationController.get(conversationId); return conversation?.get('draftAttachments') || []; } function addAttachment( conversationId: string, attachment: InMemoryAttachmentDraftType ): ThunkAction { return async (dispatch, getState) => { // We do async operations first so multiple in-process addAttachments don't stomp on // each other. const onDisk = await writeDraftAttachment(attachment); const isSelectedConversation = getState().conversations.selectedConversationId === conversationId; const draftAttachments = isSelectedConversation ? getState().composer.attachments : getAttachmentsFromConversationModel(conversationId); // We expect there to either be a pending draft attachment or an existing // attachment that we'll be replacing. const hasDraftAttachmentPending = draftAttachments.some( draftAttachment => draftAttachment.path === attachment.path ); // User has canceled the draft so we don't need to continue processing if (!hasDraftAttachmentPending) { await deleteDraftAttachment(onDisk); return; } // Remove any pending attachments that were transcoding const index = draftAttachments.findIndex( draftAttachment => draftAttachment.path === attachment.path ); let nextAttachments = draftAttachments; if (index < 0) { log.warn( `addAttachment: Failed to find pending attachment with path ${attachment.path}` ); nextAttachments = [...draftAttachments, onDisk]; } else { nextAttachments = replaceIndex(draftAttachments, index, onDisk); } replaceAttachments(conversationId, nextAttachments)( dispatch, getState, null ); const conversation = window.ConversationController.get(conversationId); if (conversation) { conversation.attributes.draftAttachments = nextAttachments; window.Signal.Data.updateConversation(conversation.attributes); } }; } function addPendingAttachment( conversationId: string, pendingAttachment: AttachmentDraftType ): ThunkAction { return (dispatch, getState) => { const isSelectedConversation = getState().conversations.selectedConversationId === conversationId; const draftAttachments = isSelectedConversation ? getState().composer.attachments : getAttachmentsFromConversationModel(conversationId); const nextAttachments = [...draftAttachments, pendingAttachment]; dispatch({ type: REPLACE_ATTACHMENTS, payload: nextAttachments, }); const conversation = window.ConversationController.get(conversationId); if (conversation) { conversation.attributes.draftAttachments = nextAttachments; window.Signal.Data.updateConversation(conversation.attributes); } }; } function setComposerFocus( conversationId: string ): ThunkAction { return async (dispatch, getState) => { if (getState().conversations.selectedConversationId !== conversationId) { return; } dispatch({ type: SET_FOCUS, }); }; } function onEditorStateChange( conversationId: string | undefined, messageText: string, bodyRanges: DraftBodyRangesType, caretLocation?: number ): NoopActionType { if (!conversationId) { throw new Error( 'onEditorStateChange: Got falsey conversationId, needs local override' ); } const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error('processAttachments: Unable to find conversation'); } if (messageText.length && conversation.throttledBumpTyping) { conversation.throttledBumpTyping(); } debouncedSaveDraft(conversationId, messageText, bodyRanges); // If we have attachments, don't add link preview if (!hasDraftAttachments(conversation.attributes, { includePending: true })) { maybeGrabLinkPreview(messageText, LinkPreviewSourceType.Composer, { caretLocation, }); } return { type: 'NOOP', payload: null, }; } function processAttachments({ conversationId, files, }: { conversationId: string; files: ReadonlyArray; }): ThunkAction< void, RootStateType, unknown, NoopActionType | ShowToastActionType > { return async (dispatch, getState) => { if (!files.length) { return; } // If the call came from a conversation we are no longer in we do not // update the state. if (getState().conversations.selectedConversationId !== conversationId) { return; } const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error('processAttachments: Unable to find conv'); } const state = getState(); const isRecording = state.audioRecorder.recordingState === RecordingState.Recording; if (hasLinkPreviewLoaded() || isRecording) { return; } let toastToShow: | { toastType: ToastType; parameters?: ReplacementValuesType } | undefined; const nextDraftAttachments = ( conversation.get('draftAttachments') || [] ).slice(); const filesToProcess: Array = []; for (let i = 0; i < files.length; i += 1) { const file = files[i]; const processingResult = preProcessAttachment(file, nextDraftAttachments); if (processingResult != null) { toastToShow = processingResult; } else { const pendingAttachment = getPendingAttachment(file); if (pendingAttachment) { addPendingAttachment(conversationId, pendingAttachment)( dispatch, getState, undefined ); filesToProcess.push(file); // we keep a running count of the draft attachments so we can show a // toast in case we add too many attachments at once nextDraftAttachments.push(pendingAttachment); } } } await Promise.all( filesToProcess.map(async file => { try { const attachment = await processAttachment(file); if (!attachment) { removeAttachment(conversationId, file.path)( dispatch, getState, undefined ); return; } addAttachment(conversationId, attachment)( dispatch, getState, undefined ); } catch (err) { log.error( 'handleAttachmentsProcessing: failed to process attachment:', err.stack ); removeAttachment(conversationId, file.path)( dispatch, getState, undefined ); toastToShow = { toastType: ToastType.UnableToLoadAttachment }; } }) ); if (toastToShow) { dispatch({ type: SHOW_TOAST, payload: toastToShow, }); return; } dispatch({ type: 'NOOP', payload: null, }); }; } function preProcessAttachment( file: File, draftAttachments: Array ): { toastType: ToastType; parameters?: ReplacementValuesType } | undefined { if (!file) { return; } const limitKb = getMaximumAttachmentSize(); if (file.size > limitKb) { return { toastType: ToastType.FileSize, parameters: getRenderDetailsForLimit(limitKb), }; } if (isFileDangerous(file.name)) { return { toastType: ToastType.DangerousFileType }; } if (draftAttachments.length >= 32) { return { toastType: ToastType.MaxAttachments }; } const haveNonImageOrVideo = draftAttachments.some( (attachment: AttachmentDraftType) => { return ( !isImage(attachment.contentType) && !isVideo(attachment.contentType) ); } ); // You can't add another attachment if you already have a non-image staged if (haveNonImageOrVideo) { return { toastType: ToastType.UnsupportedMultiAttachment }; } const fileType = stringToMIMEType(file.type); const imageOrVideo = isImage(fileType) || isVideo(fileType); // You can't add a non-image attachment if you already have attachments staged if (!imageOrVideo && draftAttachments.length > 0) { return { toastType: ToastType.CannotMixMultiAndNonMultiAttachments }; } return undefined; } function getPendingAttachment(file: File): AttachmentDraftType | undefined { if (!file) { return; } const fileType = stringToMIMEType(file.type); const { name: fileName } = path.parse(file.name); return { contentType: fileType, fileName, size: file.size, path: file.name, pending: true, }; } function removeAttachment( conversationId: string, filePath: string ): ThunkAction { return async (dispatch, getState) => { const { attachments } = getState().composer; const [targetAttachment] = attachments.filter( attachment => attachment.path === filePath ); if (!targetAttachment) { return; } const nextAttachments = attachments.filter( attachment => attachment.path !== filePath ); const conversation = window.ConversationController.get(conversationId); if (conversation) { conversation.attributes.draftAttachments = nextAttachments; conversation.attributes.draftChanged = true; window.Signal.Data.updateConversation(conversation.attributes); } replaceAttachments(conversationId, nextAttachments)( dispatch, getState, null ); if ( targetAttachment.path && targetAttachment.fileName !== targetAttachment.path ) { await deleteDraftAttachment(targetAttachment); } }; } function replaceAttachments( conversationId: string, attachments: ReadonlyArray ): ThunkAction { return (dispatch, getState) => { // If the call came from a conversation we are no longer in we do not // update the state. if (getState().conversations.selectedConversationId !== conversationId) { return; } dispatch({ type: REPLACE_ATTACHMENTS, payload: attachments.map(resolveDraftAttachmentOnDisk), }); }; } function resetComposer(): ResetComposerActionType { return { type: RESET_COMPOSER, }; } const debouncedSaveDraft = debounce(saveDraft); function saveDraft( conversationId: string, messageText: string, bodyRanges: DraftBodyRangesType ) { const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error('saveDraft: Unable to find conversation'); } const trimmed = messageText && messageText.length > 0 ? messageText.trim() : ''; if (conversation.get('draft') && (!messageText || trimmed.length === 0)) { conversation.set({ draft: null, draftChanged: true, draftBodyRanges: [], }); window.Signal.Data.updateConversation(conversation.attributes); return; } if (messageText !== conversation.get('draft')) { const now = Date.now(); let activeAt = conversation.get('active_at'); let timestamp = conversation.get('timestamp'); if (!activeAt) { activeAt = now; timestamp = now; } conversation.set({ active_at: activeAt, draft: messageText, draftBodyRanges: bodyRanges, draftChanged: true, timestamp, }); window.Signal.Data.updateConversation(conversation.attributes); } } function setComposerDisabledState( value: boolean ): SetComposerDisabledStateActionType { return { type: SET_COMPOSER_DISABLED, payload: value, }; } function setMediaQualitySetting( payload: boolean ): SetHighQualitySettingActionType { return { type: SET_HIGH_QUALITY_SETTING, payload, }; } function setQuotedMessage( payload?: Pick ): SetQuotedMessageActionType { return { type: SET_QUOTED_MESSAGE, payload, }; } // Reducer export function getEmptyState(): ComposerStateType { return { attachments: [], focusCounter: 0, isDisabled: false, linkPreviewLoading: false, messageCompositionId: UUID.generate().toString(), }; } export function reducer( state: Readonly = getEmptyState(), action: Readonly ): ComposerStateType { if (action.type === RESET_COMPOSER) { return getEmptyState(); } if (action.type === REPLACE_ATTACHMENTS) { const { payload: attachments } = action; return { ...state, attachments, ...(attachments.length ? {} : { shouldSendHighQualityAttachments: undefined }), }; } if (action.type === SET_FOCUS) { return { ...state, focusCounter: state.focusCounter + 1, }; } if (action.type === SET_HIGH_QUALITY_SETTING) { return { ...state, shouldSendHighQualityAttachments: action.payload, }; } if (action.type === SET_QUOTED_MESSAGE) { return { ...state, quotedMessage: action.payload, }; } if (action.type === ADD_LINK_PREVIEW) { if (action.payload.source !== LinkPreviewSourceType.Composer) { return state; } return { ...state, linkPreviewLoading: true, linkPreviewResult: action.payload.linkPreview, }; } if (action.type === REMOVE_LINK_PREVIEW) { return assignWithNoUnnecessaryAllocation(state, { linkPreviewLoading: false, linkPreviewResult: undefined, }); } if (action.type === ADD_PENDING_ATTACHMENT) { return { ...state, attachments: [...state.attachments, action.payload], }; } if (action.type === SET_COMPOSER_DISABLED) { return { ...state, isDisabled: action.payload, }; } return state; }