Browse Source

Merge pull request #772 from braydonf/docs/block

Block: Updated documentation, and changed to use consistent naming.
patch-2
Manuel Aráoz 10 years ago
parent
commit
9f51b1b4ce
  1. 57
      docs/Block.md
  2. 73
      lib/block.js
  3. 23
      lib/blockheader.js
  4. 30
      test/block.js
  5. 32
      test/blockheader.js
  6. 10
      test/data/blk86756-testnet.json

57
docs/Block.md

@ -1,30 +1,55 @@
# Block # Block
A Block instance represents the information on a block in the bitcoin network. Instantiating it is not a cheap operation, as the tree of transactions needs to be created or verified. There's a (BlockHeader)[https://github.com/bitpay/bitcore/tree/master/lib/blockheader.js] interface that is easier on the javascript VM. A Block instance represents the information on a block in the bitcoin network. Given a hexa or base64 string representation of the serialization of a block with its transactions, you can instantiate a Block instance. Methods are provided to calculate and check the merkle root hash (if enough data is provided), but transactions won't necessarily be valid spends, and this class won't validate them. A binary representation as a `Buffer` instance is also valid input for a Block's constructor.
Given a hexa or base64 string representation of the serialization of a block with its transactions, you can instantiate a Block instance. It will calculate and check the merkle root hash (if enough data is provided), but transactions won't necessarily be valid spends, and this class won't validate them. A binary representation as a `Buffer` instance is also valid input for a Block's constructor.
```javascript ```javascript
assert(Block.isValidHeader(data);
assert(Block.isValidBlock(data);
// instantiate a new block instance
var block = new Block(hexaEncodedBlock); var block = new Block(hexaEncodedBlock);
assert(block.id && block.hash && block.id === block.hash);
assert(block.version === Block.CurrentVersion); // will verify that the correspending block transactions match the header
assert(block.prevHash); assert(block.validMerkleRoot());
assert(block.timestamp);
assert(block.nonce); // blocks have several properties
assert(block.magicnum);
assert(block.size); assert(block.size);
assert(block.transactions[0] instanceof Transaction); assert(block.header); // an instance of block header, more info below
assert(block.txs); // an array of transactions, more info below
``` ```
## Navigating through transactions For detailed technical information about a block please visit [Blocks](https://en.bitcoin.it/wiki/Blocks#Block_structure) on the Bitcoin Wiki.
The set of transactions in a block can be explored by iterating on the block's ## Block Header
`transactions` member.
Each instance of Block has a BlockHeader *(which can be instantiated seperately)*. The header has validation methods, to verify that the block.
```javascript ```javascript
for (var transaction in block.transactions) {
// ... // will verify that the nonce demonstrates enough proof of work
assert(block.header.validProofOfWork());
// will verify that timestamp is not too far in the future
assert(block.header.validTimestamp());
// each header has the following properties
assert(block.header.version);
assert(block.header.prevHash);
assert(block.header.merkleRoot);
assert(block.header.time);
assert(block.header.bits);
assert(block.header.nonce);
```
For more information about the specific properties of a block header please visit the [Block hashing algorithm](https://en.bitcoin.it/wiki/Block_hashing_algorithm) page on the Bitcoin Wiki.
## Transactions
The set of transactions in a block is an array of instances of [Transaction](Transaction.md) and can be explored by iterating on the block's `transactions` member.
```javascript
for (var i in block.txs) {
var transaction = block.txs[i];
} }
``` ```

73
lib/block.js

@ -25,7 +25,6 @@ function Block(arg) {
return new Block(arg); return new Block(arg);
} }
_.extend(this, Block._from(arg)); _.extend(this, Block._from(arg));
this._setupProperties();
return this; return this;
} }
@ -52,15 +51,15 @@ Block._from = function _from(arg) {
*/ */
magicnum: arg.magicnum, magicnum: arg.magicnum,
/** /**
* @name Block#blocksize * @name Block#size
* @type number * @type number
*/ */
blocksize: arg.blocksize, size: arg.size,
/** /**
* @name Block#blockheader * @name Block#header
* @type {BlockHeader} * @type {BlockHeader}
*/ */
blockheader: arg.blockheader, header: arg.header,
txsvi: arg.txsvi, txsvi: arg.txsvi,
/** /**
* @name Block#txs * @name Block#txs
@ -74,46 +73,6 @@ Block._from = function _from(arg) {
return info; return info;
}; };
/**
* Internal function that sets up a some properties from blockheader for
* easier access
*/
Block.prototype._setupProperties = function() {
/**
* @name Block#version
* @type {number}
*/
Object.defineProperty(this, 'version', {
configurable: false,
value: this.blockheader.version
});
/**
* @name Block#timestamp
* @type {number}
*/
Object.defineProperty(this, 'timestamp', {
configurable: false,
value: this.blockheader.timestamp
});
/**
* @name Block#nonce
* @type {number}
*/
Object.defineProperty(this, 'nonce', {
configurable: false,
value: this.blockheader.nonce
});
/**
* Amount of bytes of the serialized block
* @name Block#size
* @type {number}
*/
Object.defineProperty(this, 'size', {
configurable: false,
value: this.blockheader.size
});
};
/** /**
* @param {String|Object} - A JSON string or object * @param {String|Object} - A JSON string or object
* @returns {Object} - An object representing block data * @returns {Object} - An object representing block data
@ -129,8 +88,8 @@ Block._fromJSON = function _fromJSON(data) {
}); });
var info = { var info = {
magicnum: data.magicnum, magicnum: data.magicnum,
blocksize: data.blocksize, size: data.size,
blockheader: BlockHeader.fromJSON(data.blockheader), header: BlockHeader.fromJSON(data.header),
txsvi: Varint().fromString(data.txsvi), txsvi: Varint().fromString(data.txsvi),
txs: txs txs: txs
}; };
@ -154,8 +113,8 @@ Block.fromJSON = function fromJSON(json) {
Block._fromBufferReader = function _fromBufferReader(br) { Block._fromBufferReader = function _fromBufferReader(br) {
var info = {}; var info = {};
info.magicnum = br.readUInt32LE(); info.magicnum = br.readUInt32LE();
info.blocksize = br.readUInt32LE(); info.size = br.readUInt32LE();
info.blockheader = BlockHeader.fromBufferReader(br); info.header = BlockHeader.fromBufferReader(br);
info.txsvi = Varint(br.readVarintBuf()); info.txsvi = Varint(br.readVarintBuf());
var txslen = info.txsvi.toNumber(); var txslen = info.txsvi.toNumber();
info.txs = []; info.txs = [];
@ -214,8 +173,8 @@ Block.prototype.toObject = function toObject() {
}); });
return { return {
magicnum: this.magicnum, magicnum: this.magicnum,
blocksize: this.blocksize, size: this.size,
blockheader: this.blockheader.toObject(), header: this.header.toObject(),
txsvi: this.txsvi.toString(), txsvi: this.txsvi.toString(),
txs: txs txs: txs
}; };
@ -251,8 +210,8 @@ Block.prototype.toBufferWriter = function toBufferWriter(bw) {
bw = new BufferWriter(); bw = new BufferWriter();
} }
bw.writeUInt32LE(this.magicnum); bw.writeUInt32LE(this.magicnum);
bw.writeUInt32LE(this.blocksize); bw.writeUInt32LE(this.size);
bw.write(this.blockheader.toBuffer()); bw.write(this.header.toBuffer());
bw.write(this.txsvi.buf); bw.write(this.txsvi.buf);
var txslen = this.txsvi.toNumber(); var txslen = this.txsvi.toNumber();
for (var i = 0; i < txslen; i++) { for (var i = 0; i < txslen; i++) {
@ -309,12 +268,12 @@ Block.prototype.getMerkleRoot = function getMerkleRoot() {
}; };
/** /**
* Verifies that the transactions in the block match the blockheader merkle root * Verifies that the transactions in the block match the header merkle root
* @returns {Boolean} - If the merkle roots match * @returns {Boolean} - If the merkle roots match
*/ */
Block.prototype.validMerkleRoot = function validMerkleRoot() { Block.prototype.validMerkleRoot = function validMerkleRoot() {
var h = new BN(this.blockheader.merklerootbuf.toString('hex'), 'hex'); var h = new BN(this.header.merkleRoot.toString('hex'), 'hex');
var c = new BN(this.getMerkleRoot().toString('hex'), 'hex'); var c = new BN(this.getMerkleRoot().toString('hex'), 'hex');
if (h.cmp(c) !== 0) { if (h.cmp(c) !== 0) {
@ -328,7 +287,7 @@ Block.prototype.validMerkleRoot = function validMerkleRoot() {
* @returns {Buffer} - The little endian hash buffer of the header * @returns {Buffer} - The little endian hash buffer of the header
*/ */
Block.prototype._getHash = function() { Block.prototype._getHash = function() {
return this.blockheader._getHash(); return this.header._getHash();
}; };
var idProperty = { var idProperty = {
@ -339,7 +298,7 @@ var idProperty = {
*/ */
get: function() { get: function() {
if (!this._id) { if (!this._id) {
this._id = this.blockheader.id; this._id = this.header.id;
} }
return this._id; return this._id;
}, },

23
lib/blockheader.js

@ -39,11 +39,9 @@ BlockHeader._from = function _from(arg) {
} else if (_.isObject(arg)) { } else if (_.isObject(arg)) {
info = { info = {
version: arg.version, version: arg.version,
prevblockidbuf: arg.prevblockidbuf, prevHash: arg.prevHash,
prevHash: arg.prevblockidbuf, merkleRoot: arg.merkleRoot,
merklerootbuf: arg.merklerootbuf,
time: arg.time, time: arg.time,
timestamp: arg.time,
bits: arg.bits, bits: arg.bits,
nonce: arg.nonce nonce: arg.nonce
}; };
@ -64,9 +62,8 @@ BlockHeader._fromJSON = function _fromJSON(data) {
} }
var info = { var info = {
version: data.version, version: data.version,
prevblockidbuf: new Buffer(data.prevblockidbuf, 'hex'), prevHash: new Buffer(data.prevHash, 'hex'),
prevHash: new Buffer(data.prevblockidbuf, 'hex'), merkleRoot: new Buffer(data.merkleRoot, 'hex'),
merklerootbuf: new Buffer(data.merklerootbuf, 'hex'),
time: data.time, time: data.time,
timestamp: data.time, timestamp: data.time,
bits: data.bits, bits: data.bits,
@ -124,8 +121,8 @@ BlockHeader.fromString = function fromString(str) {
BlockHeader._fromBufferReader = function _fromBufferReader(br) { BlockHeader._fromBufferReader = function _fromBufferReader(br) {
var info = { var info = {
version: br.readUInt32LE(), version: br.readUInt32LE(),
prevblockidbuf: br.read(32), prevHash: br.read(32),
merklerootbuf: br.read(32), merkleRoot: br.read(32),
time: br.readUInt32LE(), time: br.readUInt32LE(),
bits: br.readUInt32LE(), bits: br.readUInt32LE(),
nonce: br.readUInt32LE() nonce: br.readUInt32LE()
@ -148,8 +145,8 @@ BlockHeader.fromBufferReader = function fromBufferReader(br) {
BlockHeader.prototype.toObject = function toObject() { BlockHeader.prototype.toObject = function toObject() {
return { return {
version: this.version, version: this.version,
prevblockidbuf: this.prevblockidbuf.toString('hex'), prevHash: this.prevHash.toString('hex'),
merklerootbuf: this.merklerootbuf.toString('hex'), merkleRoot: this.merkleRoot.toString('hex'),
time: this.time, time: this.time,
bits: this.bits, bits: this.bits,
nonce: this.nonce nonce: this.nonce
@ -186,8 +183,8 @@ BlockHeader.prototype.toBufferWriter = function toBufferWriter(bw) {
bw = new BufferWriter(); bw = new BufferWriter();
} }
bw.writeUInt32LE(this.version); bw.writeUInt32LE(this.version);
bw.write(this.prevblockidbuf); bw.write(this.prevHash);
bw.write(this.merklerootbuf); bw.write(this.merkleRoot);
bw.writeUInt32LE(this.time); bw.writeUInt32LE(this.time);
bw.writeUInt32LE(this.bits); bw.writeUInt32LE(this.bits);
bw.writeUInt32LE(this.nonce); bw.writeUInt32LE(this.nonce);

30
test/block.js

@ -22,7 +22,7 @@ describe('Block', function() {
var magicnum = data.magicnum; var magicnum = data.magicnum;
var blockhex = data.blockhex; var blockhex = data.blockhex;
var blockbuf = new Buffer(blockhex, 'hex'); var blockbuf = new Buffer(blockhex, 'hex');
var blocksize = data.blocksize; var size = data.blocksize;
var bh = BlockHeader.fromBuffer(new Buffer(data.blockheaderhex, 'hex')); var bh = BlockHeader.fromBuffer(new Buffer(data.blockheaderhex, 'hex'));
var txsvi = Varint().fromNumber(data.txsvi); var txsvi = Varint().fromNumber(data.txsvi);
var txs = []; var txs = [];
@ -51,15 +51,15 @@ describe('Block', function() {
it('should set these known values', function() { it('should set these known values', function() {
var b = new Block({ var b = new Block({
magicnum: magicnum, magicnum: magicnum,
blocksize: blocksize, size: size,
blockheader: bh, header: bh,
txsvi: txsvi, txsvi: txsvi,
txs: txs txs: txs
}); });
should.exist(b.magicnum); should.exist(b.magicnum);
should.exist(b.blocksize); should.exist(b.size);
should.exist(b.txsvi); should.exist(b.txsvi);
should.exist(b.blockheader); should.exist(b.header);
should.exist(b.txs); should.exist(b.txs);
}); });
@ -69,14 +69,14 @@ describe('Block', function() {
it('should instantiate from a raw block binary', function() { it('should instantiate from a raw block binary', function() {
var x = Block.fromRawBlock(dataRawBlockBinary); var x = Block.fromRawBlock(dataRawBlockBinary);
x.blockheader.version.should.equal(2); x.header.version.should.equal(2);
BN(x.blockheader.bits).toString('hex').should.equal('1c3fffc0'); BN(x.header.bits).toString('hex').should.equal('1c3fffc0');
}); });
it('should instantiate from raw block buffer', function() { it('should instantiate from raw block buffer', function() {
var x = Block.fromRawBlock(dataRawBlockBuffer); var x = Block.fromRawBlock(dataRawBlockBuffer);
x.blockheader.version.should.equal(2); x.header.version.should.equal(2);
BN(x.blockheader.bits).toString('hex').should.equal('1c3fffc0'); BN(x.header.bits).toString('hex').should.equal('1c3fffc0');
}); });
}); });
@ -86,8 +86,8 @@ describe('Block', function() {
it('should set these known values', function() { it('should set these known values', function() {
var block = Block.fromJSON(json); var block = Block.fromJSON(json);
should.exist(block.magicnum); should.exist(block.magicnum);
should.exist(block.blocksize); should.exist(block.size);
should.exist(block.blockheader); should.exist(block.header);
should.exist(block.txsvi); should.exist(block.txsvi);
should.exist(block.txs); should.exist(block.txs);
}); });
@ -96,8 +96,8 @@ describe('Block', function() {
var block = Block(json); var block = Block(json);
should.exist(block.magicnum); should.exist(block.magicnum);
should.exist(block.blocksize); should.exist(block.size);
should.exist(block.blockheader); should.exist(block.header);
should.exist(block.txsvi); should.exist(block.txsvi);
should.exist(block.txs); should.exist(block.txs);
}); });
@ -110,8 +110,8 @@ describe('Block', function() {
var block = Block.fromJSON(json); var block = Block.fromJSON(json);
var b = JSON.parse(block.toJSON()); var b = JSON.parse(block.toJSON());
should.exist(b.magicnum); should.exist(b.magicnum);
should.exist(b.blocksize); should.exist(b.size);
should.exist(b.blockheader); should.exist(b.header);
should.exist(b.txsvi); should.exist(b.txsvi);
should.exist(b.txs); should.exist(b.txs);
}); });

32
test/blockheader.js

@ -23,8 +23,8 @@ describe('BlockHeader', function() {
var nonce = data.nonce; var nonce = data.nonce;
var bh = new BlockHeader({ var bh = new BlockHeader({
version: version, version: version,
prevblockidbuf: prevblockidbuf, prevHash: prevblockidbuf,
merklerootbuf: merklerootbuf, merkleRoot: merklerootbuf,
time: time, time: time,
bits: bits, bits: bits,
nonce: nonce nonce: nonce
@ -47,15 +47,15 @@ describe('BlockHeader', function() {
it('should set all the variables', function() { it('should set all the variables', function() {
var bh = new BlockHeader({ var bh = new BlockHeader({
version: version, version: version,
prevblockidbuf: prevblockidbuf, prevHash: prevblockidbuf,
merklerootbuf: merklerootbuf, merkleRoot: merklerootbuf,
time: time, time: time,
bits: bits, bits: bits,
nonce: nonce nonce: nonce
}); });
should.exist(bh.version); should.exist(bh.version);
should.exist(bh.prevblockidbuf); should.exist(bh.prevHash);
should.exist(bh.merklerootbuf); should.exist(bh.merkleRoot);
should.exist(bh.time); should.exist(bh.time);
should.exist(bh.bits); should.exist(bh.bits);
should.exist(bh.nonce); should.exist(bh.nonce);
@ -68,15 +68,15 @@ describe('BlockHeader', function() {
it('should set all the variables', function() { it('should set all the variables', function() {
var bh = BlockHeader.fromJSON({ var bh = BlockHeader.fromJSON({
version: version, version: version,
prevblockidbuf: prevblockidbuf.toString('hex'), prevHash: prevblockidbuf.toString('hex'),
merklerootbuf: merklerootbuf.toString('hex'), merkleRoot: merklerootbuf.toString('hex'),
time: time, time: time,
bits: bits, bits: bits,
nonce: nonce nonce: nonce
}); });
should.exist(bh.version); should.exist(bh.version);
should.exist(bh.prevblockidbuf); should.exist(bh.prevHash);
should.exist(bh.merklerootbuf); should.exist(bh.merkleRoot);
should.exist(bh.time); should.exist(bh.time);
should.exist(bh.bits); should.exist(bh.bits);
should.exist(bh.nonce); should.exist(bh.nonce);
@ -89,8 +89,8 @@ describe('BlockHeader', function() {
it('should set all the variables', function() { it('should set all the variables', function() {
var json = JSON.parse(bh.toJSON()); var json = JSON.parse(bh.toJSON());
should.exist(json.version); should.exist(json.version);
should.exist(json.prevblockidbuf); should.exist(json.prevHash);
should.exist(json.merklerootbuf); should.exist(json.merkleRoot);
should.exist(json.time); should.exist(json.time);
should.exist(json.bits); should.exist(json.bits);
should.exist(json.nonce); should.exist(json.nonce);
@ -104,8 +104,8 @@ describe('BlockHeader', function() {
var jsonString = JSON.stringify({ var jsonString = JSON.stringify({
version: version, version: version,
prevblockidbuf: prevblockidbuf, prevHash: prevblockidbuf,
merklerootbuf: merklerootbuf, merkleRoot: merklerootbuf,
time: time, time: time,
bits: bits, bits: bits,
nonce: nonce nonce: nonce
@ -113,8 +113,8 @@ describe('BlockHeader', function() {
var json = new BlockHeader(jsonString); var json = new BlockHeader(jsonString);
should.exist(json.version); should.exist(json.version);
should.exist(json.prevblockidbuf); should.exist(json.prevHash);
should.exist(json.merklerootbuf); should.exist(json.merkleRoot);
should.exist(json.time); should.exist(json.time);
should.exist(json.bits); should.exist(json.bits);
should.exist(json.nonce); should.exist(json.nonce);

10
test/data/blk86756-testnet.json

@ -1,10 +1,10 @@
{ {
"magicnum": 118034699, "magicnum": 118034699,
"blocksize": 8003, "size": 8003,
"blockheader": { "header": {
"version": 2, "version": 2,
"prevblockidbuf": "4baaa9507c3b27908397ea7bc177a998e9f4fe38b9d5130be7b5353c00000000", "prevHash": "4baaa9507c3b27908397ea7bc177a998e9f4fe38b9d5130be7b5353c00000000",
"merklerootbuf": "97fc4c97868288e984ff9f246f2c38510f5c37a3d5b41aae7004b01e2dd5e658", "merkleRoot": "97fc4c97868288e984ff9f246f2c38510f5c37a3d5b41aae7004b01e2dd5e658",
"time": 1371410638, "time": 1371410638,
"bits": 473956288, "bits": 473956288,
"nonce": 3594009557 "nonce": 3594009557
@ -597,4 +597,4 @@
}], }],
"nlocktime": 0 "nlocktime": 0
}] }]
} }

Loading…
Cancel
Save