bitcoinjs-lib/src/block.js

174 lines
4 KiB
JavaScript
Raw Normal View History

2016-05-05 00:42:05 +10:00
var createHash = require('create-hash')
2014-10-16 15:30:57 +11:00
var bufferutils = require('./bufferutils')
var bcrypto = require('./crypto')
var bufferCompare = require('buffer-compare')
var bufferReverse = require('buffer-reverse')
2014-10-16 15:30:57 +11:00
var Transaction = require('./transaction')
2015-02-23 10:36:57 +11:00
function Block () {
2014-10-16 15:30:57 +11:00
this.version = 1
this.prevHash = null
this.merkleRoot = null
this.timestamp = 0
this.bits = 0
this.nonce = 0
}
2015-02-23 10:36:57 +11:00
Block.fromBuffer = function (buffer) {
2015-08-11 17:03:46 +10:00
if (buffer.length < 80) throw new Error('Buffer too small (< 80 bytes)')
2014-10-16 15:30:57 +11:00
var offset = 0
2015-02-23 10:36:57 +11:00
function readSlice (n) {
2014-10-16 15:30:57 +11:00
offset += n
return buffer.slice(offset - n, offset)
}
2015-02-23 10:36:57 +11:00
function readUInt32 () {
2014-10-16 15:30:57 +11:00
var i = buffer.readUInt32LE(offset)
offset += 4
return i
}
var block = new Block()
block.version = readUInt32()
block.prevHash = readSlice(32)
block.merkleRoot = readSlice(32)
block.timestamp = readUInt32()
block.bits = readUInt32()
block.nonce = readUInt32()
if (buffer.length === 80) return block
2015-02-23 10:36:57 +11:00
function readVarInt () {
2014-10-16 15:30:57 +11:00
var vi = bufferutils.readVarInt(buffer, offset)
offset += vi.size
return vi.number
}
2015-02-23 10:36:57 +11:00
function readTransaction () {
2015-02-19 18:59:56 +11:00
var tx = Transaction.fromBuffer(buffer.slice(offset), true)
2014-10-16 15:30:57 +11:00
offset += tx.byteLength()
2014-10-16 15:30:57 +11:00
return tx
}
var nTransactions = readVarInt()
block.transactions = []
for (var i = 0; i < nTransactions; ++i) {
var tx = readTransaction()
block.transactions.push(tx)
}
return block
}
2015-02-23 10:36:57 +11:00
Block.fromHex = function (hex) {
2014-10-16 15:30:57 +11:00
return Block.fromBuffer(new Buffer(hex, 'hex'))
}
2015-02-23 10:36:57 +11:00
Block.prototype.getHash = function () {
return bcrypto.hash256(this.toBuffer(true))
}
2015-02-23 10:36:57 +11:00
Block.prototype.getId = function () {
return bufferReverse(this.getHash()).toString('hex')
}
2015-02-23 10:36:57 +11:00
Block.prototype.getUTCDate = function () {
var date = new Date(0) // epoch
date.setUTCSeconds(this.timestamp)
return date
}
2015-02-23 10:36:57 +11:00
Block.prototype.toBuffer = function (headersOnly) {
2014-10-16 15:30:57 +11:00
var buffer = new Buffer(80)
var offset = 0
2015-02-23 10:36:57 +11:00
function writeSlice (slice) {
2014-10-16 15:30:57 +11:00
slice.copy(buffer, offset)
offset += slice.length
}
2015-02-23 10:36:57 +11:00
function writeUInt32 (i) {
2014-10-16 15:30:57 +11:00
buffer.writeUInt32LE(i, offset)
offset += 4
}
writeUInt32(this.version)
writeSlice(this.prevHash)
writeSlice(this.merkleRoot)
writeUInt32(this.timestamp)
writeUInt32(this.bits)
writeUInt32(this.nonce)
if (headersOnly || !this.transactions) return buffer
2014-10-17 13:07:14 +11:00
var txLenBuffer = bufferutils.varIntBuffer(this.transactions.length)
2015-02-23 10:36:57 +11:00
var txBuffers = this.transactions.map(function (tx) {
2014-10-16 15:30:57 +11:00
return tx.toBuffer()
})
return Buffer.concat([buffer, txLenBuffer].concat(txBuffers))
}
2015-02-23 10:36:57 +11:00
Block.prototype.toHex = function (headersOnly) {
2014-10-16 15:30:57 +11:00
return this.toBuffer(headersOnly).toString('hex')
}
Block.calculateTarget = function (bits) {
var exponent = ((bits & 0xff000000) >> 24) - 3
2015-12-10 08:02:58 +11:00
var mantissa = bits & 0x007fffff
var i = 31 - exponent
var target = new Buffer(32)
target.fill(0)
target[i] = mantissa & 0xff
target[i - 1] = mantissa >> 8
target[i - 2] = mantissa >> 16
target[i - 3] = mantissa >> 24
2015-12-10 08:02:58 +11:00
return target
}
2016-05-05 00:42:05 +10:00
Block.calculateMerkleRoot = function (transactions) {
var length = transactions.length
if (length === 0) throw TypeError('Cannot compute merkle root for zero transactions')
var hashes = transactions.map(function (transaction) { return transaction.getHash() })
while (length > 1) {
var j = 0
for (var i = 0; i < length; i += 2, ++j) {
var hasher = createHash('sha256')
hasher.update(hashes[i])
hasher.update(i + 1 !== length ? hashes[i + 1] : hashes[i])
hashes[j] = bcrypto.sha256(hasher.digest())
}
length = j
}
return hashes[0]
}
Block.prototype.checkMerkleRoot = function () {
if (!this.transactions) return false
var actualMerkleRoot = Block.calculateMerkleRoot(this.transactions)
return bufferCompare(this.merkleRoot, actualMerkleRoot) === 0
}
Block.prototype.checkProofOfWork = function () {
var hash = bufferReverse(this.getHash())
2015-12-11 12:39:09 +11:00
var target = Block.calculateTarget(this.bits)
2015-12-08 18:51:35 +11:00
return bufferCompare(hash, target) <= 0
2015-12-08 18:51:35 +11:00
}
2014-10-16 15:30:57 +11:00
module.exports = Block