mirror of
https://github.com/signalapp/Signal-Desktop.git
synced 2025-12-24 12:19:41 +00:00
This chrome alarm business is in place to help us wake up and check for messages when running in the background. Without it, chrome will suspend our app after a short period of inactivity. Upon waking, if the app discovers it is not linked, it prompts you to link it. However, if you've declined registration (i.e., because you already maxed out your linked device limit, but chrome auto-added the app to another machine), we should just wait until explicitly launched again. Fixes #519 // FREEBIE
266 lines
9.1 KiB
JavaScript
266 lines
9.1 KiB
JavaScript
/*
|
|
* vim: ts=4:sw=4:expandtab
|
|
*/
|
|
|
|
;(function() {
|
|
'use strict';
|
|
// register some chrome listeners
|
|
if (chrome.notifications) {
|
|
chrome.notifications.onClicked.addListener(function() {
|
|
extension.notification.clear();
|
|
Whisper.Notifications.onclick();
|
|
});
|
|
chrome.notifications.onButtonClicked.addListener(function() {
|
|
extension.notification.clear();
|
|
Whisper.Notifications.clear();
|
|
getInboxCollection().each(function(model) {
|
|
model.markRead();
|
|
});
|
|
});
|
|
chrome.notifications.onClosed.addListener(function(id, byUser) {
|
|
if (byUser) {
|
|
Whisper.Notifications.clear();
|
|
}
|
|
});
|
|
}
|
|
|
|
// Close and reopen existing windows
|
|
var open = false;
|
|
chrome.app.window.getAll().forEach(function(appWindow) {
|
|
open = true;
|
|
appWindow.close();
|
|
});
|
|
|
|
// start a background worker for ecc
|
|
textsecure.protocol_wrapper.startWorker();
|
|
|
|
// load the initial set of conversations into memory
|
|
ConversationController.updateInbox();
|
|
|
|
extension.onLaunched(function() {
|
|
storage.onready(function() {
|
|
if (textsecure.registration.isDone()) {
|
|
openInbox();
|
|
} else {
|
|
extension.install();
|
|
}
|
|
});
|
|
});
|
|
|
|
var SERVER_URL = 'https://textsecure-service-staging.whispersystems.org';
|
|
var ATTACHMENT_SERVER_URL = 'https://whispersystems-textsecure-attachments-staging.s3.amazonaws.com';
|
|
var messageReceiver;
|
|
window.getSocketStatus = function() {
|
|
if (messageReceiver) {
|
|
return messageReceiver.getStatus();
|
|
} else {
|
|
return -1;
|
|
}
|
|
};
|
|
window.getAccountManager = function() {
|
|
var USERNAME = storage.get('number_id');
|
|
var PASSWORD = storage.get('password');
|
|
return new textsecure.AccountManager(SERVER_URL, USERNAME, PASSWORD);
|
|
};
|
|
|
|
storage.fetch();
|
|
storage.onready(function() {
|
|
setUnreadCount(storage.get("unreadCount", 0));
|
|
|
|
if (textsecure.registration.isDone()) {
|
|
extension.keepAwake();
|
|
init();
|
|
}
|
|
|
|
extension.on('registration_done', function() {
|
|
extension.keepAwake();
|
|
init(true);
|
|
});
|
|
|
|
if (open) {
|
|
openInbox();
|
|
}
|
|
});
|
|
|
|
function init(firstRun) {
|
|
window.removeEventListener('online', init);
|
|
if (!textsecure.registration.isDone()) { return; }
|
|
|
|
if (messageReceiver) { messageReceiver.close(); }
|
|
|
|
var USERNAME = storage.get('number_id');
|
|
var PASSWORD = storage.get('password');
|
|
var mySignalingKey = storage.get('signaling_key');
|
|
|
|
// initialize the socket and start listening for messages
|
|
messageReceiver = new textsecure.MessageReceiver(SERVER_URL, USERNAME, PASSWORD, mySignalingKey, ATTACHMENT_SERVER_URL);
|
|
messageReceiver.addEventListener('message', onMessageReceived);
|
|
messageReceiver.addEventListener('receipt', onDeliveryReceipt);
|
|
messageReceiver.addEventListener('contact', onContactReceived);
|
|
messageReceiver.addEventListener('group', onGroupReceived);
|
|
messageReceiver.addEventListener('sent', onSentMessage);
|
|
messageReceiver.addEventListener('error', onError);
|
|
|
|
messageReceiver.addEventListener('contactsync', onContactSyncComplete);
|
|
|
|
window.textsecure.messaging = new textsecure.MessageSender(SERVER_URL, USERNAME, PASSWORD, ATTACHMENT_SERVER_URL);
|
|
if (firstRun === true && textsecure.storage.user.getDeviceId() != '1') {
|
|
textsecure.messaging.sendRequestContactSyncMessage().then(function() {
|
|
textsecure.messaging.sendRequestGroupSyncMessage();
|
|
});
|
|
}
|
|
}
|
|
|
|
function onContactSyncComplete() {
|
|
window.dispatchEvent(new Event('textsecure:contactsync'));
|
|
}
|
|
|
|
function onContactReceived(ev) {
|
|
var contactDetails = ev.contactDetails;
|
|
ConversationController.create({
|
|
name: contactDetails.name,
|
|
id: contactDetails.number,
|
|
avatar: contactDetails.avatar,
|
|
type: 'private'
|
|
}).save();
|
|
}
|
|
|
|
function onGroupReceived(ev) {
|
|
var groupDetails = ev.groupDetails;
|
|
ConversationController.create({
|
|
id: groupDetails.id,
|
|
name: groupDetails.name,
|
|
members: groupDetails.members,
|
|
avatar: groupDetails.avatar,
|
|
type: 'group',
|
|
}).save();
|
|
}
|
|
|
|
function onMessageReceived(ev) {
|
|
var data = ev.data;
|
|
var message = initIncomingMessage(data.source, data.timestamp);
|
|
message.handleDataMessage(data.message);
|
|
}
|
|
|
|
function onSentMessage(ev) {
|
|
var now = new Date().getTime();
|
|
var data = ev.data;
|
|
|
|
var message = new Whisper.Message({
|
|
source : textsecure.storage.user.getNumber(),
|
|
sent_at : data.timestamp,
|
|
received_at : now,
|
|
conversationId : data.destination,
|
|
type : 'outgoing',
|
|
sent : true
|
|
});
|
|
|
|
message.handleDataMessage(data.message);
|
|
}
|
|
|
|
function initIncomingMessage(source, timestamp) {
|
|
var now = new Date().getTime();
|
|
|
|
var message = new Whisper.Message({
|
|
source : source,
|
|
sent_at : timestamp,
|
|
received_at : now,
|
|
conversationId : source,
|
|
type : 'incoming'
|
|
});
|
|
|
|
return message;
|
|
}
|
|
|
|
function onError(ev) {
|
|
var e = ev.error;
|
|
console.log(e);
|
|
console.log(e.stack);
|
|
|
|
if (e.name === 'HTTPError' && (e.code == 401 || e.code == 403)) {
|
|
extension.install();
|
|
return;
|
|
}
|
|
|
|
if (e.name === 'HTTPError' && e.code == -1) {
|
|
// Failed to connect to server
|
|
if (navigator.onLine) {
|
|
console.log('retrying in 1 minute');
|
|
setTimeout(init, 60000);
|
|
} else {
|
|
console.log('offline');
|
|
messageReceiver.close();
|
|
window.addEventListener('online', init);
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (ev.proto) {
|
|
if (e.name === 'MessageCounterError') {
|
|
// Ignore this message. It is likely a duplicate delivery
|
|
// because the server lost our ack the first time.
|
|
return;
|
|
}
|
|
var envelope = ev.proto;
|
|
var message = initIncomingMessage(envelope.source, envelope.timestamp.toNumber());
|
|
message.saveErrors(e).then(function() {
|
|
ConversationController.findOrCreatePrivateById(message.get('conversationId')).then(function(conversation) {
|
|
conversation.save({
|
|
active_at: Date.now(),
|
|
unreadCount: conversation.get('unreadCount') + 1
|
|
});
|
|
conversation.trigger('newmessage', message);
|
|
conversation.notify(message);
|
|
});
|
|
});
|
|
return;
|
|
}
|
|
|
|
throw e;
|
|
}
|
|
|
|
// lazy hack
|
|
window.receipts = new Backbone.Collection();
|
|
|
|
function onDeliveryReceipt(ev) {
|
|
var pushMessage = ev.proto;
|
|
var timestamp = pushMessage.timestamp.toNumber();
|
|
var messages = new Whisper.MessageCollection();
|
|
var groups = new Whisper.ConversationCollection();
|
|
console.log('delivery receipt', pushMessage.source, timestamp);
|
|
|
|
if (pushMessage.source === textsecure.storage.user.getNumber()) {
|
|
// disregard delivery receipts from myself
|
|
return;
|
|
}
|
|
|
|
groups.fetchGroups(pushMessage.source).then(function() {
|
|
messages.fetchSentAt(timestamp).then(function() {
|
|
var found = false;
|
|
messages.where({type: 'outgoing'}).forEach(function(message) {
|
|
var deliveries = message.get('delivered') || 0;
|
|
var conversationId = message.get('conversationId');
|
|
if (conversationId === pushMessage.source || groups.get(conversationId)) {
|
|
message.save({delivered: deliveries + 1}).then(function() {
|
|
// notify frontend listeners
|
|
var conversation = ConversationController.get(conversationId);
|
|
if (conversation) {
|
|
conversation.trigger('newmessage', message);
|
|
}
|
|
});
|
|
found = true;
|
|
// TODO: consider keeping a list of numbers we've
|
|
// successfully delivered to?
|
|
}
|
|
});
|
|
if (found) { return; }
|
|
// if we get here, we didn't find a matching message.
|
|
// keep the receipt in memory in case it shows up later
|
|
// as a sync message.
|
|
receipts.add({ timestamp: timestamp, source: pushMessage.source });
|
|
return;
|
|
});
|
|
});
|
|
}
|
|
})();
|