Merge pull request #97 from dcousens/reverseVarInt
Adds varIntToNum conversion function
This commit is contained in:
commit
2f5b302288
4 changed files with 83 additions and 5 deletions
|
@ -134,6 +134,28 @@ function numToVarInt(num) {
|
||||||
return [255].concat(numToBytes(num, 8));
|
return [255].concat(numToBytes(num, 8));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turn an VarInt into an integer
|
||||||
|
*
|
||||||
|
* "var_int" is a variable length integer used by Bitcoin's binary format.
|
||||||
|
*
|
||||||
|
* Returns { bytes: bytesUsed, number: theNumber }
|
||||||
|
*/
|
||||||
|
function varIntToNum(bytes) {
|
||||||
|
var prefix = bytes[0]
|
||||||
|
|
||||||
|
var viBytes =
|
||||||
|
prefix < 253 ? bytes.slice(0, 1)
|
||||||
|
: prefix === 253 ? bytes.slice(1, 3)
|
||||||
|
: prefix === 254 ? bytes.slice(1, 5)
|
||||||
|
: bytes.slice(1, 9)
|
||||||
|
|
||||||
|
return {
|
||||||
|
bytes: prefix < 253 ? viBytes : bytes.slice(0, viBytes.length + 1),
|
||||||
|
number: bytesToNum(viBytes)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function bytesToWords(bytes) {
|
function bytesToWords(bytes) {
|
||||||
var words = [];
|
var words = [];
|
||||||
for (var i = 0, b = 0; i < bytes.length; i++, b += 8) {
|
for (var i = 0, b = 0; i < bytes.length; i++, b += 8) {
|
||||||
|
@ -176,6 +198,7 @@ module.exports = {
|
||||||
numToBytes: numToBytes,
|
numToBytes: numToBytes,
|
||||||
bytesToNum: bytesToNum,
|
bytesToNum: bytesToNum,
|
||||||
numToVarInt: numToVarInt,
|
numToVarInt: numToVarInt,
|
||||||
|
varIntToNum: varIntToNum,
|
||||||
bytesToWords: bytesToWords,
|
bytesToWords: bytesToWords,
|
||||||
wordsToBytes: wordsToBytes,
|
wordsToBytes: wordsToBytes,
|
||||||
bytesToWordArray: bytesToWordArray,
|
bytesToWordArray: bytesToWordArray,
|
||||||
|
|
|
@ -254,11 +254,11 @@ Transaction.deserialize = function(buffer) {
|
||||||
return buffer[pos-1] + readAsInt(bytes-1) * 256;
|
return buffer[pos-1] + readAsInt(bytes-1) * 256;
|
||||||
}
|
}
|
||||||
var readVarInt = function() {
|
var readVarInt = function() {
|
||||||
pos++;
|
var bytes = buffer.slice(pos, pos + 9) // maximum possible number of bytes to read
|
||||||
if (buffer[pos-1] < 253) {
|
var result = convert.varIntToNum(bytes)
|
||||||
return buffer[pos-1];
|
|
||||||
}
|
pos += result.bytes.length
|
||||||
return readAsInt(buffer[pos-1] - 251);
|
return result.number
|
||||||
}
|
}
|
||||||
var readBytes = function(bytes) {
|
var readBytes = function(bytes) {
|
||||||
pos += bytes;
|
pos += bytes;
|
||||||
|
|
|
@ -94,6 +94,45 @@ describe('convert', function() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('varIntToNum', function() {
|
||||||
|
it('works on valid input', function() {
|
||||||
|
var data = [
|
||||||
|
[0], [128], [252], // 8-bit
|
||||||
|
[253, 0, 1], [253, 0, 2], [253, 0, 4], // 16-bit
|
||||||
|
[254, 5, 0, 1, 0], // 32-bit
|
||||||
|
[255, 3, 0, 0, 0, 1, 0, 0, 0] // 64-bit
|
||||||
|
]
|
||||||
|
var expected = [
|
||||||
|
0, 128, 252, // 8-bit
|
||||||
|
256, 512, 1024, // 16-bit
|
||||||
|
65541, // 32-bit
|
||||||
|
4294967299, // 64-bit
|
||||||
|
]
|
||||||
|
|
||||||
|
for (var i = 0; i < data.length; ++i) {
|
||||||
|
var actual = convert.varIntToNum(data[i])
|
||||||
|
assert.equal(actual.number, expected[i])
|
||||||
|
assert.deepEqual(actual.bytes, data[i])
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
it('uses only what is necessary', function() {
|
||||||
|
var data = [
|
||||||
|
[0, 99],
|
||||||
|
[253, 0, 1, 99],
|
||||||
|
[254, 5, 0, 1, 0, 99],
|
||||||
|
[255, 3, 0, 0, 0, 1, 0, 0, 0, 99]
|
||||||
|
]
|
||||||
|
var expected = [0, 256, 65541, 4294967299]
|
||||||
|
|
||||||
|
for (var i = 0; i < data.length; ++i) {
|
||||||
|
var actual = convert.varIntToNum(data[i])
|
||||||
|
assert.equal(actual.number, expected[i])
|
||||||
|
assert.deepEqual(actual.bytes, data[i].slice(0, -1))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('reverseEndian', function() {
|
describe('reverseEndian', function() {
|
||||||
it('works', function() {
|
it('works', function() {
|
||||||
var bigEndian = "6a4062273ac4f9ea4ffca52d9fd102b08f6c32faa0a4d1318e3a7b2e437bb9c7"
|
var bigEndian = "6a4062273ac4f9ea4ffca52d9fd102b08f6c32faa0a4d1318e3a7b2e437bb9c7"
|
||||||
|
|
|
@ -69,6 +69,22 @@ describe('Transaction', function() {
|
||||||
var hashHex = "a9d4599e15b53f3eb531608ddb31f48c695c3d0b3538a6bda871e8b34f2f430c"
|
var hashHex = "a9d4599e15b53f3eb531608ddb31f48c695c3d0b3538a6bda871e8b34f2f430c"
|
||||||
assert.deepEqual(tx.hash, convert.hexToBytes(hashHex))
|
assert.deepEqual(tx.hash, convert.hexToBytes(hashHex))
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('decodes large inputs correctly', function() {
|
||||||
|
// transaction has only 1 input
|
||||||
|
var tx = new Transaction()
|
||||||
|
tx.addInput("0cb859105100ebc3344f749c835c7af7d7103ec0d8cbc3d8ccbd5d28c3c36b57", 0)
|
||||||
|
tx.addOutput("15mMHKL96tWAUtqF3tbVf99Z8arcmnJrr3", 100)
|
||||||
|
|
||||||
|
// but we're going to replace the tx.ins.length VarInt with a 32-bit equivalent
|
||||||
|
// however the same resultant number of inputs (1)
|
||||||
|
var bytes = tx.serialize()
|
||||||
|
var mutated = bytes.slice(0, 4).concat([254, 1, 0, 0, 0], bytes.slice(5))
|
||||||
|
|
||||||
|
// the deserialized-serialized transaction should return to its original state (== tx)
|
||||||
|
var bytes2 = Transaction.deserialize(mutated).serialize()
|
||||||
|
assert.deepEqual(bytes, bytes2)
|
||||||
|
});
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('creating a transaction', function() {
|
describe('creating a transaction', function() {
|
||||||
|
|
Loading…
Add table
Reference in a new issue