signal-desktop/js/expiring_messages.js

149 lines
3.8 KiB
JavaScript
Raw Normal View History

2018-06-25 22:17:49 +00:00
/* global _: false */
/* global Backbone: false */
/* global i18n: false */
/* global moment: false */
/* global Whisper: false */
// eslint-disable-next-line func-names
2018-04-27 21:25:04 +00:00
(function() {
'use strict';
2018-06-25 22:17:49 +00:00
2018-04-27 21:25:04 +00:00
window.Whisper = window.Whisper || {};
async function destroyExpiredMessages() {
try {
window.log.info('destroyExpiredMessages: Loading messages...');
const messages = await window.Signal.Data.getExpiredMessages({
MessageCollection: Whisper.MessageCollection,
});
await Promise.all(
messages.map(async message => {
window.log.info('Message expired', {
sentAt: message.get('sent_at'),
});
// 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.removeMessage(message.id, {
Message: Whisper.Message,
});
const conversation = message.getConversation();
if (conversation) {
conversation.trigger('expired', message);
}
})
);
} catch (error) {
window.log.error(
'destroyExpiredMessages: Error deleting expired messages',
error && error.stack ? error.stack : error
);
}
window.log.info('destroyExpiredMessages: complete');
checkExpiringMessages();
2018-04-27 21:25:04 +00:00
}
2018-06-25 22:17:49 +00:00
let timeout;
async function checkExpiringMessages() {
2018-04-27 21:25:04 +00:00
// Look up the next expiring message and set a timer to destroy it
const messages = await window.Signal.Data.getNextExpiringMessage({
MessageCollection: Whisper.MessageCollection,
});
const next = messages.at(0);
if (!next) {
return;
}
const expiresAt = next.get('expires_at');
Whisper.ExpiringMessagesListener.nextExpiration = expiresAt;
window.log.info('next message expires', new Date(expiresAt).toISOString());
let wait = expiresAt - Date.now();
// 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
}
2018-06-25 22:17:49 +00:00
const throttledCheckExpiringMessages = _.throttle(
checkExpiringMessages,
1000
);
2018-04-27 21:25:04 +00:00
Whisper.ExpiringMessagesListener = {
nextExpiration: null,
2018-06-25 22:17:49 +00:00
init(events) {
2018-04-27 21:25:04 +00:00
checkExpiringMessages();
events.on('timetravel', throttledCheckExpiringMessages);
},
update: throttledCheckExpiringMessages,
};
2018-06-25 22:17:49 +00:00
const TimerOption = Backbone.Model.extend({
getName() {
2018-04-27 21:25:04 +00:00
return (
i18n(['timerOption', this.get('time'), this.get('unit')].join('_')) ||
moment.duration(this.get('time'), this.get('unit')).humanize()
);
},
2018-06-25 22:17:49 +00:00
getAbbreviated() {
2018-04-27 21:25:04 +00:00
return i18n(
['timerOption', this.get('time'), this.get('unit'), 'abbreviated'].join(
'_'
)
);
},
});
Whisper.ExpirationTimerOptions = new (Backbone.Collection.extend({
model: TimerOption,
2018-06-25 22:17:49 +00:00
getName(seconds = 0) {
const o = this.findWhere({ seconds });
2018-04-27 21:25:04 +00:00
if (o) {
return o.getName();
}
2018-06-25 22:17:49 +00:00
return [seconds, 'seconds'].join(' ');
2018-04-27 21:25:04 +00:00
},
2018-06-25 22:17:49 +00:00
getAbbreviated(seconds = 0) {
const o = this.findWhere({ seconds });
2018-04-27 21:25:04 +00:00
if (o) {
return o.getAbbreviated();
}
2018-06-25 22:17:49 +00:00
return [seconds, 's'].join('');
2018-04-27 21:25:04 +00:00
},
}))(
[
[0, 'seconds'],
[5, 'seconds'],
[10, 'seconds'],
[30, 'seconds'],
[1, 'minute'],
[5, 'minutes'],
[30, 'minutes'],
[1, 'hour'],
[6, 'hours'],
[12, 'hours'],
[1, 'day'],
[1, 'week'],
2018-06-25 22:17:49 +00:00
].map(o => {
const duration = moment.duration(o[0], o[1]); // 5, 'seconds'
return {
time: o[0],
unit: o[1],
2018-04-27 21:25:04 +00:00
seconds: duration.asSeconds(),
};
2018-04-27 21:25:04 +00:00
})
);
})();