2014-01-07 10:21:59 -08:00
|
|
|
require('classtool');
|
|
|
|
|
|
|
|
function spec(b) {
|
|
|
|
var mongoose = require('mongoose');
|
|
|
|
var util = require('util');
|
|
|
|
|
|
|
|
var RpcClient = require('bitcore/RpcClient').class();
|
|
|
|
var networks = require('bitcore/networks');
|
2014-01-07 11:49:42 -08:00
|
|
|
var async = require('async');
|
2014-01-07 10:21:59 -08:00
|
|
|
|
|
|
|
var config = require('./config/config');
|
|
|
|
var Block = require('./app/models/Block');
|
2014-01-07 11:49:42 -08:00
|
|
|
var Transaction=require('./app/models/Transaction');
|
2014-01-07 10:21:59 -08:00
|
|
|
|
|
|
|
function Sync(config) {
|
|
|
|
this.network = config.networkName == 'testnet' ? networks.testnet : networks.livenet;
|
|
|
|
}
|
|
|
|
|
|
|
|
Sync.prototype.getNextBlock = function (blockHash,cb) {
|
|
|
|
var that = this;
|
|
|
|
|
|
|
|
if ( !blockHash ) {
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.rpc.getBlock(blockHash, function(err, blockInfo) {
|
|
|
|
if (err) {
|
|
|
|
return cb(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! ( blockInfo.result.height % 1000) ) {
|
|
|
|
var h = blockInfo.result.height,
|
|
|
|
d = blockInfo.result.confirmations;
|
|
|
|
console.log( util.format("Height: %d/%d [%d%%]", h, d, 100*h/(h+d)));
|
|
|
|
}
|
|
|
|
|
|
|
|
Block.create( blockInfo.result, function(err, inBlock) {
|
|
|
|
|
|
|
|
// E11000 => already exists
|
|
|
|
if (err && ! err.toString().match(/E11000/)) {
|
|
|
|
return cb(err);
|
|
|
|
}
|
|
|
|
|
2014-01-07 11:49:42 -08:00
|
|
|
if (inBlock) {
|
|
|
|
inBlock.explodeTransactions(function (err) {
|
|
|
|
return that.getNextBlock(blockInfo.result.nextblockhash, cb);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return that.getNextBlock(blockInfo.result.nextblockhash, cb);
|
2014-01-07 10:21:59 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
Sync.prototype.syncBlocks = function (reindex, cb) {
|
|
|
|
|
|
|
|
var that = this;
|
|
|
|
var genesisHash = this.network.genesisBlock.hash.reverse().toString('hex');
|
|
|
|
|
|
|
|
if (reindex)
|
|
|
|
return this.getNextBlock(genesisHash, cb);
|
|
|
|
|
|
|
|
|
|
|
|
Block.findOne({}, {}, { sort: { 'confirmations' : 1 } }, function(err, block) {
|
|
|
|
if (err) return cb(err);
|
|
|
|
|
|
|
|
var nextHash =
|
|
|
|
block && block.hash
|
|
|
|
? block.hash
|
|
|
|
: genesisHash
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
|
|
console.log('Starting at hash: ' + nextHash);
|
|
|
|
return that.getNextBlock(nextHash, cb);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-07 11:49:42 -08:00
|
|
|
Sync.prototype.start = function (opts, next) {
|
2014-01-07 10:21:59 -08:00
|
|
|
|
|
|
|
|
|
|
|
mongoose.connect(config.db);
|
|
|
|
var db = mongoose.connection;
|
|
|
|
this.rpc = new RpcClient(config.bitcoind);
|
|
|
|
var that = this;
|
|
|
|
|
|
|
|
|
|
|
|
db.on('error', console.error.bind(console, 'connection error:'));
|
|
|
|
|
2014-01-07 11:49:42 -08:00
|
|
|
db.once('open', function (){
|
|
|
|
|
|
|
|
async.series([
|
|
|
|
function(cb){
|
|
|
|
if (opts.destroy) {
|
|
|
|
console.log("Deleting Blocks...");
|
|
|
|
Block.remove().exec(cb);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function(cb){
|
|
|
|
if (opts.destroy) {
|
|
|
|
console.log("Deleting TXs...");
|
|
|
|
Transaction.remove().exec(cb);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
function(cb) {
|
|
|
|
that.syncBlocks(opts.reindex, function(err) {
|
|
|
|
if (err) {
|
|
|
|
return cb(err);
|
|
|
|
|
|
|
|
}
|
|
|
|
db.close();
|
|
|
|
return cb();
|
|
|
|
});
|
2014-01-07 10:21:59 -08:00
|
|
|
}
|
2014-01-07 11:49:42 -08:00
|
|
|
],
|
|
|
|
function(err) {
|
|
|
|
if (err) return next(er);
|
|
|
|
return next();
|
2014-01-07 10:21:59 -08:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return Sync;
|
|
|
|
};
|
|
|
|
module.defineClass(spec);
|
|
|
|
|