DB/Index Redux

This change removes the timestamp field from messages and conversations
in favor of multiple semantically named timestamp fields: sent_at,
received_at on messages; active_at on conversations. This requires/lets
us rethink and improve our indexing scheme thusly:

The inbox index on conversations will order entries by the
conversation.active_at property, which should only appear on
conversations destined for the inbox.

The receipt index will use the message.sent_at property, for effecient
lookup of outgoing messages by timestamp, for use in processing delivery
receipts.

The group index on conversation.members is multi-entry, meaning that
looking up any phone number in this index will efficiently yield all
groups the number belongs to.

The conversation index lets us scan messages in a single conversation,
in the order they were received (or the reverse order). It is a compound
index on [conversationId, received_at].
This commit is contained in:
lilia 2014-12-11 19:41:40 -08:00
parent 9c736df7d0
commit 006653ed8e
5 changed files with 58 additions and 35 deletions

View file

@ -24,12 +24,14 @@
{ {
version: "1.0", version: "1.0",
migrate: function(transaction, next) { migrate: function(transaction, next) {
console.log('migratetion 1.0'); console.log('migration 1.0');
var messages = transaction.db.createObjectStore("messages"); var messages = transaction.db.createObjectStore("messages");
messages.createIndex("conversation", "conversationId", { unique: false }); messages.createIndex("conversation", ["conversationId", "received_at"], { unique: false });
messages.createIndex("receipt", "sent_at", { unique: false });
var conversations = transaction.db.createObjectStore("conversations"); var conversations = transaction.db.createObjectStore("conversations");
conversations.createIndex("timestamp", "timestamp", { unique: false }); conversations.createIndex("inbox", "active_at", { unique: false });
conversations.createIndex("group", "members", { unique: false, multiEntry: true });
next(); next();
} }
} }

View file

@ -22,12 +22,13 @@
database: Whisper.Database, database: Whisper.Database,
storeName: 'conversations', storeName: 'conversations',
defaults: function() { defaults: function() {
var timestamp = new Date().getTime();
return { return {
name: 'New Conversation', name : 'New Conversation',
image: '/images/default.png', image : '/images/default.png',
unreadCount: 0, unreadCount : 0,
timestamp: new Date().getTime(), timestamp : timestamp,
active: true active_at : timestamp
}; };
}, },
@ -42,18 +43,20 @@
}, },
sendMessage: function(message, attachments) { sendMessage: function(message, attachments) {
var timestamp = Date.now(); var now = Date.now();
this.messageCollection.add({ this.messageCollection.add({
body : message, body : message,
timestamp : timestamp, conversationId : this.id,
conversationId : this.id, type : 'outgoing',
type : 'outgoing', attachments : attachments,
attachments : attachments, sent_at : now,
received_at : now
}).save(); }).save();
this.save({ timestamp: timestamp, this.save({
unreadCount: 0, unreadCount : 0,
active: true}); active_at : now
});
if (this.get('type') == 'private') { if (this.get('type') == 'private') {
return textsecure.messaging.sendMessageToNumber(this.get('id'), message, attachments); return textsecure.messaging.sendMessageToNumber(this.get('id'), message, attachments);
@ -85,15 +88,29 @@
fetchMessages: function(options) { fetchMessages: function(options) {
options = options || {}; options = options || {};
options.conditions = {conversationId: this.id }; options.index = {
// 'conversation' index on conversationId
// WHERE conversationId = this.id ORDER received_at DESC
name : 'conversation',
lower : [this.id],
upper : [this.id, Number.MAX_VALUE],
order : 'desc'
};
return this.messageCollection.fetch(options); return this.messageCollection.fetch(options);
// TODO pagination/infinite scroll
// limit: 10, offset: page*10,
},
archive: function() {
this.unset('active_at');
}, },
destroyMessages: function() { destroyMessages: function() {
var models = this.messageCollection.models; var models = this.messageCollection.models;
this.messageCollection.reset([]); this.messageCollection.reset([]);
_.each(models, function(message) { message.destroy(); }); _.each(models, function(message) { message.destroy(); });
return this.save({active: false}); this.archive();
return this.save();
} }
}); });
@ -110,7 +127,7 @@
var attributes = {}; var attributes = {};
attributes = { attributes = {
name : name, name : name,
numbers : recipients, members : recipients,
type : 'group', type : 'group',
}; };
var conversation = this.add(attributes, {merge: true}); var conversation = this.add(attributes, {merge: true});

View file

@ -19,16 +19,16 @@
window.Whisper = window.Whisper || {}; window.Whisper = window.Whisper || {};
var Message = Backbone.Model.extend({ var Message = Backbone.Model.extend({
database: Whisper.Database, database : Whisper.Database,
storeName: 'messages', storeName : 'messages',
defaults: function() { defaults : function() {
return { return {
timestamp: new Date().getTime(), timestamp: new Date().getTime(),
attachments: [] attachments: []
}; };
}, },
validate: function(attributes, options) { validate: function(attributes, options) {
var required = ['timestamp', 'conversationId']; var required = ['conversationId', 'received_at', 'sent_at'];
var missing = _.filter(required, function(attr) { return !attributes[attr]; }); var missing = _.filter(required, function(attr) { return !attributes[attr]; });
if (missing.length) { if (missing.length) {
console.log("Message missing attributes: " + missing); console.log("Message missing attributes: " + missing);
@ -37,11 +37,11 @@
}); });
Whisper.MessageCollection = Backbone.Collection.extend({ Whisper.MessageCollection = Backbone.Collection.extend({
model: Message, model : Message,
database: Whisper.Database, database : Whisper.Database,
storeName: 'messages', storeName : 'messages',
comparator: 'timestamp', comparator : 'received_at',
destroyAll: function () { destroyAll : function () {
return Promise.all(this.models.map(function(m) { return Promise.all(this.models.map(function(m) {
return new Promise(function(resolve, reject) { return new Promise(function(resolve, reject) {
m.destroy().then(resolve).fail(reject); m.destroy().then(resolve).fail(reject);

View file

@ -32,11 +32,15 @@
collection : this.conversations collection : this.conversations
}); });
this.conversations.fetch({ reset: true }).then(function() { this.conversations.fetch({
this.conversations.reset( index: {
//TODO: Add an index to support this operation at the db level name: 'inbox', // 'inbox' index on active_at
this.conversations.filter(function(c) { return c.get('active'); }) order: 'desc' // ORDER timestamp DESC
); },
reset: true
// TODO pagination/infinite scroll
// limit: 10, offset: page*10,
}).then(function() {
if (this.conversations.length) { if (this.conversations.length) {
this.conversations.at(0).trigger('render'); this.conversations.at(0).trigger('render');
} }

View file

@ -73,7 +73,7 @@
this.$el.html( this.$el.html(
Mustache.render(this.template, { Mustache.render(this.template, {
message: this.model.get('body'), message: this.model.get('body'),
timestamp: moment(this.model.get('timestamp')).fromNow(), timestamp: moment(this.model.get('received_at')).fromNow(),
bubble_class: this.model.get('type') === 'outgoing' ? 'sent' : 'incoming', bubble_class: this.model.get('type') === 'outgoing' ? 'sent' : 'incoming',
sender: this.model.get('sender') sender: this.model.get('sender')
}) })