flocore-node/lib/node.js
2015-04-07 17:29:17 -03:00

165 lines
4.8 KiB
JavaScript

'use strict';
var util = require('util');
var EventEmitter = require('eventemitter2').EventEmitter2;
var bitcore = require('bitcore');
var _ = bitcore.deps._;
var config = require('config');
var p2p = require('bitcore-p2p');
var messages = new p2p.Messages();
var $ = bitcore.util.preconditions;
var Promise = require('bluebird');
var RPC = require('bitcoind-rpc');
var NetworkMonitor = require('./networkmonitor');
var EventBus = require('./eventbus');
var LevelUp = require('levelup');
var BlockService = require('./services/block');
var TransactionService = require('./services/transaction');
var AddressService = require('./services/address');
var BlockChain = require('./blockchain');
var genesisBlocks = require('./data/genesis');
var BitcoreNode = function(bus, networkMonitor, blockService, transactionService, addressService) {
$.checkArgument(bus);
$.checkArgument(networkMonitor);
var self = this;
this.bus = bus;
this.networkMonitor = networkMonitor;
this.tip = null;
this.addressService = addressService;
this.transactionService = transactionService;
this.blockService = blockService;
this.blockCache = {};
this.inventory = {}; // blockHash -> bool (has data)
this.networkMonitor.on('inv', function(inventory) {
_.each(inventory, function(info) {
var hash = bitcore.util.buffer.reverse(info.hash).toString('hex');
$.checkState(_.isUndefined(self.inventory[hash]));
if (info.type === 2) { // TODO: use static field from bitcore-p2p
self.inventory[hash] = false;
}
});
});
this.bus.register(bitcore.Block, function(block) {
console.log('Block', block.id);
var prevHash = bitcore.util.buffer.reverse(block.header.prevHash).toString('hex');
self.blockCache[block.hash] = block;
self.inventory[block.hash] = true;
console.log('prevHash', prevHash);
console.log('height', self.blockchain.height[self.blockchain.tip]);
if (!self.blockchain.hasData(prevHash)) {
self.networkMonitor.requestBlocks(self.blockchain.getBlockLocator());
return;
}
var blockchainChanges = self.blockchain.proposeNewBlock(block);
Promise.each(blockchainChanges.unconfirmed, function(hash) {
return self.blockService.unconfirm(self.blockCache[hash]);
})
.then(function() {
return Promise.all(blockchainChanges.confirmed.map(function(hash) {
return self.blockService.confirm(self.blockCache[hash]);
}));
})
.then(function() {
if (_.size(self.inventory) && _.all(_.values(self.inventory))) {
self.inventory = {};
console.log('requesting ...', self.blockchain.getBlockLocator().length);
self.networkMonitor.requestBlocks(self.blockchain.getBlockLocator());
}
})
.catch(function(error) {
self.stop(error);
});
});
this.bus.onAny(function(value) {
self.emit(this.event, value);
});
this.networkMonitor.on('error', function(err) {
self.emit('error', err);
});
this.networkMonitor.on('disconnect', function() {
console.log('network monitor disconnected');
});
};
util.inherits(BitcoreNode, EventEmitter);
BitcoreNode.create = function(opts) {
opts = opts || {};
var bus = new EventBus();
var networkMonitor = NetworkMonitor.create(bus, opts.NetworkMonitor);
var database = Promise.promisifyAll(
new LevelUp(opts.LevelUp || config.get('LevelUp'))
);
var rpc = Promise.promisifyAll(new RPC(config.get('RPC')));
var transactionService = new TransactionService({
rpc: rpc,
database: database
});
var blockService = new BlockService({
rpc: rpc,
database: database,
transactionService: transactionService
});
var addressService = new AddressService({
rpc: rpc,
database: database,
transactionService: transactionService,
blockService: blockService
});
return new BitcoreNode(bus, networkMonitor, blockService, transactionService, addressService);
};
BitcoreNode.prototype.start = function() {
var self = this;
var genesis = bitcore.Block.fromBuffer(genesisBlocks[bitcore.Networks.defaultNetwork.name]);
this.blockService.getBlockchain().then(function(blockchain) {
if (!blockchain) {
self.blockchain = new BlockChain();
self.bus.process(genesis);
}
self.sync();
self.networkMonitor.start();
});
this.networkMonitor.on('stop', function() {
self.blockService.saveBlockchain(self.blockchain);
});
};
BitcoreNode.prototype.stop = function(reason) {
this.networkMonitor.stop(reason);
};
BitcoreNode.prototype.sync = function() {
var self = this;
this.networkMonitor.on('ready', function() {
self.blockService.getBlockchain().then(function(blockchain) {
self.networkMonitor.requestBlocks(self.blockchain.getBlockLocator());
}).catch(function(err) {
self.networkMonitor.stop();
throw err;
});
});
};
module.exports = BitcoreNode;