Merge pull request #5 from eordano/sync

Sync
This commit is contained in:
Manuel Aráoz 2015-04-07 16:24:17 -03:00
commit 0bd69fa74f
4 changed files with 57 additions and 27 deletions

View File

@ -5,12 +5,16 @@ var $ = bitcore.util.preconditions;
var _ = bitcore.deps._; var _ = bitcore.deps._;
function BlockChain() { function BlockChain() {
this.tip = null; this.tip = '0000000000000000000000000000000000000000000000000000000000000000';
this.work = { this.work = {
'0000000000000000000000000000000000000000000000000000000000000000': 0 '0000000000000000000000000000000000000000000000000000000000000000': 0
}; };
this.height = {}; this.height = {
this.hashByHeight = {}; '0000000000000000000000000000000000000000000000000000000000000000': -1
};
this.hashByHeight = {
'-1': '0000000000000000000000000000000000000000000000000000000000000000'
};
this.next = {}; this.next = {};
this.prev = {}; this.prev = {};
} }
@ -44,7 +48,7 @@ BlockChain.prototype.proposeNewBlock = function(block) {
$.checkArgument(block instanceof bitcore.Block, 'Argument is not a Block instance'); $.checkArgument(block instanceof bitcore.Block, 'Argument is not a Block instance');
var prevHash = bitcore.util.buffer.reverse(block.header.prevHash).toString('hex'); var prevHash = bitcore.util.buffer.reverse(block.header.prevHash).toString('hex');
if (!this.work[prevHash]) { if (_.isUndefined(this.work[prevHash])) {
throw new Error('No previous data to estimate work'); throw new Error('No previous data to estimate work');
} }
this.addData(block); this.addData(block);
@ -118,7 +122,9 @@ BlockChain.prototype.getBlockLocator = function() {
var currentHeight = this.height[this.tip]; var currentHeight = this.height[this.tip];
var exponentialBackOff = 1; var exponentialBackOff = 1;
for (var i = 0; i < 10; i++) { for (var i = 0; i < 10; i++) {
result.push(this.hashByHeight[currentHeight--]); if (currentHeight >= 0) {
result.push(this.hashByHeight[currentHeight--]);
}
} }
while (currentHeight > 0) { while (currentHeight > 0) {
result.push(this.hashByHeight[currentHeight]); result.push(this.hashByHeight[currentHeight]);

View File

@ -65,8 +65,9 @@ NetworkMonitor.prototype.setupPeer = function(peer) {
}; };
NetworkMonitor.prototype.requestBlocks = function(start) { NetworkMonitor.prototype.requestBlocks = function(start) {
$.checkArgument(_.isString(start), 'start must be a block hash string'); $.checkArgument(_.isArray(start) ||
this.peer.sendMessage(messages.GetBlocks([start])); _.isString(start), 'start must be a block hash string or array');
this.peer.sendMessage(messages.GetBlocks(_.isArray(start) ? start : [start]));
}; };
NetworkMonitor.prototype.start = function() { NetworkMonitor.prototype.start = function() {

View File

@ -35,7 +35,29 @@ var BitcoreNode = function(bus, networkMonitor, blockService, transactionService
this.transactionService = transactionService; this.transactionService = transactionService;
this.blockService = blockService; this.blockService = blockService;
this.bus.register(bitcore.Block, this.blockService.onBlock.bind(this.blockService)); this.blockCache = {};
this.bus.register(bitcore.Block, function(block) {
var prevHash = bitcore.util.buffer.reverse(block.header.prevHash).toString('hex');
self.blockCache[block.hash] = block;
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(Promise.each(blockchainChanges.confirmed, function(hash) {
return self.blockService.confirm(self.blockCache[hash]);
}))
.then(function() {
self.networkMonitor.requestBlocks(self.blockchain.getBlockLocator());
});
});
this.bus.onAny(function(value) { this.bus.onAny(function(value) {
self.emit(this.event, value); self.emit(this.event, value);
@ -82,8 +104,19 @@ BitcoreNode.create = function(opts) {
}; };
BitcoreNode.prototype.start = function() { BitcoreNode.prototype.start = function() {
this.sync(); var self = this;
this.networkMonitor.start(); var genesis = bitcore.Block.fromBuffer(genesisBlocks[bitcore.Networks.defaultNetwork.name]);
this.blockService.getBlockchain().then(function(blockchain) {
if (!blockchain) {
self.blockchain = new BlockChain();
self.blockchain.proposeNewBlock(genesis);
}
self.sync();
self.networkMonitor.start();
});
this.networkMonitor.on('stop', function() {
self.blockService.saveBlockchain(self.blockchain);
});
}; };
BitcoreNode.prototype.stop = function() { BitcoreNode.prototype.stop = function() {
@ -92,24 +125,15 @@ BitcoreNode.prototype.stop = function() {
BitcoreNode.prototype.sync = function() { BitcoreNode.prototype.sync = function() {
var genesis = bitcore.Block.fromBuffer(genesisBlocks[bitcore.Networks.defaultNetwork.name]);
var self = this; var self = this;
this.networkMonitor.on('ready', function() { this.networkMonitor.on('ready', function() {
self.blockService.getBlockchain().then(function(blockchain) { self.blockService.getBlockchain().then(function(blockchain) {
if (blockchain) { self.networkMonitor.requestBlocks(self.blockchain.getBlockLocator());
self.blockchain = blockchain; }).catch(function(err) {
} else { self.networkMonitor.stop();
self.blockchain = new BlockChain(); throw err;
self.blockchain.setTip(genesis); });
}
self.networkMonitor.requestBlocks(self.blockchain.tip);
})
.catch(function(err) {
self.networkMonitor.stop();
throw err;
});
}); });
}; };
module.exports = BitcoreNode; module.exports = BitcoreNode;

View File

@ -226,9 +226,8 @@ BlockService.prototype.onBlock = function(block) {
* @param {bitcore.Block} block * @param {bitcore.Block} block
* @return {Promise<Block>} a promise of the same block, for chaining * @return {Promise<Block>} a promise of the same block, for chaining
*/ */
BlockService.prototype.save = function(block) { BlockService.prototype.unconfirm = function(block) {
// TODO: unconfirm previous tip, confirm new tip. // TODO: unconfirm previous tip, confirm new tip.
return this._confirmBlock(block);
}; };
@ -238,7 +237,7 @@ BlockService.prototype.save = function(block) {
* @param {bitcore.Block} block * @param {bitcore.Block} block
* @return {Promise<Block>} a promise of the same block, for chaining * @return {Promise<Block>} a promise of the same block, for chaining
*/ */
BlockService.prototype._confirmBlock = function(block) { BlockService.prototype.confirm = function(block) {
$.checkArgument(block instanceof bitcore.Block); $.checkArgument(block instanceof bitcore.Block);
var self = this; var self = this;