2014-07-29 13:28:15 -07:00
|
|
|
'use strict';
|
2014-07-31 10:06:47 -07:00
|
|
|
var soop = require('soop');
|
|
|
|
var imports = soop.imports();
|
2014-07-29 13:28:15 -07:00
|
|
|
var levelup = require('levelup');
|
|
|
|
var config = require('../config/config');
|
|
|
|
var Rpc = imports.rpc || require('./Rpc');
|
|
|
|
var async = require('async');
|
|
|
|
var logger = require('./logger').logger;
|
2014-07-30 13:29:28 -07:00
|
|
|
var util = require('util');
|
|
|
|
var EventEmitter = require('events').EventEmitter;
|
2014-07-31 11:16:05 -07:00
|
|
|
var microtime = require('microtime');
|
2014-08-01 15:00:52 -07:00
|
|
|
var bitcore = require('bitcore');
|
|
|
|
var AuthMessage = bitcore.AuthMessage;
|
2014-08-04 11:43:38 -07:00
|
|
|
var preconditions = require('preconditions').singleton();
|
2014-07-29 13:28:15 -07:00
|
|
|
|
2014-08-05 09:04:38 -07:00
|
|
|
var MESSAGE_PREFIX = 'msg-'; // msg-<recieving_pubkey>-<ts> => <message>
|
2014-07-29 13:28:15 -07:00
|
|
|
|
|
|
|
var MAX_OPEN_FILES = 500;
|
|
|
|
var CONCURRENCY = 5;
|
|
|
|
|
|
|
|
|
2014-07-30 13:29:28 -07:00
|
|
|
var db;
|
|
|
|
var MessageDb = function(opts) {
|
|
|
|
opts = opts || {};
|
2014-07-31 10:06:47 -07:00
|
|
|
this.path = config.leveldb + '/messages' + (opts.name ? ('-' + opts.name) : '');
|
|
|
|
this.db = opts.db || db || levelup(this.path, {
|
|
|
|
maxOpenFiles: MAX_OPEN_FILES,
|
2014-07-31 11:16:05 -07:00
|
|
|
valueEncoding: 'json'
|
2014-07-31 10:06:47 -07:00
|
|
|
});
|
2014-07-30 13:29:28 -07:00
|
|
|
this.initEvents();
|
2014-07-31 11:16:05 -07:00
|
|
|
db = this.db;
|
2014-07-30 13:29:28 -07:00
|
|
|
};
|
|
|
|
util.inherits(MessageDb, EventEmitter);
|
2014-07-29 13:28:15 -07:00
|
|
|
|
2014-07-30 13:29:28 -07:00
|
|
|
MessageDb.prototype.initEvents = function() {
|
2014-07-31 11:16:05 -07:00
|
|
|
if (db) return;
|
2014-08-04 11:43:38 -07:00
|
|
|
var self = this;
|
2014-07-30 13:29:28 -07:00
|
|
|
this.db.on('put', function(key, value) {
|
2014-08-04 11:43:38 -07:00
|
|
|
var data = {};
|
|
|
|
data.key = key;
|
|
|
|
data.value = value;
|
|
|
|
var message = MessageDb.fromStorage(data);
|
|
|
|
self.emit('message', message);
|
2014-07-29 13:28:15 -07:00
|
|
|
});
|
2014-07-30 13:29:28 -07:00
|
|
|
this.db.on('ready', function() {
|
2014-07-31 10:06:47 -07:00
|
|
|
//console.log('Database ready!');
|
2014-07-29 13:28:15 -07:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
MessageDb.prototype.close = function(cb) {
|
2014-07-30 13:29:28 -07:00
|
|
|
this.db.close(cb);
|
2014-07-29 13:28:15 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-08-05 09:04:38 -07:00
|
|
|
var messageKey = function(to, ts) {
|
|
|
|
preconditions.checkArgument(typeof to === 'string');
|
|
|
|
preconditions.checkArgument(to.length === 66);
|
|
|
|
preconditions.checkArgument(!ts || typeof ts === 'number');
|
2014-07-31 11:16:05 -07:00
|
|
|
if (!ts) ts = Math.round(microtime.now());
|
2014-08-05 09:04:38 -07:00
|
|
|
return MESSAGE_PREFIX + to.toString() + '-' + ts;
|
2014-07-29 13:28:15 -07:00
|
|
|
};
|
|
|
|
|
2014-08-04 11:43:38 -07:00
|
|
|
MessageDb.prototype.addMessage = function(m, cb) {
|
|
|
|
|
2014-08-01 15:00:52 -07:00
|
|
|
if (!this.authenticate(m)) {
|
|
|
|
cb(new Error('Authentication failed'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-08-05 09:04:38 -07:00
|
|
|
var key = messageKey(m.to);
|
2014-07-29 13:28:15 -07:00
|
|
|
var value = m;
|
2014-07-30 13:29:28 -07:00
|
|
|
this.db.put(key, value, cb);
|
2014-07-29 13:28:15 -07:00
|
|
|
};
|
|
|
|
|
2014-08-01 15:00:52 -07:00
|
|
|
MessageDb.prototype.authenticate = function(m) {
|
2014-08-04 11:43:38 -07:00
|
|
|
preconditions.checkArgument(m.pubkey);
|
|
|
|
preconditions.checkArgument(m.sig);
|
|
|
|
preconditions.checkArgument(m.encrypted);
|
|
|
|
|
2014-08-01 15:00:52 -07:00
|
|
|
var frompubkey = new Buffer(m.pubkey, 'hex');
|
|
|
|
var sig = new Buffer(m.sig, 'hex');
|
|
|
|
var encrypted = new Buffer(m.encrypted, 'hex');
|
|
|
|
return AuthMessage._verify(frompubkey, sig, encrypted);
|
|
|
|
};
|
|
|
|
|
2014-08-04 11:43:38 -07:00
|
|
|
MessageDb.fromStorage = function(data) {
|
|
|
|
var spl = data.key.split('-');
|
2014-08-05 09:04:38 -07:00
|
|
|
var to = spl[1];
|
|
|
|
var ts = +spl[2];
|
2014-08-04 11:43:38 -07:00
|
|
|
var message = data.value;
|
|
|
|
message.ts = ts;
|
|
|
|
message.to = to;
|
|
|
|
return message;
|
|
|
|
};
|
|
|
|
|
2014-08-05 09:04:38 -07:00
|
|
|
MessageDb.prototype.getMessages = function(to, lower_ts, upper_ts, cb) {
|
2014-07-31 10:06:47 -07:00
|
|
|
var list = [];
|
2014-08-05 11:58:08 -07:00
|
|
|
lower_ts = lower_ts || 1;
|
2014-07-31 10:06:47 -07:00
|
|
|
var opts = {
|
2014-08-05 09:04:38 -07:00
|
|
|
end: messageKey(to, lower_ts),
|
|
|
|
start: messageKey(to, upper_ts),
|
2014-07-31 11:16:05 -07:00
|
|
|
// limit: limit, TODO
|
|
|
|
reverse: true,
|
2014-07-31 10:06:47 -07:00
|
|
|
};
|
2014-07-29 13:28:15 -07:00
|
|
|
|
2014-07-31 10:06:47 -07:00
|
|
|
db.createReadStream(opts)
|
|
|
|
.on('data', function(data) {
|
2014-08-04 11:43:38 -07:00
|
|
|
var message = MessageDb.fromStorage(data);
|
|
|
|
list.push(message);
|
2014-07-31 10:06:47 -07:00
|
|
|
})
|
|
|
|
.on('error', function(err) {
|
|
|
|
return cb(err);
|
|
|
|
})
|
|
|
|
.on('end', function() {
|
|
|
|
return cb(null, list.reverse());
|
|
|
|
});
|
2014-07-29 13:28:15 -07:00
|
|
|
};
|
|
|
|
|
2014-09-01 07:36:25 -07:00
|
|
|
MessageDb.prototype.remove = function(ts) {
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2014-07-31 10:06:47 -07:00
|
|
|
module.exports = soop(MessageDb);
|