Do not confirm DOE or edit until it is processed

This commit is contained in:
Josh Perez 2023-08-07 12:36:37 -04:00 committed by GitHub
parent 7b6cd00b3f
commit d8ea785f4e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
12 changed files with 145 additions and 138 deletions

View file

@ -113,7 +113,7 @@ import type { BadgesStateType } from './state/ducks/badges';
import { areAnyCallsActiveOrRinging } from './state/selectors/calling'; import { areAnyCallsActiveOrRinging } from './state/selectors/calling';
import { badgeImageFileDownloader } from './badges/badgeImageFileDownloader'; import { badgeImageFileDownloader } from './badges/badgeImageFileDownloader';
import { actionCreators } from './state/actions'; import { actionCreators } from './state/actions';
import { Deletes } from './messageModifiers/Deletes'; import * as Deletes from './messageModifiers/Deletes';
import type { EditAttributesType } from './messageModifiers/Edits'; import type { EditAttributesType } from './messageModifiers/Edits';
import * as Edits from './messageModifiers/Edits'; import * as Edits from './messageModifiers/Edits';
import { import {
@ -2482,16 +2482,14 @@ export async function startApp(): Promise<void> {
strictAssert(fromConversation, 'Delete missing fromConversation'); strictAssert(fromConversation, 'Delete missing fromConversation');
const attributes: DeleteAttributesType = { const attributes: DeleteAttributesType = {
envelopeId: data.envelopeId,
targetSentTimestamp: del.targetSentTimestamp, targetSentTimestamp: del.targetSentTimestamp,
serverTimestamp: data.serverTimestamp, serverTimestamp: data.serverTimestamp,
fromId: fromConversation.id, fromId: fromConversation.id,
removeFromMessageReceiverCache: confirm,
}; };
const deleteModel = Deletes.getSingleton().add(attributes); drop(Deletes.onDelete(attributes));
// Note: We do not wait for completion here
void Deletes.getSingleton().onDelete(deleteModel);
confirm();
return; return;
} }
@ -2512,16 +2510,17 @@ export async function startApp(): Promise<void> {
}); });
const editAttributes: EditAttributesType = { const editAttributes: EditAttributesType = {
envelopeId: data.envelopeId,
conversationId: message.attributes.conversationId, conversationId: message.attributes.conversationId,
fromId: fromConversation.id, fromId: fromConversation.id,
fromDevice: data.sourceDevice ?? 1, fromDevice: data.sourceDevice ?? 1,
message: copyDataMessageIntoMessage(data.message, message.attributes), message: copyDataMessageIntoMessage(data.message, message.attributes),
targetSentTimestamp: editedMessageTimestamp, targetSentTimestamp: editedMessageTimestamp,
removeFromMessageReceiverCache: confirm,
}; };
drop(Edits.onEdit(editAttributes)); drop(Edits.onEdit(editAttributes));
confirm();
return; return;
} }
@ -2815,14 +2814,13 @@ export async function startApp(): Promise<void> {
log.info('Queuing sent DOE for', del.targetSentTimestamp); log.info('Queuing sent DOE for', del.targetSentTimestamp);
const attributes: DeleteAttributesType = { const attributes: DeleteAttributesType = {
envelopeId: data.envelopeId,
targetSentTimestamp: del.targetSentTimestamp, targetSentTimestamp: del.targetSentTimestamp,
serverTimestamp: data.serverTimestamp, serverTimestamp: data.serverTimestamp,
fromId: window.ConversationController.getOurConversationIdOrThrow(), fromId: window.ConversationController.getOurConversationIdOrThrow(),
removeFromMessageReceiverCache: confirm,
}; };
const deleteModel = Deletes.getSingleton().add(attributes); drop(Deletes.onDelete(attributes));
// Note: We do not wait for completion here
void Deletes.getSingleton().onDelete(deleteModel);
confirm();
return; return;
} }
@ -2837,16 +2835,16 @@ export async function startApp(): Promise<void> {
}); });
const editAttributes: EditAttributesType = { const editAttributes: EditAttributesType = {
envelopeId: data.envelopeId,
conversationId: message.attributes.conversationId, conversationId: message.attributes.conversationId,
fromId: window.ConversationController.getOurConversationIdOrThrow(), fromId: window.ConversationController.getOurConversationIdOrThrow(),
fromDevice: window.storage.user.getDeviceId() ?? 1, fromDevice: window.storage.user.getDeviceId() ?? 1,
message: copyDataMessageIntoMessage(data.message, message.attributes), message: copyDataMessageIntoMessage(data.message, message.attributes),
targetSentTimestamp: editedMessageTimestamp, targetSentTimestamp: editedMessageTimestamp,
removeFromMessageReceiverCache: confirm,
}; };
drop(Edits.onEdit(editAttributes)); drop(Edits.onEdit(editAttributes));
confirm();
return; return;
} }

View file

@ -1,109 +1,98 @@
// Copyright 2020 Signal Messenger, LLC // Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
/* eslint-disable max-classes-per-file */ import type { MessageAttributesType } from '../model-types.d';
import { Collection, Model } from 'backbone';
import type { MessageModel } from '../models/messages';
import { getContactId } from '../messages/helpers'; import { getContactId } from '../messages/helpers';
import * as log from '../logging/log'; import * as log from '../logging/log';
import * as Errors from '../types/errors'; import * as Errors from '../types/errors';
import { deleteForEveryone } from '../util/deleteForEveryone'; import { deleteForEveryone } from '../util/deleteForEveryone';
import { drop } from '../util/drop'; import { drop } from '../util/drop';
import { filter, size } from '../util/iterables';
import { getMessageSentTimestampSet } from '../util/getMessageSentTimestampSet'; import { getMessageSentTimestampSet } from '../util/getMessageSentTimestampSet';
export type DeleteAttributesType = { export type DeleteAttributesType = {
envelopeId: string;
targetSentTimestamp: number; targetSentTimestamp: number;
serverTimestamp: number; serverTimestamp: number;
fromId: string; fromId: string;
removeFromMessageReceiverCache: () => unknown;
}; };
export class DeleteModel extends Model<DeleteAttributesType> {} const deletes = new Map<string, DeleteAttributesType>();
let singleton: Deletes | undefined; export function forMessage(
messageAttributes: MessageAttributesType
): Array<DeleteAttributesType> {
const sentTimestamps = getMessageSentTimestampSet(messageAttributes);
const matchingDeletes = filter(deletes, ([_envelopeId, item]) => {
return (
item.fromId === getContactId(messageAttributes) &&
sentTimestamps.has(item.targetSentTimestamp)
);
});
export class Deletes extends Collection<DeleteModel> { if (size(matchingDeletes) > 0) {
static getSingleton(): Deletes { log.info('Found early DOE for message');
if (!singleton) { const result = Array.from(matchingDeletes);
singleton = new Deletes(); result.forEach(([envelopeId, del]) => {
} del.removeFromMessageReceiverCache();
deletes.delete(envelopeId);
return singleton;
}
forMessage(message: MessageModel): Array<DeleteModel> {
const sentTimestamps = getMessageSentTimestampSet(message.attributes);
const matchingDeletes = this.filter(item => {
return (
item.get('fromId') === getContactId(message.attributes) &&
sentTimestamps.has(item.get('targetSentTimestamp'))
);
}); });
return result.map(([_envelopeId, item]) => item);
if (matchingDeletes.length > 0) {
log.info('Found early DOE for message');
this.remove(matchingDeletes);
return matchingDeletes;
}
return [];
} }
async onDelete(del: DeleteModel): Promise<void> { return [];
try { }
// The conversation the deleted message was in; we have to find it in the database
// to to figure that out.
const targetConversation =
await window.ConversationController.getConversationForTargetMessage(
del.get('fromId'),
del.get('targetSentTimestamp')
);
if (!targetConversation) { export async function onDelete(del: DeleteAttributesType): Promise<void> {
log.info( deletes.set(del.envelopeId, del);
'No target conversation for DOE',
del.get('fromId'),
del.get('targetSentTimestamp')
);
return; const logId = `Deletes.onDelete(timestamp=${del.targetSentTimestamp})`;
}
// Do not await, since this can deadlock the queue try {
drop( // The conversation the deleted message was in; we have to find it in the database
targetConversation.queueJob('Deletes.onDelete', async () => { // to to figure that out.
log.info('Handling DOE for', del.get('targetSentTimestamp')); const targetConversation =
await window.ConversationController.getConversationForTargetMessage(
const messages = await window.Signal.Data.getMessagesBySentAt( del.fromId,
del.get('targetSentTimestamp') del.targetSentTimestamp
);
const targetMessage = messages.find(
m => del.get('fromId') === getContactId(m) && !m.deletedForEveryone
);
if (!targetMessage) {
log.info(
'No message for DOE',
del.get('fromId'),
del.get('targetSentTimestamp')
);
return;
}
const message = window.MessageController.register(
targetMessage.id,
targetMessage
);
await deleteForEveryone(message, del);
this.remove(del);
})
); );
} catch (error) {
log.error('Deletes.onDelete error:', Errors.toLogFormat(error)); if (!targetConversation) {
log.info(`${logId}: No message for DOE`);
return;
} }
// Do not await, since this can deadlock the queue
drop(
targetConversation.queueJob('Deletes.onDelete', async () => {
log.info(`${logId}: Handling DOE`);
const messages = await window.Signal.Data.getMessagesBySentAt(
del.targetSentTimestamp
);
const targetMessage = messages.find(
m => del.fromId === getContactId(m) && !m.deletedForEveryone
);
if (!targetMessage) {
log.info(`${logId}: No message for DOE 2`);
return;
}
const message = window.MessageController.register(
targetMessage.id,
targetMessage
);
await deleteForEveryone(message, del);
deletes.delete(del.envelopeId);
del.removeFromMessageReceiverCache();
})
);
} catch (error) {
log.error(`${logId}: error`, Errors.toLogFormat(error));
} }
} }

View file

@ -13,17 +13,19 @@ import { getMessageSentTimestamp } from '../util/getMessageSentTimestamp';
export type EditAttributesType = { export type EditAttributesType = {
conversationId: string; conversationId: string;
envelopeId: string;
fromId: string; fromId: string;
fromDevice: number; fromDevice: number;
message: MessageAttributesType; message: MessageAttributesType;
targetSentTimestamp: number; targetSentTimestamp: number;
removeFromMessageReceiverCache: () => unknown;
}; };
const edits = new Set<EditAttributesType>(); const edits = new Map<string, EditAttributesType>();
export function forMessage(message: MessageModel): Array<EditAttributesType> { export function forMessage(message: MessageModel): Array<EditAttributesType> {
const sentAt = getMessageSentTimestamp(message.attributes, { log }); const sentAt = getMessageSentTimestamp(message.attributes, { log });
const matchingEdits = filter(edits, item => { const matchingEdits = filter(edits, ([_envelopeId, item]) => {
return ( return (
item.targetSentTimestamp === sentAt && item.targetSentTimestamp === sentAt &&
item.fromId === getContactId(message.attributes) item.fromId === getContactId(message.attributes)
@ -31,13 +33,20 @@ export function forMessage(message: MessageModel): Array<EditAttributesType> {
}); });
if (size(matchingEdits) > 0) { if (size(matchingEdits) > 0) {
const result = Array.from(matchingEdits); const result: Array<EditAttributesType> = [];
const editsLogIds = result.map(x => x.message.sent_at); const editsLogIds: Array<number> = [];
Array.from(matchingEdits).forEach(([envelopeId, item]) => {
result.push(item);
editsLogIds.push(item.message.sent_at);
edits.delete(envelopeId);
item.removeFromMessageReceiverCache();
});
log.info( log.info(
`Edits.forMessage(${message.get('sent_at')}): ` + `Edits.forMessage(${message.get('sent_at')}): ` +
`Found early edits for message ${editsLogIds.join(', ')}` `Found early edits for message ${editsLogIds.join(', ')}`
); );
filter(matchingEdits, item => edits.delete(item));
return result; return result;
} }
@ -45,7 +54,7 @@ export function forMessage(message: MessageModel): Array<EditAttributesType> {
} }
export async function onEdit(edit: EditAttributesType): Promise<void> { export async function onEdit(edit: EditAttributesType): Promise<void> {
edits.add(edit); edits.set(edit.envelopeId, edit);
const logId = `Edits.onEdit(timestamp=${edit.message.timestamp};target=${edit.targetSentTimestamp})`; const logId = `Edits.onEdit(timestamp=${edit.message.timestamp};target=${edit.targetSentTimestamp})`;
@ -93,7 +102,8 @@ export async function onEdit(edit: EditAttributesType): Promise<void> {
await handleEditMessage(message.attributes, edit); await handleEditMessage(message.attributes, edit);
edits.delete(edit); edits.delete(edit.envelopeId);
edit.removeFromMessageReceiverCache();
}) })
); );
} catch (error) { } catch (error) {

View file

@ -21,7 +21,7 @@ import type {
} from '../model-types.d'; } from '../model-types.d';
import { filter, find, map, repeat, zipObject } from '../util/iterables'; import { filter, find, map, repeat, zipObject } from '../util/iterables';
import * as GoogleChrome from '../util/GoogleChrome'; import * as GoogleChrome from '../util/GoogleChrome';
import type { DeleteModel } from '../messageModifiers/Deletes'; import type { DeleteAttributesType } from '../messageModifiers/Deletes';
import type { SentEventData } from '../textsecure/messageReceiverEvents'; import type { SentEventData } from '../textsecure/messageReceiverEvents';
import { isNotNil } from '../util/isNotNil'; import { isNotNil } from '../util/isNotNil';
import { isNormalNumber } from '../util/isNormalNumber'; import { isNormalNumber } from '../util/isNormalNumber';
@ -3230,7 +3230,10 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
} }
async handleDeleteForEveryone( async handleDeleteForEveryone(
del: DeleteModel, del: Pick<
DeleteAttributesType,
'fromId' | 'targetSentTimestamp' | 'serverTimestamp'
>,
shouldPersist = true shouldPersist = true
): Promise<void> { ): Promise<void> {
if (this.deletingForEveryone || this.get('deletedForEveryone')) { if (this.deletingForEveryone || this.get('deletedForEveryone')) {
@ -3239,10 +3242,10 @@ export class MessageModel extends window.Backbone.Model<MessageAttributesType> {
log.info('Handling DOE.', { log.info('Handling DOE.', {
messageId: this.id, messageId: this.id,
fromId: del.get('fromId'), fromId: del.fromId,
targetSentTimestamp: del.get('targetSentTimestamp'), targetSentTimestamp: del.targetSentTimestamp,
messageServerTimestamp: this.get('serverTimestamp'), messageServerTimestamp: this.get('serverTimestamp'),
deleteServerTimestamp: del.get('serverTimestamp'), deleteServerTimestamp: del.serverTimestamp,
}); });
try { try {

View file

@ -2076,6 +2076,7 @@ export default class MessageReceiver
const ev = new SentEvent( const ev = new SentEvent(
{ {
envelopeId: envelope.id,
destination: dropNull(destination), destination: dropNull(destination),
destinationUuid, destinationUuid,
timestamp: timestamp?.toNumber(), timestamp: timestamp?.toNumber(),
@ -2178,6 +2179,7 @@ export default class MessageReceiver
log.warn(`${logId}: envelope is a sent group story`); log.warn(`${logId}: envelope is a sent group story`);
const ev = new SentEvent( const ev = new SentEvent(
{ {
envelopeId: envelope.id,
destinationUuid: { destinationUuid: {
aci: envelope.destinationUuid.toString(), aci: envelope.destinationUuid.toString(),
}, },
@ -2250,6 +2252,7 @@ export default class MessageReceiver
distributionListToSentUuid.forEach((sentToUuids, listId) => { distributionListToSentUuid.forEach((sentToUuids, listId) => {
const ev = new SentEvent( const ev = new SentEvent(
{ {
envelopeId: envelope.id,
destinationUuid: { destinationUuid: {
aci: envelope.destinationUuid.toString(), aci: envelope.destinationUuid.toString(),
pni: undefined, pni: undefined,
@ -2287,6 +2290,7 @@ export default class MessageReceiver
log.warn(`${logId}: envelope is a received story`); log.warn(`${logId}: envelope is a received story`);
const ev = new MessageEvent( const ev = new MessageEvent(
{ {
envelopeId: envelope.id,
source: envelope.source, source: envelope.source,
sourceUuid: envelope.sourceUuid, sourceUuid: envelope.sourceUuid,
sourceDevice: envelope.sourceDevice, sourceDevice: envelope.sourceDevice,
@ -2352,6 +2356,7 @@ export default class MessageReceiver
const ev = new MessageEvent( const ev = new MessageEvent(
{ {
envelopeId: envelope.id,
source: envelope.source, source: envelope.source,
sourceUuid: envelope.sourceUuid, sourceUuid: envelope.sourceUuid,
sourceDevice: envelope.sourceDevice, sourceDevice: envelope.sourceDevice,
@ -2459,6 +2464,7 @@ export default class MessageReceiver
const ev = new MessageEvent( const ev = new MessageEvent(
{ {
envelopeId: envelope.id,
source: envelope.source, source: envelope.source,
sourceUuid: envelope.sourceUuid, sourceUuid: envelope.sourceUuid,
sourceDevice: envelope.sourceDevice, sourceDevice: envelope.sourceDevice,
@ -3085,6 +3091,7 @@ export default class MessageReceiver
const ev = new SentEvent( const ev = new SentEvent(
{ {
envelopeId: envelope.id,
destination: dropNull(destination), destination: dropNull(destination),
destinationUuid, destinationUuid,
timestamp: envelope.timestamp, timestamp: envelope.timestamp,

View file

@ -177,6 +177,7 @@ export class RetryRequestEvent extends ConfirmableEvent {
} }
export type SentEventData = Readonly<{ export type SentEventData = Readonly<{
envelopeId: string;
destination?: string; destination?: string;
destinationUuid?: TaggedUUIDStringType; destinationUuid?: TaggedUUIDStringType;
timestamp?: number; timestamp?: number;
@ -213,6 +214,7 @@ export class ProfileKeyUpdateEvent extends ConfirmableEvent {
} }
export type MessageEventData = Readonly<{ export type MessageEventData = Readonly<{
envelopeId: string;
source?: string; source?: string;
sourceUuid: UUIDStringType; sourceUuid: UUIDStringType;
sourceDevice?: number; sourceDevice?: number;

View file

@ -1,7 +1,7 @@
// Copyright 2020 Signal Messenger, LLC // Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import type { DeleteModel } from '../messageModifiers/Deletes'; import type { DeleteAttributesType } from '../messageModifiers/Deletes';
import type { MessageModel } from '../models/messages'; import type { MessageModel } from '../models/messages';
import * as log from '../logging/log'; import * as log from '../logging/log';
import { DAY } from './durations'; import { DAY } from './durations';
@ -11,7 +11,10 @@ import { isStory } from '../state/selectors/message';
export async function deleteForEveryone( export async function deleteForEveryone(
message: MessageModel, message: MessageModel,
doe: DeleteModel, doe: Pick<
DeleteAttributesType,
'fromId' | 'targetSentTimestamp' | 'serverTimestamp'
>,
shouldPersist = true shouldPersist = true
): Promise<void> { ): Promise<void> {
if (isDeletionByMe(message, doe)) { if (isDeletionByMe(message, doe)) {
@ -32,11 +35,11 @@ export async function deleteForEveryone(
if (isDeletionTooOld(message, doe)) { if (isDeletionTooOld(message, doe)) {
log.warn('Received late DOE. Dropping.', { log.warn('Received late DOE. Dropping.', {
fromId: doe.get('fromId'), fromId: doe.fromId,
targetSentTimestamp: doe.get('targetSentTimestamp'), targetSentTimestamp: doe.targetSentTimestamp,
messageServerTimestamp: message.get('serverTimestamp'), messageServerTimestamp: message.get('serverTimestamp'),
messageSentAt: message.get('sent_at'), messageSentAt: message.get('sent_at'),
deleteServerTimestamp: doe.get('serverTimestamp'), deleteServerTimestamp: doe.serverTimestamp,
}); });
return; return;
} }
@ -46,22 +49,22 @@ export async function deleteForEveryone(
function isDeletionByMe( function isDeletionByMe(
message: Readonly<MessageModel>, message: Readonly<MessageModel>,
doe: Readonly<DeleteModel> doe: Pick<DeleteAttributesType, 'fromId'>
): boolean { ): boolean {
const ourConversationId = const ourConversationId =
window.ConversationController.getOurConversationIdOrThrow(); window.ConversationController.getOurConversationIdOrThrow();
return ( return (
getContactId(message.attributes) === ourConversationId && getContactId(message.attributes) === ourConversationId &&
doe.get('fromId') === ourConversationId doe.fromId === ourConversationId
); );
} }
function isDeletionTooOld( function isDeletionTooOld(
message: Readonly<MessageModel>, message: Readonly<MessageModel>,
doe: Readonly<DeleteModel> doe: Pick<DeleteAttributesType, 'serverTimestamp'>
): boolean { ): boolean {
const messageTimestamp = const messageTimestamp =
message.get('serverTimestamp') || message.get('sent_at') || 0; message.get('serverTimestamp') || message.get('sent_at') || 0;
const delta = Math.abs(doe.get('serverTimestamp') - messageTimestamp); const delta = Math.abs(doe.serverTimestamp - messageTimestamp);
return delta > DAY; return delta > DAY;
} }

View file

@ -25,7 +25,10 @@ import { modifyTargetMessage } from './modifyTargetMessage';
export async function handleEditMessage( export async function handleEditMessage(
mainMessage: MessageAttributesType, mainMessage: MessageAttributesType,
editAttributes: EditAttributesType editAttributes: Pick<
EditAttributesType,
'message' | 'conversationId' | 'fromDevice' | 'fromId'
>
): Promise<void> { ): Promise<void> {
const idLog = `handleEditMessage(${getMessageIdForLogging(mainMessage)})`; const idLog = `handleEditMessage(${getMessageIdForLogging(mainMessage)})`;

View file

@ -8,7 +8,7 @@ import type { SendStateByConversationId } from '../messages/MessageSendState';
import * as Edits from '../messageModifiers/Edits'; import * as Edits from '../messageModifiers/Edits';
import * as log from '../logging/log'; import * as log from '../logging/log';
import { Deletes } from '../messageModifiers/Deletes'; import * as Deletes from '../messageModifiers/Deletes';
import { import {
MessageReceipts, MessageReceipts,
MessageReceiptType, MessageReceiptType,
@ -242,7 +242,7 @@ export async function modifyTargetMessage(
// Does message message have any pending, previously-received associated // Does message message have any pending, previously-received associated
// delete for everyone messages? // delete for everyone messages?
const deletes = Deletes.getSingleton().forMessage(message); const deletes = Deletes.forMessage(message.attributes);
await Promise.all( await Promise.all(
deletes.map(async del => { deletes.map(async del => {
await deleteForEveryone(message, del, false); await deleteForEveryone(message, del, false);

View file

@ -2,10 +2,8 @@
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import { isEqual } from 'lodash'; import { isEqual } from 'lodash';
import type { DeleteAttributesType } from '../messageModifiers/Deletes';
import type { StoryRecipientUpdateEvent } from '../textsecure/messageReceiverEvents'; import type { StoryRecipientUpdateEvent } from '../textsecure/messageReceiverEvents';
import * as log from '../logging/log'; import * as log from '../logging/log';
import { DeleteModel } from '../messageModifiers/Deletes';
import { SendStatus } from '../messages/MessageSendState'; import { SendStatus } from '../messages/MessageSendState';
import { getConversationIdForLogging } from './idForLogging'; import { getConversationIdForLogging } from './idForLogging';
import { isStory } from '../state/selectors/message'; import { isStory } from '../state/selectors/message';
@ -175,12 +173,6 @@ export async function onStoryRecipientUpdate(
messageId: item.id, messageId: item.id,
storyDistributionListId, storyDistributionListId,
}); });
const delAttributes: DeleteAttributesType = {
fromId: ourConversationId,
serverTimestamp: Number(item.serverTimestamp),
targetSentTimestamp: item.timestamp,
};
const doe = new DeleteModel(delAttributes);
// There are no longer any remaining members for this message so lets // There are no longer any remaining members for this message so lets
// run it through deleteForEveryone which marks the message as // run it through deleteForEveryone which marks the message as
@ -189,7 +181,13 @@ export async function onStoryRecipientUpdate(
// NOTE: We don't call `Deletes.onDelete()` so the message lookup by // NOTE: We don't call `Deletes.onDelete()` so the message lookup by
// sent timestamp doesn't happen (it would return all copies of the // sent timestamp doesn't happen (it would return all copies of the
// story, not just the one we want to delete). // story, not just the one we want to delete).
void message.handleDeleteForEveryone(doe); drop(
message.handleDeleteForEveryone({
fromId: ourConversationId,
serverTimestamp: Number(item.serverTimestamp),
targetSentTimestamp: item.timestamp,
})
);
} else { } else {
message.set({ message.set({
sendStateByConversationId: nextSendStateByConversationId, sendStateByConversationId: nextSendStateByConversationId,

View file

@ -6,7 +6,6 @@ import type { ConversationQueueJobData } from '../jobs/conversationJobQueue';
import * as Errors from '../types/errors'; import * as Errors from '../types/errors';
import * as durations from './durations'; import * as durations from './durations';
import * as log from '../logging/log'; import * as log from '../logging/log';
import { DeleteModel } from '../messageModifiers/Deletes';
import { import {
conversationJobQueue, conversationJobQueue,
conversationQueueJobEnum, conversationQueueJobEnum,
@ -91,10 +90,9 @@ export async function sendDeleteForEveryoneMessage(
throw error; throw error;
} }
const deleteModel = new DeleteModel({ await deleteForEveryone(message, {
targetSentTimestamp: targetTimestamp, targetSentTimestamp: targetTimestamp,
serverTimestamp: Date.now(), serverTimestamp: Date.now(),
fromId: window.ConversationController.getOurConversationIdOrThrow(), fromId: window.ConversationController.getOurConversationIdOrThrow(),
}); });
await deleteForEveryone(message, deleteModel);
} }

View file

@ -194,18 +194,14 @@ export async function sendEditedMessage(
type: 'outgoing', type: 'outgoing',
}; };
// Building up the dependencies for handling the edit message // Takes care of putting the message in the edit history, replacing the
const editAttributes = { // main message's values, and updating the conversation's properties.
await handleEditMessage(targetMessage.attributes, {
conversationId, conversationId,
fromId, fromId,
fromDevice: window.storage.user.getDeviceId() ?? 1, fromDevice: window.storage.user.getDeviceId() ?? 1,
message: tmpMessage, message: tmpMessage,
targetSentTimestamp, });
};
// Takes care of putting the message in the edit history, replacing the
// main message's values, and updating the conversation's properties.
await handleEditMessage(targetMessage.attributes, editAttributes);
// Inserting the send into a job and saving it to the message // Inserting the send into a job and saving it to the message
await timeAndLogIfTooLong( await timeAndLogIfTooLong(