Merge pull request #113 from matiu/bug/fix-reorg-tests

Bug/fix reorg tests
This commit is contained in:
Gustavo Maximiliano Cortez 2014-06-03 11:03:08 -03:00
commit fcbd1f657b
3 changed files with 209 additions and 395 deletions

View File

@ -91,7 +91,7 @@ PeerSync.prototype.handleBlock = function(info) {
}, self.allowReorgs, function(err, height) { }, self.allowReorgs, function(err, height) {
if (err && err.message.match(/NEED_SYNC/) && self.historicSync) { if (err && err.message.match(/NEED_SYNC/) && self.historicSync) {
console.log('[p2p_sync] Orphan block received. Triggering sync'); console.log('[p2p_sync] Orphan block received. Triggering sync');
self.historicSync.start({}, function(){ self.historicSync.start({forceRPC:1}, function(){
console.log('[p2p_sync] Done resync.'); console.log('[p2p_sync] Done resync.');
}); });
} }

View File

@ -83,8 +83,18 @@ Sync.prototype.storeTipBlock = function(b, allowReorgs, cb) {
allowReorgs = true; allowReorgs = true;
} }
if (!b) return cb(); if (!b) return cb();
var self = this; var self = this;
if ( self.storingBlock ) {
console.log('## Storing a block already. Delaying storeTipBlock with:' +
b.hash);
return setTimeout( function() {
console.log('## Retrying storeTipBlock with: ' + b.hash);
self.storeTipBlock(b,allowReorgs,cb);
}, 1000);
}
self.storingBlock=1;
var oldTip, oldNext, oldHeight, needReorg = false, height = -1; var oldTip, oldNext, oldHeight, needReorg = false, height = -1;
var newPrev = b.previousblockhash; var newPrev = b.previousblockhash;
@ -164,6 +174,7 @@ Sync.prototype.storeTipBlock = function(b, allowReorgs, cb) {
if (err && err.toString().match(/WARN/)) { if (err && err.toString().match(/WARN/)) {
err = null; err = null;
} }
self.storingBlock=0;
return cb(err, height); return cb(err, height);
}); });
}; };
@ -177,7 +188,10 @@ Sync.prototype.processReorg = function(oldTip, oldNext, newPrev, oldHeight, cb)
function(c) { function(c) {
self.bDb.getHeight(newPrev, function(err, height) { self.bDb.getHeight(newPrev, function(err, height) {
if (!height) return c(new Error('Could not found block:' + newPrev)); if (!height) {
// Case 3 + allowReorgs = true
return c(new Error('Could not found block:' + newPrev));
}
if (height<0) return c(); if (height<0) return c();
newHeight = height + 1; newHeight = height + 1;

View File

@ -157,7 +157,7 @@ describe('Sync Reorgs', function(){
}; };
describe('reorg, case 1', function(done) { describe('reorg, case 1', function() {
checkTxs([t[0], t[1], t[2], t[3], t[4]],[16,17,18,19,20]); checkTxs([t[0], t[1], t[2], t[3], t[4]],[16,17,18,19,20]);
checkBlocks([b[0], b[1], b[2], b[3], b[4]],[16,17,18,19,20]); checkBlocks([b[0], b[1], b[2], b[3], b[4]],[16,17,18,19,20]);
it('trigger reorg case 1', function(done1){ it('trigger reorg case 1', function(done1){
@ -168,399 +168,199 @@ describe('Sync Reorgs', function(){
}); });
checkTxs([t[0], t[1], t[2], t[3], t[4],case1.tx],[16,17,18,-1,-1,19]); checkTxs([t[0], t[1], t[2], t[3], t[4],case1.tx],[16,17,18,-1,-1,19]);
checkBlocks([b[0], b[1], b[2],b[3],b[4],case1.hash],[16,17,18,-1,-1,19]); checkBlocks([b[0], b[1], b[2],b[3],b[4],case1.hash],[16,17,18,-1,-1,19]);
it('reorg, case 1 (repeat)', function(done) {
s.sync.storeTipBlock(case1, function(err) {
assert(!err, 'shouldnt return error' + err);
return done();
});
});
}); });
var case2 = {
hash: '00000000c262f9428bb84407780bb0bd008b74941d651111ab2500cf649fa45d',
tx: [ '3fa6fce216e91c9dc9a6267168e9d8bfb4ae57aec0d26590442cfec6e8233682' ],
time: 1296690099,
previousblockhash: b[3],
};
// * case 2)
// * 0-1-2---3-4
// * \ \
// * C1 C2*
describe('reorg, case 2', function() {
checkTxs([t[0], t[1], t[2], t[3], t[4],case1.tx[0]],[16,17,18,-1,-1,19]);
checkBlocks([b[0], b[1], b[2],b[3],b[4],case1.hash],[16,17,18,-1,-1,19]);
it('trigger reorg case 2', function(done1){
s.sync.storeTipBlock(case2, function(err) {
assert(!err, 'shouldnt return error' + err);
return done1();
});
});
checkBlocks([b[0], b[1], b[2],b[3],b[4],case2.hash],[16,17,18,19,-1,20]);
checkTxs([t[0], t[1], t[2],t[3],t[4],case2.tx[0]],[16,17,18,19,-1,20]);
it('next from block 2', function(done1){
s.sync.bDb.getNext(b[2], function(err, val) {
assert(!err);
assert.equal(val,b[3]);
return done1();
});
});
});
// * case 2b)
// * 0-1-2---3-4
// * \ \
// * C1 C2-C2b(TX=C1.TX)*
var case2b = {
hash: '0000000022bb34bc09f8d8d0ae26b86c87f8483e54b9c3addfe6f30b12bc656a',
tx: case1.tx,
time: 1296690099,
previousblockhash: case2.hash,
};
describe('reorg, case 2', function() {
it('reorg case 2b', function(done1){
s.sync.storeTipBlock(case2b, function(err) {
assert(!err, 'shouldnt return error' + err);
return done1();
});
});
checkBlocks([b[0], b[1], b[2],b[3],b[4],case2.hash, case2b.hash, case1.hash],[16,17,18,19,-1,20, 21, -1]);
checkTxs([t[0], t[1], t[2],t[3],t[4],case2.tx[0], case1.tx],[16,17,18,19,-1,20, 21]);
});
var case2c = {
hash: '0000000000000000000000000000000000000000000000000000000000000004',
tx: case2.tx,
time: 1296690099,
previousblockhash: case1.hash,
};
// * case 2c)
// * 0-1-2---3-4
// * \ \
// * C1 C2-C2b(TX=C1.TX)
// * \
// * C2c(TX=C2.TX)*
describe('reorg, case 2c', function() {
it('trigger reorg case 2c', function(done1){
s.sync.storeTipBlock(case2c, function(err) {
assert(!err, 'shouldnt return error' + err);
return done1();
});
});
checkBlocks([b[0], b[1], b[2],
b[3],b[4],case2.hash, case2b.hash,
case1.hash, case2c.hash],
[16,17,18,
-1,-1,-1, -1,
19, 20
]);
checkTxs(
[t[0], t[1], t[2],
t[3],t[4], case2.tx[0],
case1.tx, case2c.tx[0]],
[16,17,18,
-1,-1, 20,
19, 20]);
});
// * case 3)
// * 0-1-2---3-4
// * \ \
// * C1 C2-C2b(TX=C1.TX)
// * \
// * C2c(TX=C2.TX)
// //
// it('reorg, case 1 (repeat)', function(done) { // (orphan)-C3*
// s.sync.storeTipBlock(case1, function(err) { // -> returns error
// assert(!err, 'shouldnt return error' + err);
// return done(); var case3 = {
// }); hash: '0000000000000000000000000000000000000000000000000000000000000005',
// }); tx: case2.tx,
// time: 1296690099,
// var case2 = { previousblockhash: '666',
// hash: '00000000c262f9428bb84407780bb0bd008b74941d651111ab2500cf649fa45d', };
// tx: [ '3fa6fce216e91c9dc9a6267168e9d8bfb4ae57aec0d26590442cfec6e8233682' ],
// time: 1296690099, describe('reorg, case 3)', function() {
// previousblockhash: b[3], it('case 3). Should return an error', function(done1){
// }; s.sync.storeTipBlock(case3, function(err) {
// assert(err, 'should return error' + err);
// return done1();
// it('reorg, case 2', function(done) { });
// async.series([ });
// function (c) { //shoudnt change anything
// s.sync.txDb.getBlock(t[0], function(err,is) { checkBlocks([b[0], b[1], b[2],
// assert(!err); b[3],b[4],case2.hash, case2b.hash,
// assert(is); case1.hash, case2c.hash],
// return c(); [16,17,18,
// }); -1,-1,-1, -1,
// }, 19, 20
// function (c) { ]);
// s.sync.txDb.getBlock(case1.tx[0], function(err,is) { checkTxs(
// assert(!err); [t[0], t[1], t[2],
// assert(is); t[3],t[4], case2.tx[0],
// return c(); case1.tx, case2c.tx[0]],
// }); [16,17,18,
// }, -1,-1, 20,
// function (c) { 19, 20]);
// s.sync.txDb.getBlock(t[4], function(err,is) { });
// assert(!err);
// assert(!is); var p2p = {
// return c(); hash: '0000000000000000000000000000000000000000000000000000000000000006',
// }); tx: ['f6c2901f39fd07f2f2e503183d76f73ecc1aee9ac9216fde58e867bc29ce674e'],
// }, time: 1296690099,
// // BEFORE previousblockhash: '111',
// function (c) { };
// s.sync.bDb.getHeight(b[3], function(err,is) {
// assert(!err); describe('p2p sync. no reorgs', function() {
// assert.equal(is,-1); it('Should return an error', function(done1){
// return c(); s.sync.storeTipBlock(p2p, false, function(err) {
// }); assert(!err, 'shouldnt return error' + err);
// }, return done1();
// function (c) { });
// s.sync.storeTipBlock(case2, function(err) { it('Block should be stored', function(done1){
// assert(!err, 'shouldnt return error' + err); s.sync.bDb.has(p2p.hash, function(err,is) {
// return c(); assert(!err);
// }); assert(is);
// }, return done1();
// // AFTER });
// function (c) { });
// s.sync.bDb.getHeight(b[3], function(err,is) { //shoudnt change anything
// assert(!err); checkBlocks([b[0], b[1], b[2],
// assert.equal(is,19); b[3],b[4],case2.hash, case2b.hash,
// return c(); case1.hash, case2c.hash,
// }); p2p.hash],
// }, [16,17,18,
// function (c) { -1,-1,-1, -1,
// s.sync.bDb.getHeight(b[4], function(err,is) { 19, 20,
// assert(!err); -1
// assert.equal( is,-1, b[3] + 'should not be on main chain: #'+is); ]);
// return c(); });
// }); it('next Block should be stored', function(done1){
// }, s.sync.bDb.getNext(p2p.hash, function(err,v) {
// function (c) { assert(!err);
// s.sync.bDb.getHeight(case1.hash, function(err,is) { assert.equal(v,p2p.nextblockhash);
// assert(!err); return done1();
// assert.equal(is,-1); });
// return c(); });
// }); it('next Block should be stored', function(done1){
// }, s.sync.bDb.getNext(p2p.previousblockhash, function(err,v) {
// function (c) { assert(!err);
// s.sync.bDb.getHeight(case2.hash, function(err,is) { assert.equal(v,p2p.hash);
// assert(!err); return done1();
// assert.equal(is, 20); });
// return c(); });
// }); });
// },
// function (c) {
// s.sync.txDb.getBlock(t[3], function(err,is) {
// console.log('[99-sync.js.descructive-test.300:is:]',is); //TODO
// assert(!err);
// assert(is, 'transaction t[3] should be valid:' + t[3]);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case1.tx[0], function(err,is) {
// assert(!err);
// assert(!is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[4], function(err,is) {
// assert(!err);
// assert(!is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getNext(b[2], function(err, val) {
// assert(!err);
// assert.equal(val,b[3]);
// return c();
// });
// },
//
//
//
// ], done );
// });
//
//
// var case2b = {
// hash: '0000000022bb34bc09f8d8d0ae26b86c87f8483e54b9c3addfe6f30b12bc656a',
// tx: case1.tx,
// time: 1296690099,
// previousblockhash: case2.hash,
// };
//
// it('reorg, case 2b', function(done) {
// async.series([
// function (c) {
// s.sync.txDb.getBlock(case2b.tx[0], function(err,is) {
// assert(!err);
// assert(!is);
// return c();
// });
// },
// function (c) {
// s.sync.storeTipBlock(case2b, function(err) {
// assert(!err, 'shouldnt return error' + err);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[3], function(err,is) {
// assert(!err);
// assert(is, 'transaction t[3] should be valid:' + t[3]);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2b.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// ], done );
// });
//
//
//
// var case2c = {
// hash: '0000000000000000000000000000000000000000000000000000000000000004',
// tx: case2.tx,
// time: 1296690099,
// previousblockhash: case1.hash,
// };
//
// it('reorg, case 2c', function(done) {
// async.series([
// function (c) {
// s.sync.txDb.getBlock(case1.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getHeight(case1.hash, function(err,is) {
// assert(!err);
// assert.equal(is, -1, 'case1 block shouldnt be main:' + case1.hash);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2c.tx[0], function(err,is) {
// assert(!err);
// assert(is); //It was there before (from case2)
// return c();
// });
// },
// function (c) {
// s.sync.storeTipBlock(case2c, function(err) {
// assert(!err, 'shouldnt return error' + err);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case1.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.has(case1.hash, function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.has(case2c.hash, function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getHeight(case1.hash, function(err,h) {
// assert(!err);
// assert.equal(h,19);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getHeight(case2c.hash, function(err,h) {
// assert(!err);
// assert.equal(h,20);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2c.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[3], function(err,is) {
// assert(!err);
// assert(!is, 'TX t[3]: shouldnt be confirmed:' + t[3] +':'+ is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[4], function(err,is) {
// assert(!err);
// assert(!is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
//
// ], done );
// });
//
// var case3 = {
// hash: '0000000000000000000000000000000000000000000000000000000000000005',
// tx: case2.tx,
// time: 1296690099,
// previousblockhash: '666',
// };
//
// it('reorg, case 3)', function(done) {
// async.series([
// function (c) {
// s.sync.storeTipBlock(case3, function(err) {
// assert(err, 'should return error' + err);
// return c();
// });
// },
//
// //shoudnt change anything
// function (c) {
// s.sync.txDb.getBlock(case1.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.has(case1.hash, function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.has(case2c.hash, function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2c.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[3], function(err,is) {
// assert(!err);
// assert(!is, 'TX t[3]: shouldnt be confirmed:' + t[3] +':'+ is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(t[4], function(err,is) {
// assert(!err);
// assert(!is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(case2.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
//
// ], done );
// });
//
// var p2p = {
// hash: '0000000000000000000000000000000000000000000000000000000000000006',
// tx: ['f6c2901f39fd07f2f2e503183d76f73ecc1aee9ac9216fde58e867bc29ce674e'],
// time: 1296690099,
// previousblockhash: '111',
// };
//
// it('p2p, no reorg allowed', function(done) {
// async.series([
// function (c) {
// s.sync.storeTipBlock(p2p, false, function(err) {
// assert(!err, 'shouldnt return error' + err);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.has(p2p.hash, function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.txDb.getBlock(p2p.tx[0], function(err,is) {
// assert(!err);
// assert(is);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getNext(p2p.hash, function(err,v) {
// assert(!err);
// assert.equal(v,p2p.nextblockhash);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getNext(p2p.previousblockhash, function(err,v) {
// assert(!err);
// assert.equal(v,p2p.hash);
// return c();
// });
// },
// function (c) {
// s.sync.bDb.getHeight(p2p.hash, function(err,is) {
// assert(!err);
// assert.equal(is,-1);
// return c();
// });
// },
//
// ], done );
// });
}); });