72b7e4ec34
* Mark group as left = false if it is active in contact sync * Handle expireTimer + blocked state along with contact/group sync
1284 lines
43 KiB
JavaScript
1284 lines
43 KiB
JavaScript
/*
|
|
* vim: ts=4:sw=4:expandtab
|
|
*/
|
|
(function () {
|
|
'use strict';
|
|
window.Whisper = window.Whisper || {};
|
|
|
|
// TODO: Factor out private and group subclasses of Conversation
|
|
|
|
var COLORS = [
|
|
'red',
|
|
'pink',
|
|
'purple',
|
|
'deep_purple',
|
|
'indigo',
|
|
'blue',
|
|
'light_blue',
|
|
'cyan',
|
|
'teal',
|
|
'green',
|
|
'light_green',
|
|
'orange',
|
|
'deep_orange',
|
|
'amber',
|
|
'blue_grey',
|
|
];
|
|
|
|
function constantTimeEqualArrayBuffers(ab1, ab2) {
|
|
if (!(ab1 instanceof ArrayBuffer && ab2 instanceof ArrayBuffer)) {
|
|
return false;
|
|
}
|
|
if (ab1.byteLength !== ab2.byteLength) {
|
|
return false;
|
|
}
|
|
var result = true;
|
|
var ta1 = new Uint8Array(ab1);
|
|
var ta2 = new Uint8Array(ab2);
|
|
for (var i = 0; i < ab1.byteLength; ++i) {
|
|
if (ta1[i] !== ta2[i]) { result = false; }
|
|
}
|
|
return result;
|
|
}
|
|
|
|
Whisper.Conversation = Backbone.Model.extend({
|
|
database: Whisper.Database,
|
|
storeName: 'conversations',
|
|
defaults: function() {
|
|
return {
|
|
unreadCount: 0,
|
|
verified: textsecure.storage.protocol.VerifiedStatus.DEFAULT
|
|
};
|
|
},
|
|
|
|
idForLogging: function() {
|
|
if (this.isPrivate()) {
|
|
return this.id;
|
|
}
|
|
|
|
return 'group(' + this.id + ')';
|
|
},
|
|
|
|
handleMessageError: function(message, errors) {
|
|
this.trigger('messageError', message, errors);
|
|
},
|
|
|
|
initialize: function() {
|
|
this.ourNumber = textsecure.storage.user.getNumber();
|
|
this.verifiedEnum = textsecure.storage.protocol.VerifiedStatus;
|
|
|
|
// This may be overridden by ConversationController.getOrCreate, and signify
|
|
// our first save to the database. Or first fetch from the database.
|
|
this.initialPromise = Promise.resolve();
|
|
|
|
this.contactCollection = new Backbone.Collection();
|
|
var collator = new Intl.Collator();
|
|
this.contactCollection.comparator = function(left, right) {
|
|
left = left.getTitle().toLowerCase();
|
|
right = right.getTitle().toLowerCase();
|
|
return collator.compare(left, right);
|
|
};
|
|
this.messageCollection = new Whisper.MessageCollection([], {
|
|
conversation: this
|
|
});
|
|
|
|
this.messageCollection.on('change:errors', this.handleMessageError, this);
|
|
this.messageCollection.on('send-error', this.onMessageError, this);
|
|
|
|
this.on('change:avatar', this.updateAvatarUrl);
|
|
this.on('change:profileAvatar', this.updateAvatarUrl);
|
|
this.on('change:profileKey', this.onChangeProfileKey);
|
|
this.on('destroy', this.revokeAvatarUrl);
|
|
},
|
|
|
|
isMe: function() {
|
|
return this.id === this.ourNumber;
|
|
},
|
|
|
|
onMessageError: function() {
|
|
this.updateVerified();
|
|
},
|
|
safeGetVerified: function() {
|
|
return textsecure.storage.protocol.getVerified(this.id).catch(function() {
|
|
return textsecure.storage.protocol.VerifiedStatus.DEFAULT;
|
|
});
|
|
},
|
|
updateVerified: function() {
|
|
if (this.isPrivate()) {
|
|
return Promise.all([
|
|
this.safeGetVerified(),
|
|
this.initialPromise,
|
|
]).then(function(results) {
|
|
var trust = results[0];
|
|
// we don't return here because we don't need to wait for this to finish
|
|
this.save({verified: trust});
|
|
}.bind(this));
|
|
} else {
|
|
return this.fetchContacts().then(function() {
|
|
return Promise.all(this.contactCollection.map(function(contact) {
|
|
if (!contact.isMe()) {
|
|
return contact.updateVerified();
|
|
}
|
|
}.bind(this)));
|
|
}.bind(this)).then(this.onMemberVerifiedChange.bind(this));
|
|
}
|
|
},
|
|
setVerifiedDefault: function(options) {
|
|
var DEFAULT = this.verifiedEnum.DEFAULT;
|
|
return this.queueJob(function() {
|
|
return this._setVerified(DEFAULT, options);
|
|
}.bind(this));
|
|
},
|
|
setVerified: function(options) {
|
|
var VERIFIED = this.verifiedEnum.VERIFIED;
|
|
return this.queueJob(function() {
|
|
return this._setVerified(VERIFIED, options);
|
|
}.bind(this));
|
|
},
|
|
setUnverified: function(options) {
|
|
var UNVERIFIED = this.verifiedEnum.UNVERIFIED;
|
|
return this.queueJob(function() {
|
|
return this._setVerified(UNVERIFIED, options);
|
|
}.bind(this));
|
|
},
|
|
_setVerified: function(verified, options) {
|
|
options = options || {};
|
|
_.defaults(options, {viaSyncMessage: false, viaContactSync: false, key: null});
|
|
|
|
var DEFAULT = this.verifiedEnum.DEFAULT;
|
|
var VERIFIED = this.verifiedEnum.VERIFIED;
|
|
var UNVERIFIED = this.verifiedEnum.UNVERIFIED;
|
|
|
|
if (!this.isPrivate()) {
|
|
throw new Error('You cannot verify a group conversation. ' +
|
|
'You must verify individual contacts.');
|
|
}
|
|
|
|
var beginningVerified = this.get('verified');
|
|
var promise;
|
|
if (options.viaSyncMessage) {
|
|
// handle the incoming key from the sync messages - need different
|
|
// behavior if that key doesn't match the current key
|
|
promise = textsecure.storage.protocol.processVerifiedMessage(
|
|
this.id, verified, options.key
|
|
);
|
|
} else {
|
|
promise = textsecure.storage.protocol.setVerified(
|
|
this.id, verified
|
|
);
|
|
}
|
|
|
|
var keychange;
|
|
return promise.then(function(updatedKey) {
|
|
keychange = updatedKey;
|
|
return new Promise(function(resolve) {
|
|
return this.save({verified: verified}).always(resolve);
|
|
}.bind(this));
|
|
}.bind(this)).then(function() {
|
|
// Three situations result in a verification notice in the conversation:
|
|
// 1) The message came from an explicit verification in another client (not
|
|
// a contact sync)
|
|
// 2) The verification value received by the contact sync is different
|
|
// from what we have on record (and it's not a transition to UNVERIFIED)
|
|
// 3) Our local verification status is VERIFIED and it hasn't changed,
|
|
// but the key did change (Key1/VERIFIED to Key2/VERIFIED - but we don't
|
|
// want to show DEFAULT->DEFAULT or UNVERIFIED->UNVERIFIED)
|
|
if (!options.viaContactSync
|
|
|| (beginningVerified !== verified && verified !== UNVERIFIED)
|
|
|| (keychange && verified === VERIFIED)) {
|
|
|
|
this.addVerifiedChange(this.id, verified === VERIFIED, {local: !options.viaSyncMessage});
|
|
}
|
|
if (!options.viaSyncMessage) {
|
|
return this.sendVerifySyncMessage(this.id, verified);
|
|
}
|
|
}.bind(this));
|
|
},
|
|
sendVerifySyncMessage: function(number, state) {
|
|
return textsecure.storage.protocol.loadIdentityKey(number).then(function(key) {
|
|
return textsecure.messaging.syncVerification(number, state, key);
|
|
});
|
|
},
|
|
getIdentityKeys: function() {
|
|
var lookup = {};
|
|
|
|
if (this.isPrivate()) {
|
|
return textsecure.storage.protocol.loadIdentityKey(this.id).then(function(key) {
|
|
lookup[this.id] = key;
|
|
return lookup;
|
|
}.bind(this)).catch(function(error) {
|
|
console.log(
|
|
'getIdentityKeys error for conversation',
|
|
this.idForLogging(),
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
return lookup;
|
|
}.bind(this));
|
|
} else {
|
|
return Promise.all(this.contactCollection.map(function(contact) {
|
|
return textsecure.storage.protocol.loadIdentityKey(contact.id).then(function(key) {
|
|
lookup[contact.id] = key;
|
|
}).catch(function(error) {
|
|
console.log(
|
|
'getIdentityKeys error for group member',
|
|
contact.idForLogging(),
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
});
|
|
})).then(function() {
|
|
return lookup;
|
|
});
|
|
}
|
|
},
|
|
replay: function(error, message) {
|
|
var replayable = new textsecure.ReplayableError(error);
|
|
return replayable.replay(message.attributes).catch(function(error) {
|
|
console.log(
|
|
'replay error:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
});
|
|
},
|
|
decryptOldIncomingKeyErrors: function() {
|
|
// We want to run just once per conversation
|
|
if (this.get('decryptedOldIncomingKeyErrors')) {
|
|
return Promise.resolve();
|
|
}
|
|
console.log('decryptOldIncomingKeyErrors start for', this.idForLogging());
|
|
|
|
var messages = this.messageCollection.filter(function(message) {
|
|
var errors = message.get('errors');
|
|
if (!errors || !errors[0]) {
|
|
return false;
|
|
}
|
|
var error = _.find(errors, function(error) {
|
|
return error.name === 'IncomingIdentityKeyError';
|
|
});
|
|
|
|
return Boolean(error);
|
|
});
|
|
|
|
var markComplete = function() {
|
|
console.log('decryptOldIncomingKeyErrors complete for', this.idForLogging());
|
|
return new Promise(function(resolve) {
|
|
this.save({decryptedOldIncomingKeyErrors: true}).always(resolve);
|
|
}.bind(this));
|
|
}.bind(this);
|
|
|
|
if (!messages.length) {
|
|
return markComplete();
|
|
}
|
|
|
|
console.log('decryptOldIncomingKeyErrors found', messages.length, 'messages to process');
|
|
var safeDelete = function(message) {
|
|
return new Promise(function(resolve) {
|
|
message.destroy().always(resolve);
|
|
});
|
|
};
|
|
|
|
return this.getIdentityKeys().then(function(lookup) {
|
|
return Promise.all(_.map(messages, function(message) {
|
|
var source = message.get('source');
|
|
var error = _.find(message.get('errors'), function(error) {
|
|
return error.name === 'IncomingIdentityKeyError';
|
|
});
|
|
|
|
var key = lookup[source];
|
|
if (!key) {
|
|
return;
|
|
}
|
|
|
|
if (constantTimeEqualArrayBuffers(key, error.identityKey)) {
|
|
return this.replay(error, message).then(function() {
|
|
return safeDelete(message);
|
|
});
|
|
}
|
|
}.bind(this)));
|
|
}.bind(this)).catch(function(error) {
|
|
console.log(
|
|
'decryptOldIncomingKeyErrors error:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
}).then(markComplete);
|
|
},
|
|
isVerified: function() {
|
|
if (this.isPrivate()) {
|
|
return this.get('verified') === this.verifiedEnum.VERIFIED;
|
|
} else {
|
|
if (!this.contactCollection.length) {
|
|
return false;
|
|
}
|
|
|
|
return this.contactCollection.every(function(contact) {
|
|
if (contact.isMe()) {
|
|
return true;
|
|
} else {
|
|
return contact.isVerified();
|
|
}
|
|
}.bind(this));
|
|
}
|
|
},
|
|
isUnverified: function() {
|
|
if (this.isPrivate()) {
|
|
var verified = this.get('verified');
|
|
return verified !== this.verifiedEnum.VERIFIED && verified !== this.verifiedEnum.DEFAULT;
|
|
} else {
|
|
if (!this.contactCollection.length) {
|
|
return true;
|
|
}
|
|
|
|
return this.contactCollection.any(function(contact) {
|
|
if (contact.isMe()) {
|
|
return false;
|
|
} else {
|
|
return contact.isUnverified();
|
|
}
|
|
}.bind(this));
|
|
}
|
|
},
|
|
getUnverified: function() {
|
|
if (this.isPrivate()) {
|
|
return this.isUnverified() ? new Backbone.Collection([this]) : new Backbone.Collection();
|
|
} else {
|
|
return new Backbone.Collection(this.contactCollection.filter(function(contact) {
|
|
if (contact.isMe()) {
|
|
return false;
|
|
} else {
|
|
return contact.isUnverified();
|
|
}
|
|
}.bind(this)));
|
|
}
|
|
},
|
|
setApproved: function() {
|
|
if (!this.isPrivate()) {
|
|
throw new Error('You cannot set a group conversation as trusted. ' +
|
|
'You must set individual contacts as trusted.');
|
|
}
|
|
|
|
return textsecure.storage.protocol.setApproval(this.id, true);
|
|
},
|
|
safeIsUntrusted: function() {
|
|
return textsecure.storage.protocol.isUntrusted(this.id).catch(function() {
|
|
return false;
|
|
});
|
|
},
|
|
isUntrusted: function() {
|
|
if (this.isPrivate()) {
|
|
return this.safeIsUntrusted();
|
|
} else {
|
|
if (!this.contactCollection.length) {
|
|
return Promise.resolve(false);
|
|
}
|
|
|
|
return Promise.all(this.contactCollection.map(function(contact) {
|
|
if (contact.isMe()) {
|
|
return false;
|
|
} else {
|
|
return contact.safeIsUntrusted();
|
|
}
|
|
}.bind(this))).then(function(results) {
|
|
return _.any(results, function(result) {
|
|
return result;
|
|
});
|
|
});
|
|
}
|
|
},
|
|
getUntrusted: function() {
|
|
// This is a bit ugly because isUntrusted() is async. Could do the work to cache
|
|
// it locally, but we really only need it for this call.
|
|
if (this.isPrivate()) {
|
|
return this.isUntrusted().then(function(untrusted) {
|
|
if (untrusted) {
|
|
return new Backbone.Collection([this]);
|
|
}
|
|
|
|
return new Backbone.Collection();
|
|
}.bind(this));
|
|
} else {
|
|
return Promise.all(this.contactCollection.map(function(contact) {
|
|
if (contact.isMe()) {
|
|
return [false, contact];
|
|
} else {
|
|
return Promise.all([contact.isUntrusted(), contact]);
|
|
}
|
|
}.bind(this))).then(function(results) {
|
|
results = _.filter(results, function(result) {
|
|
var untrusted = result[0];
|
|
return untrusted;
|
|
});
|
|
return new Backbone.Collection(_.map(results, function(result) {
|
|
var contact = result[1];
|
|
return contact;
|
|
}));
|
|
}.bind(this));
|
|
}
|
|
},
|
|
onMemberVerifiedChange: function() {
|
|
// If the verified state of a member changes, our aggregate state changes.
|
|
// We trigger both events to replicate the behavior of Backbone.Model.set()
|
|
this.trigger('change:verified');
|
|
this.trigger('change');
|
|
},
|
|
toggleVerified: function() {
|
|
if (this.isVerified()) {
|
|
return this.setVerifiedDefault();
|
|
} else {
|
|
return this.setVerified();
|
|
}
|
|
},
|
|
|
|
addKeyChange: function(id) {
|
|
console.log(
|
|
'adding key change advisory for',
|
|
this.idForLogging(),
|
|
id,
|
|
this.get('timestamp')
|
|
);
|
|
|
|
var timestamp = Date.now();
|
|
var message = new Whisper.Message({
|
|
conversationId : this.id,
|
|
type : 'keychange',
|
|
sent_at : this.get('timestamp'),
|
|
received_at : timestamp,
|
|
key_changed : id,
|
|
unread : 1
|
|
});
|
|
message.save().then(this.trigger.bind(this,'newmessage', message));
|
|
},
|
|
addVerifiedChange: function(id, verified, options) {
|
|
options = options || {};
|
|
_.defaults(options, {local: true});
|
|
|
|
if (this.isMe()) {
|
|
console.log('refusing to add verified change advisory for our own number');
|
|
return;
|
|
}
|
|
|
|
var lastMessage = this.get('timestamp') || Date.now();
|
|
|
|
console.log(
|
|
'adding verified change advisory for',
|
|
this.idForLogging(),
|
|
id,
|
|
lastMessage
|
|
);
|
|
|
|
var timestamp = Date.now();
|
|
var message = new Whisper.Message({
|
|
conversationId : this.id,
|
|
type : 'verified-change',
|
|
sent_at : lastMessage,
|
|
received_at : timestamp,
|
|
verifiedChanged : id,
|
|
verified : verified,
|
|
local : options.local,
|
|
unread : 1
|
|
});
|
|
message.save().then(this.trigger.bind(this,'newmessage', message));
|
|
|
|
if (this.isPrivate()) {
|
|
ConversationController.getAllGroupsInvolvingId(id).then(function(groups) {
|
|
_.forEach(groups, function(group) {
|
|
group.addVerifiedChange(id, verified, options);
|
|
});
|
|
});
|
|
}
|
|
},
|
|
|
|
onReadMessage: function(message) {
|
|
if (this.messageCollection.get(message.id)) {
|
|
this.messageCollection.get(message.id).fetch();
|
|
}
|
|
|
|
// We mark as read everything older than this message - to clean up old stuff
|
|
// still marked unread in the database. If the user generally doesn't read in
|
|
// the desktop app, so the desktop app only gets read syncs, we can very
|
|
// easily end up with messages never marked as read (our previous early read
|
|
// sync handling, read syncs never sent because app was offline)
|
|
|
|
// We queue it because we often get a whole lot of read syncs at once, and
|
|
// their markRead calls could very easily overlap given the async pull from DB.
|
|
|
|
// Lastly, we don't send read syncs for any message marked read due to a read
|
|
// sync. That's a notification explosion we don't need.
|
|
return this.queueJob(function() {
|
|
return this.markRead(message.get('received_at'), {sendReadReceipts: false});
|
|
}.bind(this));
|
|
},
|
|
|
|
getUnread: function() {
|
|
var conversationId = this.id;
|
|
var unreadMessages = new Whisper.MessageCollection();
|
|
return new Promise(function(resolve) {
|
|
return unreadMessages.fetch({
|
|
index: {
|
|
// 'unread' index
|
|
name : 'unread',
|
|
lower : [conversationId],
|
|
upper : [conversationId, Number.MAX_VALUE],
|
|
}
|
|
}).always(function() {
|
|
resolve(unreadMessages);
|
|
});
|
|
});
|
|
|
|
},
|
|
|
|
validate: function(attributes, options) {
|
|
var required = ['id', 'type'];
|
|
var missing = _.filter(required, function(attr) { return !attributes[attr]; });
|
|
if (missing.length) { return "Conversation must have " + missing; }
|
|
|
|
if (attributes.type !== 'private' && attributes.type !== 'group') {
|
|
return "Invalid conversation type: " + attributes.type;
|
|
}
|
|
|
|
var error = this.validateNumber();
|
|
if (error) { return error; }
|
|
|
|
this.updateTokens();
|
|
},
|
|
|
|
validateNumber: function() {
|
|
if (this.isPrivate()) {
|
|
var regionCode = storage.get('regionCode');
|
|
var number = libphonenumber.util.parseNumber(this.id, regionCode);
|
|
if (number.isValidNumber) {
|
|
this.set({ id: number.e164 });
|
|
} else {
|
|
return number.error || "Invalid phone number";
|
|
}
|
|
}
|
|
},
|
|
|
|
updateTokens: function() {
|
|
var tokens = [];
|
|
var name = this.get('name');
|
|
if (typeof name === 'string') {
|
|
tokens.push(name.toLowerCase());
|
|
tokens = tokens.concat(name.trim().toLowerCase().split(/[\s\-_\(\)\+]+/));
|
|
}
|
|
if (this.isPrivate()) {
|
|
var regionCode = storage.get('regionCode');
|
|
var number = libphonenumber.util.parseNumber(this.id, regionCode);
|
|
tokens.push(
|
|
number.nationalNumber,
|
|
number.countryCode + number.nationalNumber
|
|
);
|
|
}
|
|
this.set({tokens: tokens});
|
|
},
|
|
|
|
queueJob: function(callback) {
|
|
var previous = this.pending || Promise.resolve();
|
|
|
|
var taskWithTimeout = textsecure.createTaskWithTimeout(
|
|
callback,
|
|
'conversation ' + this.idForLogging()
|
|
);
|
|
|
|
var current = this.pending = previous.then(taskWithTimeout, taskWithTimeout);
|
|
|
|
current.then(function() {
|
|
if (this.pending === current) {
|
|
delete this.pending;
|
|
}
|
|
}.bind(this));
|
|
|
|
return current;
|
|
},
|
|
|
|
getRecipients: function() {
|
|
if (this.isPrivate()) {
|
|
return [ this.id ];
|
|
} else {
|
|
var me = textsecure.storage.user.getNumber();
|
|
return _.without(this.get('members'), me);
|
|
}
|
|
},
|
|
|
|
sendMessage: function(body, attachments) {
|
|
this.queueJob(function() {
|
|
var now = Date.now();
|
|
|
|
console.log(
|
|
'Sending message to conversation',
|
|
this.idForLogging(),
|
|
'with timestamp',
|
|
now
|
|
);
|
|
|
|
var message = this.messageCollection.add({
|
|
body : body,
|
|
conversationId : this.id,
|
|
type : 'outgoing',
|
|
attachments : attachments,
|
|
sent_at : now,
|
|
received_at : now,
|
|
expireTimer : this.get('expireTimer'),
|
|
recipients : this.getRecipients()
|
|
});
|
|
if (this.isPrivate()) {
|
|
message.set({destination: this.id});
|
|
}
|
|
message.save();
|
|
|
|
this.save({
|
|
active_at : now,
|
|
timestamp : now,
|
|
lastMessage : message.getNotificationText()
|
|
});
|
|
|
|
var sendFunc;
|
|
if (this.get('type') == 'private') {
|
|
sendFunc = textsecure.messaging.sendMessageToNumber;
|
|
}
|
|
else {
|
|
sendFunc = textsecure.messaging.sendMessageToGroup;
|
|
}
|
|
|
|
var profileKey;
|
|
if (this.get('profileSharing')) {
|
|
profileKey = storage.get('profileKey');
|
|
}
|
|
|
|
message.send(sendFunc(this.get('id'), body, attachments, now, this.get('expireTimer'), profileKey));
|
|
}.bind(this));
|
|
},
|
|
|
|
updateLastMessage: function() {
|
|
var collection = new Whisper.MessageCollection();
|
|
return collection.fetchConversation(this.id, 1).then(function() {
|
|
var lastMessage = collection.at(0);
|
|
if (lastMessage) {
|
|
if (lastMessage.get('type') === 'verified-change') {
|
|
return;
|
|
}
|
|
this.set({
|
|
lastMessage : lastMessage.getNotificationText(),
|
|
timestamp : lastMessage.get('sent_at')
|
|
});
|
|
} else {
|
|
this.set({ lastMessage: '', timestamp: null });
|
|
}
|
|
if (this.hasChanged('lastMessage') || this.hasChanged('timestamp')) {
|
|
this.save();
|
|
}
|
|
}.bind(this));
|
|
},
|
|
|
|
updateExpirationTimer: function(expireTimer, source, received_at, options) {
|
|
options = options || {};
|
|
_.defaults(options, {fromSync: false});
|
|
|
|
if (!expireTimer) {
|
|
expireTimer = null;
|
|
}
|
|
if (this.get('expireTimer') === expireTimer
|
|
|| (!expireTimer && !this.get('expireTimer'))) {
|
|
|
|
return;
|
|
}
|
|
|
|
console.log(
|
|
'Updating expireTimer for conversation',
|
|
this.idForLogging(),
|
|
'via',
|
|
source
|
|
);
|
|
source = source || textsecure.storage.user.getNumber();
|
|
var timestamp = received_at || Date.now();
|
|
|
|
var message = this.messageCollection.add({
|
|
conversationId : this.id,
|
|
type : received_at ? 'incoming' : 'outgoing',
|
|
sent_at : timestamp,
|
|
received_at : timestamp,
|
|
flags : textsecure.protobuf.DataMessage.Flags.EXPIRATION_TIMER_UPDATE,
|
|
expirationTimerUpdate : {
|
|
expireTimer : expireTimer,
|
|
source : source,
|
|
fromSync : options.fromSync,
|
|
}
|
|
});
|
|
if (this.isPrivate()) {
|
|
message.set({destination: this.id});
|
|
}
|
|
if (message.isOutgoing()) {
|
|
message.set({recipients: this.getRecipients() });
|
|
}
|
|
|
|
return Promise.all([
|
|
wrapDeferred(message.save()),
|
|
wrapDeferred(this.save({ expireTimer: expireTimer })),
|
|
]).then(function() {
|
|
if (message.isIncoming()) {
|
|
return message;
|
|
}
|
|
|
|
// change was made locally, send it to the number/group
|
|
var sendFunc;
|
|
if (this.get('type') == 'private') {
|
|
sendFunc = textsecure.messaging.sendExpirationTimerUpdateToNumber;
|
|
}
|
|
else {
|
|
sendFunc = textsecure.messaging.sendExpirationTimerUpdateToGroup;
|
|
}
|
|
var profileKey;
|
|
if (this.get('profileSharing')) {
|
|
profileKey = storage.get('profileKey');
|
|
}
|
|
var promise = sendFunc(this.get('id'),
|
|
this.get('expireTimer'),
|
|
message.get('sent_at'),
|
|
profileKey
|
|
);
|
|
|
|
return message.send(promise).then(function() {
|
|
return message;
|
|
});
|
|
}.bind(this));
|
|
},
|
|
|
|
isSearchable: function() {
|
|
return !this.get('left') || !!this.get('lastMessage');
|
|
},
|
|
|
|
endSession: function() {
|
|
if (this.isPrivate()) {
|
|
var now = Date.now();
|
|
var message = this.messageCollection.create({
|
|
conversationId : this.id,
|
|
type : 'outgoing',
|
|
sent_at : now,
|
|
received_at : now,
|
|
destination : this.id,
|
|
recipients : this.getRecipients(),
|
|
flags : textsecure.protobuf.DataMessage.Flags.END_SESSION
|
|
});
|
|
message.send(textsecure.messaging.resetSession(this.id, now));
|
|
}
|
|
|
|
},
|
|
|
|
updateGroup: function(group_update) {
|
|
if (this.isPrivate()) {
|
|
throw new Error("Called update group on private conversation");
|
|
}
|
|
if (group_update === undefined) {
|
|
group_update = this.pick(['name', 'avatar', 'members']);
|
|
}
|
|
var now = Date.now();
|
|
var message = this.messageCollection.create({
|
|
conversationId : this.id,
|
|
type : 'outgoing',
|
|
sent_at : now,
|
|
received_at : now,
|
|
group_update : group_update
|
|
});
|
|
message.send(textsecure.messaging.updateGroup(
|
|
this.id,
|
|
this.get('name'),
|
|
this.get('avatar'),
|
|
this.get('members')
|
|
));
|
|
},
|
|
|
|
leaveGroup: function() {
|
|
var now = Date.now();
|
|
if (this.get('type') === 'group') {
|
|
this.save({left: true});
|
|
var message = this.messageCollection.create({
|
|
group_update: { left: 'You' },
|
|
conversationId : this.id,
|
|
type : 'outgoing',
|
|
sent_at : now,
|
|
received_at : now
|
|
});
|
|
message.send(textsecure.messaging.leaveGroup(this.id));
|
|
}
|
|
},
|
|
|
|
markRead: function(newestUnreadDate, options) {
|
|
options = options || {};
|
|
_.defaults(options, {sendReadReceipts: true});
|
|
|
|
var conversationId = this.id;
|
|
Whisper.Notifications.remove(Whisper.Notifications.where({
|
|
conversationId: conversationId
|
|
}));
|
|
|
|
return this.getUnread().then(function(unreadMessages) {
|
|
var promises = [];
|
|
var oldUnread = unreadMessages.filter(function(message) {
|
|
return message.get('received_at') <= newestUnreadDate;
|
|
});
|
|
|
|
var read = _.map(oldUnread, function(m) {
|
|
if (this.messageCollection.get(m.id)) {
|
|
m = this.messageCollection.get(m.id);
|
|
} else {
|
|
console.log('Marked a message as read in the database, but ' +
|
|
'it was not in messageCollection.');
|
|
}
|
|
promises.push(m.markRead());
|
|
var errors = m.get('errors');
|
|
return {
|
|
sender : m.get('source'),
|
|
timestamp : m.get('sent_at'),
|
|
hasErrors : Boolean(errors && errors.length)
|
|
};
|
|
}.bind(this));
|
|
|
|
// Some messages we're marking read are local notifications with no sender
|
|
read = _.filter(read, function(m) {
|
|
return Boolean(m.sender);
|
|
});
|
|
unreadMessages = unreadMessages.filter(function(m) {
|
|
return Boolean(m.isIncoming());
|
|
});
|
|
|
|
var unreadCount = unreadMessages.length - read.length;
|
|
var promise = new Promise(function(resolve, reject) {
|
|
this.save({ unreadCount: unreadCount }).then(resolve, reject);
|
|
}.bind(this));
|
|
promises.push(promise);
|
|
|
|
// If a message has errors, we don't want to send anything out about it.
|
|
// read syncs - let's wait for a client that really understands the message
|
|
// to mark it read. we'll mark our local error read locally, though.
|
|
// read receipts - here we can run into infinite loops, where each time the
|
|
// conversation is viewed, another error message shows up for the contact
|
|
read = read.filter(function(item) {
|
|
return !item.hasErrors;
|
|
});
|
|
|
|
if (read.length && options.sendReadReceipts) {
|
|
console.log('Sending', read.length, 'read receipts');
|
|
promises.push(textsecure.messaging.syncReadMessages(read));
|
|
|
|
if (storage.get('read-receipt-setting')) {
|
|
_.each(_.groupBy(read, 'sender'), function(receipts, sender) {
|
|
var timestamps = _.map(receipts, 'timestamp');
|
|
promises.push(textsecure.messaging.sendReadReceipts(sender, timestamps));
|
|
});
|
|
}
|
|
}
|
|
|
|
return Promise.all(promises);
|
|
}.bind(this));
|
|
},
|
|
|
|
onChangeProfileKey: function() {
|
|
if (this.isPrivate()) {
|
|
this.getProfiles();
|
|
}
|
|
},
|
|
|
|
getProfiles: function() {
|
|
// request all conversation members' keys
|
|
var ids = [];
|
|
if (this.isPrivate()) {
|
|
ids = [this.id];
|
|
} else {
|
|
ids = this.get('members');
|
|
}
|
|
return Promise.all(_.map(ids, this.getProfile));
|
|
},
|
|
|
|
getProfile: function(id) {
|
|
return textsecure.messaging.getProfile(id).then(function(profile) {
|
|
var identityKey = dcodeIO.ByteBuffer.wrap(profile.identityKey, 'base64').toArrayBuffer();
|
|
|
|
return textsecure.storage.protocol.saveIdentity(
|
|
id + '.1', identityKey, false
|
|
).then(function(changed) {
|
|
if (changed) {
|
|
// save identity will close all sessions except for .1, so we
|
|
// must close that one manually.
|
|
var address = new libsignal.SignalProtocolAddress(id, 1);
|
|
console.log('closing session for', address.toString());
|
|
var sessionCipher = new libsignal.SessionCipher(textsecure.storage.protocol, address);
|
|
return sessionCipher.closeOpenSessionForDevice();
|
|
}
|
|
}).then(function() {
|
|
var c = ConversationController.get(id);
|
|
return Promise.all([
|
|
c.setProfileName(profile.name),
|
|
c.setProfileAvatar(profile.avatar)
|
|
]).then(function() {
|
|
// success
|
|
return new Promise(function(resolve, reject) {
|
|
c.save().then(resolve, reject);
|
|
});
|
|
}, function(e) {
|
|
// fail
|
|
if (e.name === 'ProfileDecryptError') {
|
|
// probably the profile key has changed.
|
|
console.log(
|
|
'decryptProfile error:',
|
|
id,
|
|
profile,
|
|
e && e.stack ? e.stack : e
|
|
);
|
|
}
|
|
});
|
|
}.bind(this));
|
|
}.bind(this)).catch(function(error) {
|
|
console.log(
|
|
'getProfile error:',
|
|
error && error.stack ? error.stack : error
|
|
);
|
|
});
|
|
},
|
|
setProfileName: function(encryptedName) {
|
|
var key = this.get('profileKey');
|
|
if (!key) { return; }
|
|
|
|
try {
|
|
// decode
|
|
var data = dcodeIO.ByteBuffer.wrap(encryptedName, 'base64').toArrayBuffer();
|
|
|
|
// decrypt
|
|
return textsecure.crypto.decryptProfileName(data, key).then(function(decrypted) {
|
|
|
|
// encode
|
|
var name = dcodeIO.ByteBuffer.wrap(decrypted).toString('utf8');
|
|
|
|
// set
|
|
this.set({profileName: name});
|
|
}.bind(this));
|
|
}
|
|
catch (e) {
|
|
return Promise.reject(e);
|
|
}
|
|
},
|
|
setProfileAvatar: function(avatarPath) {
|
|
if (!avatarPath) { return; }
|
|
return textsecure.messaging.getAvatar(avatarPath).then(function(avatar) {
|
|
var key = this.get('profileKey');
|
|
if (!key) { return; }
|
|
// decrypt
|
|
return textsecure.crypto.decryptProfile(avatar, key).then(function(decrypted) {
|
|
// set
|
|
this.set({
|
|
profileAvatar: {
|
|
data: decrypted,
|
|
contentType: 'image/jpeg',
|
|
size: decrypted.byteLength
|
|
}
|
|
});
|
|
}.bind(this));
|
|
}.bind(this));
|
|
},
|
|
setProfileKey: function(key) {
|
|
return new Promise(function(resolve, reject) {
|
|
if (!constantTimeEqualArrayBuffers(this.get('profileKey'), key)) {
|
|
this.save({profileKey: key}).then(resolve, reject);
|
|
} else {
|
|
resolve();
|
|
}
|
|
}.bind(this));
|
|
},
|
|
|
|
fetchMessages: function() {
|
|
if (!this.id) {
|
|
return Promise.reject('This conversation has no id!');
|
|
}
|
|
return this.messageCollection.fetchConversation(this.id, null, this.get('unreadCount'));
|
|
},
|
|
|
|
hasMember: function(number) {
|
|
return _.contains(this.get('members'), number);
|
|
},
|
|
fetchContacts: function(options) {
|
|
if (this.isPrivate()) {
|
|
this.contactCollection.reset([this]);
|
|
return Promise.resolve();
|
|
} else {
|
|
var members = this.get('members') || [];
|
|
var promises = members.map(function(number) {
|
|
return ConversationController.getOrCreateAndWait(number, 'private');
|
|
});
|
|
|
|
return Promise.all(promises).then(function(contacts) {
|
|
_.forEach(contacts, function(contact) {
|
|
this.listenTo(contact, 'change:verified', this.onMemberVerifiedChange);
|
|
}.bind(this));
|
|
|
|
this.contactCollection.reset(contacts);
|
|
}.bind(this));
|
|
}
|
|
},
|
|
|
|
destroyMessages: function() {
|
|
this.messageCollection.fetch({
|
|
index: {
|
|
// 'conversation' index on [conversationId, received_at]
|
|
name : 'conversation',
|
|
lower : [this.id],
|
|
upper : [this.id, Number.MAX_VALUE],
|
|
}
|
|
}).then(function() {
|
|
var models = this.messageCollection.models;
|
|
this.messageCollection.reset([]);
|
|
_.each(models, function(message) {
|
|
message.destroy();
|
|
});
|
|
this.save({
|
|
lastMessage: null,
|
|
timestamp: null,
|
|
active_at: null,
|
|
});
|
|
}.bind(this));
|
|
},
|
|
|
|
getName: function() {
|
|
if (this.isPrivate()) {
|
|
return this.get('name');
|
|
} else {
|
|
return this.get('name') || 'Unknown group';
|
|
}
|
|
},
|
|
|
|
getTitle: function() {
|
|
if (this.isPrivate()) {
|
|
return this.get('name') || this.getNumber();
|
|
} else {
|
|
return this.get('name') || 'Unknown group';
|
|
}
|
|
},
|
|
|
|
getProfileName: function() {
|
|
if (this.isPrivate() && !this.get('name')) {
|
|
return this.get('profileName');
|
|
}
|
|
},
|
|
|
|
getDisplayName: function() {
|
|
if (!this.isPrivate()) {
|
|
return this.getTitle();
|
|
}
|
|
|
|
var name = this.get('name');
|
|
if (name) {
|
|
return name;
|
|
}
|
|
|
|
var profileName = this.get('profileName');
|
|
if (profileName) {
|
|
return this.getNumber() + ' ~' + profileName;
|
|
}
|
|
|
|
return this.getNumber();
|
|
},
|
|
|
|
getNumber: function() {
|
|
if (!this.isPrivate()) {
|
|
return '';
|
|
}
|
|
var number = this.id;
|
|
try {
|
|
var parsedNumber = libphonenumber.parse(number);
|
|
var regionCode = libphonenumber.getRegionCodeForNumber(parsedNumber);
|
|
if (regionCode === storage.get('regionCode')) {
|
|
return libphonenumber.format(parsedNumber, libphonenumber.PhoneNumberFormat.NATIONAL);
|
|
} else {
|
|
return libphonenumber.format(parsedNumber, libphonenumber.PhoneNumberFormat.INTERNATIONAL);
|
|
}
|
|
} catch (e) {
|
|
return number;
|
|
}
|
|
},
|
|
|
|
isPrivate: function() {
|
|
return this.get('type') === 'private';
|
|
},
|
|
|
|
revokeAvatarUrl: function() {
|
|
if (this.avatarUrl) {
|
|
URL.revokeObjectURL(this.avatarUrl);
|
|
this.avatarUrl = null;
|
|
}
|
|
},
|
|
|
|
updateAvatarUrl: function(silent) {
|
|
this.revokeAvatarUrl();
|
|
var avatar = this.get('avatar') || this.get('profileAvatar');
|
|
if (avatar) {
|
|
this.avatarUrl = URL.createObjectURL(
|
|
new Blob([avatar.data], {type: avatar.contentType})
|
|
);
|
|
} else {
|
|
this.avatarUrl = null;
|
|
}
|
|
if (!silent) {
|
|
this.trigger('change');
|
|
}
|
|
},
|
|
getColor: function() {
|
|
var title = this.get('name');
|
|
var color = this.get('color');
|
|
if (!color) {
|
|
if (this.isPrivate()) {
|
|
if (title) {
|
|
color = COLORS[Math.abs(this.hashCode()) % 15];
|
|
} else {
|
|
color = 'grey';
|
|
}
|
|
} else {
|
|
color = 'default';
|
|
}
|
|
}
|
|
return color;
|
|
},
|
|
getAvatar: function() {
|
|
if (this.avatarUrl === undefined) {
|
|
this.updateAvatarUrl(true);
|
|
}
|
|
|
|
var title = this.get('name');
|
|
var color = this.getColor();
|
|
|
|
if (this.avatarUrl) {
|
|
return { url: this.avatarUrl, color: color };
|
|
} else if (this.isPrivate()) {
|
|
return {
|
|
color: color,
|
|
content: title ? title.trim()[0] : '#'
|
|
};
|
|
} else {
|
|
return { url: 'images/group_default.png', color: color };
|
|
}
|
|
},
|
|
|
|
getNotificationIcon: function() {
|
|
return new Promise(function(resolve) {
|
|
var avatar = this.getAvatar();
|
|
if (avatar.url) {
|
|
resolve(avatar.url);
|
|
} else {
|
|
resolve(new Whisper.IdenticonSVGView(avatar).getDataUrl());
|
|
}
|
|
}.bind(this));
|
|
},
|
|
|
|
notify: function(message) {
|
|
if (!message.isIncoming()) {
|
|
return Promise.resolve();
|
|
}
|
|
var conversationId = this.id;
|
|
|
|
return ConversationController.getOrCreateAndWait(message.get('source'), 'private')
|
|
.then(function(sender) {
|
|
return sender.getNotificationIcon().then(function(iconUrl) {
|
|
console.log('adding notification');
|
|
Whisper.Notifications.add({
|
|
title : sender.getTitle(),
|
|
message : message.getNotificationText(),
|
|
iconUrl : iconUrl,
|
|
imageUrl : message.getImageUrl(),
|
|
conversationId : conversationId,
|
|
messageId : message.id
|
|
});
|
|
});
|
|
});
|
|
},
|
|
hashCode: function() {
|
|
if (this.hash === undefined) {
|
|
var string = this.getTitle() || '';
|
|
if (string.length === 0) {
|
|
return 0;
|
|
}
|
|
var hash = 0;
|
|
for (var i = 0; i < string.length; i++) {
|
|
hash = ((hash<<5)-hash) + string.charCodeAt(i);
|
|
hash = hash & hash; // Convert to 32bit integer
|
|
}
|
|
|
|
this.hash = hash;
|
|
}
|
|
return this.hash;
|
|
}
|
|
});
|
|
|
|
Whisper.ConversationCollection = Backbone.Collection.extend({
|
|
database: Whisper.Database,
|
|
storeName: 'conversations',
|
|
model: Whisper.Conversation,
|
|
|
|
comparator: function(m) {
|
|
return -m.get('timestamp');
|
|
},
|
|
|
|
destroyAll: function () {
|
|
return Promise.all(this.models.map(function(m) {
|
|
return new Promise(function(resolve, reject) {
|
|
m.destroy().then(resolve).fail(reject);
|
|
});
|
|
}));
|
|
},
|
|
|
|
search: function(query) {
|
|
query = query.trim().toLowerCase();
|
|
if (query.length > 0) {
|
|
query = query.replace(/[-.\(\)]*/g,'').replace(/^\+(\d*)$/, '$1');
|
|
var lastCharCode = query.charCodeAt(query.length - 1);
|
|
var nextChar = String.fromCharCode(lastCharCode + 1);
|
|
var upper = query.slice(0, -1) + nextChar;
|
|
return new Promise(function(resolve) {
|
|
this.fetch({
|
|
index: {
|
|
name: 'search', // 'search' index on tokens array
|
|
lower: query,
|
|
upper: upper,
|
|
excludeUpper: true
|
|
}
|
|
}).always(resolve);
|
|
}.bind(this));
|
|
}
|
|
},
|
|
|
|
fetchAlphabetical: function() {
|
|
return new Promise(function(resolve) {
|
|
this.fetch({
|
|
index: {
|
|
name: 'search', // 'search' index on tokens array
|
|
},
|
|
limit: 100
|
|
}).always(resolve);
|
|
}.bind(this));
|
|
},
|
|
|
|
fetchGroups: function(number) {
|
|
return new Promise(function(resolve) {
|
|
this.fetch({
|
|
index: {
|
|
name: 'group',
|
|
only: number
|
|
}
|
|
}).always(resolve);
|
|
}.bind(this));
|
|
}
|
|
});
|
|
|
|
Whisper.Conversation.COLORS = COLORS.concat(['grey', 'default']).join(' ');
|
|
|
|
// Special collection for fetching all the groups a certain number appears in
|
|
Whisper.GroupCollection = Backbone.Collection.extend({
|
|
database: Whisper.Database,
|
|
storeName: 'conversations',
|
|
model: Whisper.Conversation,
|
|
fetchGroups: function(number) {
|
|
return new Promise(function(resolve) {
|
|
this.fetch({
|
|
index: {
|
|
name: 'group',
|
|
only: number
|
|
}
|
|
}).always(resolve);
|
|
}.bind(this));
|
|
}
|
|
});
|
|
})();
|