2014-11-13 22:35:37 +00:00
|
|
|
/* vim: ts=4:sw=4:expandtab
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2014-05-12 00:13:09 +00:00
|
|
|
(function () {
|
2014-11-13 22:35:37 +00:00
|
|
|
'use strict';
|
|
|
|
window.Whisper = window.Whisper || {};
|
2014-05-12 00:13:09 +00:00
|
|
|
|
2015-05-20 22:52:25 +00:00
|
|
|
var Message = window.Whisper.Message = Backbone.Model.extend({
|
2014-12-12 03:41:40 +00:00
|
|
|
database : Whisper.Database,
|
|
|
|
storeName : 'messages',
|
|
|
|
defaults : function() {
|
2014-11-20 23:43:51 +00:00
|
|
|
return {
|
|
|
|
timestamp: new Date().getTime(),
|
|
|
|
attachments: []
|
|
|
|
};
|
|
|
|
},
|
2014-11-13 22:35:37 +00:00
|
|
|
validate: function(attributes, options) {
|
2014-12-12 03:41:40 +00:00
|
|
|
var required = ['conversationId', 'received_at', 'sent_at'];
|
2014-11-13 22:35:37 +00:00
|
|
|
var missing = _.filter(required, function(attr) { return !attributes[attr]; });
|
|
|
|
if (missing.length) {
|
|
|
|
console.log("Message missing attributes: " + missing);
|
|
|
|
}
|
2015-02-17 19:54:15 +00:00
|
|
|
},
|
|
|
|
isEndSession: function() {
|
|
|
|
var flag = textsecure.protobuf.PushMessageContent.Flags.END_SESSION;
|
|
|
|
return !!(this.get('flags') & flag);
|
|
|
|
},
|
|
|
|
isGroupUpdate: function() {
|
|
|
|
return !!(this.get('group_update'));
|
2015-02-25 00:02:33 +00:00
|
|
|
},
|
|
|
|
isIncoming: function() {
|
|
|
|
return this.get('type') === 'incoming';
|
2015-03-12 00:49:01 +00:00
|
|
|
},
|
2015-03-19 23:17:26 +00:00
|
|
|
getDescription: function() {
|
|
|
|
if (this.isGroupUpdate()) {
|
2015-03-23 22:44:47 +00:00
|
|
|
var group_update = this.get('group_update');
|
|
|
|
if (group_update.left) {
|
|
|
|
return group_update.left + ' left the group.';
|
|
|
|
}
|
|
|
|
|
|
|
|
var messages = ['Updated the group.'];
|
|
|
|
if (group_update.name) {
|
|
|
|
messages.push("Title is now '" + group_update.name + "'.");
|
|
|
|
}
|
|
|
|
if (group_update.joined) {
|
|
|
|
messages.push(group_update.joined.join(', ') + ' joined the group.');
|
|
|
|
}
|
|
|
|
|
|
|
|
return messages.join(' ');
|
2015-03-19 23:17:26 +00:00
|
|
|
}
|
|
|
|
if (this.isEndSession()) {
|
2015-03-23 22:44:47 +00:00
|
|
|
return 'Secure session ended.';
|
2015-03-19 23:17:26 +00:00
|
|
|
}
|
2015-03-20 17:47:58 +00:00
|
|
|
if (this.isIncoming() && this.hasKeyConflicts()) {
|
|
|
|
return 'Received message with unknown identity key.';
|
|
|
|
}
|
2015-03-19 23:17:26 +00:00
|
|
|
|
|
|
|
return this.get('body');
|
|
|
|
},
|
2015-03-12 00:49:01 +00:00
|
|
|
getContact: function() {
|
2015-03-12 03:41:15 +00:00
|
|
|
if (this.collection) {
|
|
|
|
return this.collection.conversation.contactCollection.get(this.get('source'));
|
|
|
|
}
|
2015-02-24 00:23:22 +00:00
|
|
|
},
|
|
|
|
isOutgoing: function() {
|
|
|
|
return this.get('type') === 'outgoing';
|
|
|
|
},
|
2015-02-18 02:03:05 +00:00
|
|
|
hasKeyConflicts: function() {
|
|
|
|
return _.any(this.get('errors'), function(e) {
|
|
|
|
return (e.name === 'IncomingIdentityKeyError' ||
|
|
|
|
e.name === 'OutgoingIdentityKeyError');
|
|
|
|
});
|
|
|
|
},
|
|
|
|
hasKeyConflict: function(number) {
|
|
|
|
return _.any(this.get('errors'), function(e) {
|
|
|
|
return (e.name === 'IncomingIdentityKeyError' ||
|
|
|
|
e.name === 'OutgoingIdentityKeyError') &&
|
|
|
|
e.number === number;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
getKeyConflict: function(number) {
|
2015-02-24 00:23:22 +00:00
|
|
|
return _.find(this.get('errors'), function(e) {
|
2015-02-18 02:03:05 +00:00
|
|
|
return (e.name === 'IncomingIdentityKeyError' ||
|
|
|
|
e.name === 'OutgoingIdentityKeyError') &&
|
|
|
|
e.number === number;
|
2015-02-24 00:23:22 +00:00
|
|
|
});
|
2015-02-18 02:03:05 +00:00
|
|
|
},
|
|
|
|
resolveConflict: function(number) {
|
|
|
|
var error = this.getKeyConflict(number);
|
|
|
|
if (error) {
|
|
|
|
var promise = new textsecure.ReplayableError(error).replay();
|
|
|
|
if (this.isIncoming()) {
|
|
|
|
promise.then(function(pushMessageContent) {
|
2015-03-18 23:26:55 +00:00
|
|
|
this.handlePushMessageContent(pushMessageContent);
|
2015-02-18 02:03:05 +00:00
|
|
|
this.save('errors', []);
|
|
|
|
}.bind(this)).catch(function(e) {
|
|
|
|
//this.save('errors', [_.pick(e, ['name', 'message'])]);
|
|
|
|
var errors = this.get('errors').concat(
|
|
|
|
_.pick(e, ['name', 'message'])
|
|
|
|
);
|
|
|
|
this.save('errors', errors);
|
|
|
|
}.bind(this));
|
|
|
|
} else {
|
|
|
|
promise.then(function() {
|
|
|
|
this.save('errors', _.reject(this.get('errors'), function(e) {
|
|
|
|
return e.name === 'OutgoingIdentityKeyError' &&
|
|
|
|
e.number === number;
|
|
|
|
}));
|
|
|
|
}.bind(this));
|
|
|
|
}
|
|
|
|
}
|
2015-03-18 23:26:55 +00:00
|
|
|
},
|
|
|
|
handlePushMessageContent: function(pushMessageContent) {
|
|
|
|
// This function can be called from the background script on an
|
|
|
|
// incoming message or from the frontend after the user accepts an
|
|
|
|
// identity key change.
|
|
|
|
var message = this;
|
|
|
|
var source = message.get('source');
|
|
|
|
var timestamp = message.get('sent_at');
|
|
|
|
return textsecure.processDecrypted(pushMessageContent, source).then(function(pushMessageContent) {
|
|
|
|
var conversationId = source;
|
|
|
|
if (pushMessageContent.sync) {
|
|
|
|
conversationId = pushMessageContent.sync.destination;
|
2015-05-18 21:08:19 +00:00
|
|
|
}
|
|
|
|
if (pushMessageContent.group) {
|
2015-03-18 23:26:55 +00:00
|
|
|
conversationId = pushMessageContent.group.id;
|
|
|
|
}
|
|
|
|
var conversation = new Whisper.Conversation({id: conversationId});
|
|
|
|
conversation.fetch().always(function() {
|
2015-05-18 20:48:48 +00:00
|
|
|
var now = new Date().getTime();
|
2015-05-20 19:58:48 +00:00
|
|
|
var attributes = { type: 'private' };
|
2015-03-18 23:26:55 +00:00
|
|
|
if (pushMessageContent.group) {
|
|
|
|
var group_update = {};
|
2015-03-19 20:49:09 +00:00
|
|
|
attributes = {
|
|
|
|
type: 'group',
|
|
|
|
groupId: pushMessageContent.group.id,
|
|
|
|
};
|
2015-03-18 23:26:55 +00:00
|
|
|
if (pushMessageContent.group.type === textsecure.protobuf.PushMessageContent.GroupContext.Type.UPDATE) {
|
|
|
|
attributes = {
|
|
|
|
type : 'group',
|
|
|
|
groupId : pushMessageContent.group.id,
|
|
|
|
name : pushMessageContent.group.name,
|
|
|
|
avatar : pushMessageContent.group.avatar,
|
|
|
|
members : pushMessageContent.group.members,
|
|
|
|
};
|
|
|
|
group_update = conversation.changedAttributes(_.pick(pushMessageContent.group, 'name', 'avatar'));
|
|
|
|
var difference = _.difference(pushMessageContent.group.members, conversation.get('members'));
|
|
|
|
if (difference.length > 0) {
|
|
|
|
group_update.joined = difference;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (pushMessageContent.group.type === textsecure.protobuf.PushMessageContent.GroupContext.Type.QUIT) {
|
|
|
|
group_update = { left: source };
|
2015-03-19 20:49:09 +00:00
|
|
|
attributes.members = _.without(conversation.get('members'), source);
|
2015-03-18 23:26:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_.keys(group_update).length > 0) {
|
|
|
|
message.set({group_update: group_update});
|
|
|
|
}
|
|
|
|
}
|
2015-05-18 21:23:09 +00:00
|
|
|
var type = 'incoming';
|
|
|
|
if (pushMessageContent.sync) {
|
|
|
|
type = 'outgoing';
|
|
|
|
timestamp = pushMessageContent.sync.timestamp.toNumber();
|
|
|
|
|
|
|
|
// lazy hack - check for receipts that arrived early.
|
|
|
|
if (pushMessageContent.sync.destination) {
|
|
|
|
var receipt = window.receipts.findWhere({
|
|
|
|
timestamp: timestamp,
|
|
|
|
source: pushMessageContent.sync.destination
|
|
|
|
});
|
|
|
|
if (receipt) {
|
|
|
|
window.receipts.remove(receipt);
|
|
|
|
message.set({
|
|
|
|
delivered: (message.get('delivered') || 0) + 1
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else if (pushMessageContent.group.id) { // group sync
|
|
|
|
var members = conversation.get('members') || [];
|
|
|
|
var receipts = window.receipts.where({ timestamp: timestamp });
|
|
|
|
for (var i in receipts) {
|
|
|
|
if (members.indexOf(receipts[i].get('source')) > -1) {
|
|
|
|
window.receipts.remove(receipts[i]);
|
|
|
|
message.set({
|
|
|
|
delivered: (message.get('delivered') || 0) + 1
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
throw new Error('Received sync message with no destination and no group id');
|
|
|
|
}
|
|
|
|
}
|
2015-03-18 23:26:55 +00:00
|
|
|
attributes.active_at = now;
|
|
|
|
if (type === 'incoming') {
|
|
|
|
attributes.unreadCount = conversation.get('unreadCount') + 1;
|
|
|
|
}
|
|
|
|
conversation.set(attributes);
|
|
|
|
|
|
|
|
message.set({
|
|
|
|
body : pushMessageContent.body,
|
|
|
|
conversationId : conversation.id,
|
|
|
|
attachments : pushMessageContent.attachments,
|
|
|
|
decrypted_at : now,
|
|
|
|
type : type,
|
|
|
|
sent_at : timestamp,
|
|
|
|
flags : pushMessageContent.flags,
|
|
|
|
errors : []
|
|
|
|
});
|
|
|
|
|
|
|
|
if (message.get('sent_at') > conversation.get('timestamp')) {
|
|
|
|
conversation.set({
|
|
|
|
timestamp: message.get('sent_at'),
|
|
|
|
lastMessage: message.get('body')
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
conversation.save().then(function() {
|
|
|
|
message.save().then(function() {
|
2015-05-20 20:04:16 +00:00
|
|
|
extension.trigger('updateInbox'); // inbox fetch
|
2015-03-25 01:12:11 +00:00
|
|
|
if (message.isIncoming()) {
|
|
|
|
notifyConversation(message);
|
2015-05-19 20:34:51 +00:00
|
|
|
} else {
|
|
|
|
updateConversation(conversation.id);
|
2015-03-25 01:12:11 +00:00
|
|
|
}
|
2015-03-18 23:26:55 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2014-11-13 22:35:37 +00:00
|
|
|
}
|
2015-03-18 23:26:55 +00:00
|
|
|
|
2014-11-13 22:35:37 +00:00
|
|
|
});
|
2014-05-18 21:26:55 +00:00
|
|
|
|
2014-11-13 22:35:37 +00:00
|
|
|
Whisper.MessageCollection = Backbone.Collection.extend({
|
2014-12-12 03:41:40 +00:00
|
|
|
model : Message,
|
|
|
|
database : Whisper.Database,
|
|
|
|
storeName : 'messages',
|
|
|
|
comparator : 'received_at',
|
2015-03-12 00:49:01 +00:00
|
|
|
initialize : function(models, options) {
|
|
|
|
if (options) {
|
|
|
|
this.conversation = options.conversation;
|
|
|
|
}
|
|
|
|
},
|
2014-12-12 03:41:40 +00:00
|
|
|
destroyAll : function () {
|
2014-11-13 22:35:37 +00:00
|
|
|
return Promise.all(this.models.map(function(m) {
|
|
|
|
return new Promise(function(resolve, reject) {
|
|
|
|
m.destroy().then(resolve).fail(reject);
|
|
|
|
});
|
|
|
|
}));
|
2014-12-20 01:15:57 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
fetchSentAt: function(timestamp) {
|
|
|
|
return this.fetch({
|
|
|
|
index: {
|
|
|
|
// 'receipt' index on sent_at
|
|
|
|
name: 'receipt',
|
|
|
|
only: timestamp
|
|
|
|
}
|
|
|
|
});
|
2014-12-19 20:55:29 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
fetchConversation: function(conversationId, options) {
|
|
|
|
options = options || {};
|
|
|
|
options.index = {
|
|
|
|
// 'conversation' index on [conversationId, received_at]
|
|
|
|
name : 'conversation',
|
|
|
|
lower : [conversationId],
|
2015-02-18 23:07:07 +00:00
|
|
|
upper : [conversationId, Number.MAX_VALUE]
|
2014-12-19 20:55:29 +00:00
|
|
|
// SELECT messages WHERE conversationId = this.id ORDER
|
|
|
|
// received_at DESC
|
|
|
|
};
|
|
|
|
// TODO pagination/infinite scroll
|
|
|
|
// limit: 10, offset: page*10,
|
|
|
|
return this.fetch(options);
|
2015-02-18 02:03:05 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
hasKeyConflicts: function() {
|
|
|
|
return this.any(function(m) { return m.hasKeyConflicts(); });
|
2014-11-13 22:35:37 +00:00
|
|
|
}
|
|
|
|
});
|
2015-02-19 08:20:22 +00:00
|
|
|
})();
|