2020-10-30 20:34:04 +00:00
|
|
|
// Copyright 2016-2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2019-03-26 01:10:30 +00:00
|
|
|
/* global
|
|
|
|
_,
|
|
|
|
MessageController,
|
|
|
|
Whisper
|
|
|
|
*/
|
2018-06-25 22:17:49 +00:00
|
|
|
|
|
|
|
// eslint-disable-next-line func-names
|
2020-11-18 15:15:42 +00:00
|
|
|
(function () {
|
2018-04-27 21:25:04 +00:00
|
|
|
window.Whisper = window.Whisper || {};
|
2017-02-21 23:32:40 +00:00
|
|
|
|
2018-07-25 22:02:27 +00:00
|
|
|
async function destroyExpiredMessages() {
|
2018-08-03 04:12:27 +00:00
|
|
|
try {
|
2018-08-07 19:33:56 +00:00
|
|
|
window.log.info('destroyExpiredMessages: Loading messages...');
|
2018-08-03 04:12:27 +00:00
|
|
|
const messages = await window.Signal.Data.getExpiredMessages({
|
|
|
|
MessageCollection: Whisper.MessageCollection,
|
|
|
|
});
|
|
|
|
|
2021-03-04 21:44:57 +00:00
|
|
|
const messageIds = [];
|
|
|
|
const inMemoryMessages = [];
|
|
|
|
const messageCleanup = [];
|
|
|
|
|
|
|
|
messages.forEach(dbMessage => {
|
|
|
|
const message = MessageController.register(dbMessage.id, dbMessage);
|
|
|
|
messageIds.push(message.id);
|
|
|
|
inMemoryMessages.push(message);
|
|
|
|
messageCleanup.push(message.cleanup());
|
|
|
|
});
|
|
|
|
|
|
|
|
// We delete after the trigger to allow the conversation time to process
|
|
|
|
// the expiration before the message is removed from the database.
|
|
|
|
await window.Signal.Data.removeMessages(messageIds);
|
|
|
|
await Promise.all(messageCleanup);
|
|
|
|
|
|
|
|
inMemoryMessages.forEach(message => {
|
|
|
|
window.log.info('Message expired', {
|
|
|
|
sentAt: message.get('sent_at'),
|
|
|
|
});
|
|
|
|
|
|
|
|
Whisper.events.trigger(
|
|
|
|
'messageExpired',
|
|
|
|
message.id,
|
|
|
|
message.conversationId
|
|
|
|
);
|
|
|
|
|
|
|
|
const conversation = message.getConversation();
|
|
|
|
if (conversation) {
|
|
|
|
conversation.trigger('expired', message);
|
|
|
|
}
|
|
|
|
});
|
2018-08-03 04:12:27 +00:00
|
|
|
} catch (error) {
|
|
|
|
window.log.error(
|
|
|
|
'destroyExpiredMessages: Error deleting expired messages',
|
|
|
|
error && error.stack ? error.stack : error
|
|
|
|
);
|
|
|
|
}
|
2018-07-25 22:02:27 +00:00
|
|
|
|
2018-08-07 19:33:56 +00:00
|
|
|
window.log.info('destroyExpiredMessages: complete');
|
2018-07-25 22:02:27 +00:00
|
|
|
checkExpiringMessages();
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2017-02-21 23:32:40 +00:00
|
|
|
|
2018-06-25 22:17:49 +00:00
|
|
|
let timeout;
|
2018-07-25 22:02:27 +00:00
|
|
|
async function checkExpiringMessages() {
|
2018-04-27 21:25:04 +00:00
|
|
|
// Look up the next expiring message and set a timer to destroy it
|
2020-04-01 18:59:11 +00:00
|
|
|
const message = await window.Signal.Data.getNextExpiringMessage({
|
|
|
|
Message: Whisper.Message,
|
2018-07-25 22:02:27 +00:00
|
|
|
});
|
2017-02-21 23:32:40 +00:00
|
|
|
|
2020-04-01 18:59:11 +00:00
|
|
|
if (!message) {
|
2018-07-25 22:02:27 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-08-10 18:16:20 +00:00
|
|
|
|
2020-04-01 18:59:11 +00:00
|
|
|
const expiresAt = message.get('expires_at');
|
2018-08-03 04:12:27 +00:00
|
|
|
Whisper.ExpiringMessagesListener.nextExpiration = expiresAt;
|
2018-07-25 22:02:27 +00:00
|
|
|
window.log.info('next message expires', new Date(expiresAt).toISOString());
|
2017-02-21 23:32:40 +00:00
|
|
|
|
2018-07-25 22:02:27 +00:00
|
|
|
let wait = expiresAt - Date.now();
|
2017-08-10 18:16:20 +00:00
|
|
|
|
2018-07-25 22:02:27 +00:00
|
|
|
// In the past
|
|
|
|
if (wait < 0) {
|
|
|
|
wait = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Too far in the future, since it's limited to a 32-bit value
|
|
|
|
if (wait > 2147483647) {
|
|
|
|
wait = 2147483647;
|
|
|
|
}
|
|
|
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
timeout = setTimeout(destroyExpiredMessages, wait);
|
2018-04-27 21:25:04 +00:00
|
|
|
}
|
2019-09-26 19:56:31 +00:00
|
|
|
const debouncedCheckExpiringMessages = _.debounce(
|
2018-06-25 22:17:49 +00:00
|
|
|
checkExpiringMessages,
|
|
|
|
1000
|
|
|
|
);
|
2017-02-21 23:32:40 +00:00
|
|
|
|
2018-04-27 21:25:04 +00:00
|
|
|
Whisper.ExpiringMessagesListener = {
|
2018-08-03 04:12:27 +00:00
|
|
|
nextExpiration: null,
|
2018-06-25 22:17:49 +00:00
|
|
|
init(events) {
|
2018-04-27 21:25:04 +00:00
|
|
|
checkExpiringMessages();
|
2019-09-26 19:56:31 +00:00
|
|
|
events.on('timetravel', debouncedCheckExpiringMessages);
|
2018-04-27 21:25:04 +00:00
|
|
|
},
|
2019-09-26 19:56:31 +00:00
|
|
|
update: debouncedCheckExpiringMessages,
|
2018-04-27 21:25:04 +00:00
|
|
|
};
|
2016-09-21 00:19:51 +00:00
|
|
|
})();
|