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;
|
|
|
|
var sockets = require('../app/controllers/socket.js');
|
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-07-29 13:28:15 -07:00
|
|
|
|
|
|
|
var MESSAGE_PREFIX = 'msg-'; // msg-<sin1>-<sin2> => <message>
|
|
|
|
|
|
|
|
var MAX_OPEN_FILES = 500;
|
|
|
|
var CONCURRENCY = 5;
|
|
|
|
|
|
|
|
|
|
|
|
var d = logger.log;
|
|
|
|
var info = logger.info;
|
2014-07-30 13:29:28 -07:00
|
|
|
var db;
|
2014-07-29 13:28:15 -07:00
|
|
|
|
2014-07-30 13:29:28 -07:00
|
|
|
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-07-30 13:29:28 -07:00
|
|
|
this.db.on('put', function(key, value) {
|
|
|
|
var spl = key.split('-');
|
2014-07-31 10:06:47 -07:00
|
|
|
var from = spl[1];
|
|
|
|
var to = spl[2];
|
|
|
|
var ts = spl[3];
|
2014-07-30 13:29:28 -07:00
|
|
|
var message = value;
|
|
|
|
sockets.broadcastMessage(from, to, ts, 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
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
var messageKey = function(from, to, ts) {
|
2014-07-31 11:16:05 -07:00
|
|
|
if (!ts) ts = Math.round(microtime.now());
|
2014-07-29 13:28:15 -07:00
|
|
|
return MESSAGE_PREFIX + from.toString() + '-' + to.toString() + '-' + ts;
|
|
|
|
};
|
|
|
|
|
|
|
|
MessageDb.prototype.addMessage = function(m, from, to, cb) {
|
2014-08-01 15:00:52 -07:00
|
|
|
|
|
|
|
if (!this.authenticate(m)) {
|
|
|
|
cb(new Error('Authentication failed'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-29 13:28:15 -07:00
|
|
|
var key = messageKey(from, to);
|
|
|
|
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) {
|
|
|
|
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-07-31 10:06:47 -07:00
|
|
|
MessageDb.prototype.getMessages = function(from, to, lower_ts, upper_ts, cb) {
|
|
|
|
var list = [];
|
|
|
|
var opts = {
|
2014-07-31 11:16:05 -07:00
|
|
|
end: messageKey(from, to, lower_ts),
|
|
|
|
start: messageKey(from, to, upper_ts),
|
|
|
|
// 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) {
|
|
|
|
var spl = data.key.split('-');
|
|
|
|
var from = spl[1];
|
|
|
|
var to = spl[2];
|
2014-07-31 11:16:05 -07:00
|
|
|
var ts = +spl[3];
|
2014-07-31 10:06:47 -07:00
|
|
|
list.push({
|
|
|
|
ts: ts,
|
|
|
|
message: data.value,
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.on('error', function(err) {
|
|
|
|
return cb(err);
|
|
|
|
})
|
|
|
|
.on('end', function() {
|
|
|
|
return cb(null, list.reverse());
|
|
|
|
});
|
2014-07-29 13:28:15 -07:00
|
|
|
};
|
|
|
|
|
2014-07-31 10:06:47 -07:00
|
|
|
module.exports = soop(MessageDb);
|