You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
360 lines
8.2 KiB
360 lines
8.2 KiB
'use strict';
|
|
|
|
var Put = require('bufferput');
|
|
|
|
var Block = require('../block');
|
|
var BufferReader = require('../encoding/bufferreader');
|
|
var BufferUtil = require('../util/buffer');
|
|
var Random = require('../crypto/random');
|
|
|
|
var CONNECTION_NONCE = Random.getPseudoRandomBuffer(8);
|
|
var PROTOCOL_VERSION = 70000;
|
|
|
|
var MESSAGES = {
|
|
'version' : Version,
|
|
'verack': VerAck,
|
|
'inv': Inventory,
|
|
'ping': Ping,
|
|
'pong': Pong,
|
|
'addr': Addresses,
|
|
'getaddr': GetAddresses,
|
|
'reject': Reject
|
|
}
|
|
|
|
module.exports.buildMessage = function(command, payload) {
|
|
var Message = MESSAGES[command];
|
|
try {
|
|
console.log('Message Class', Message);
|
|
return new Message().fromBuffer(payload);
|
|
} catch (err) {
|
|
console.log('Error while parrsing message', command);
|
|
console.log(err);
|
|
}
|
|
}
|
|
|
|
// ====== VERSION MESSAGE ======
|
|
function Version(subversion, nonce) {
|
|
this.command = 'version';
|
|
this.version = PROTOCOL_VERSION;
|
|
this.subversion = subversion || '/BitcoinX:0.1/';
|
|
this.nonce = nonce || CONNECTION_NONCE;
|
|
};
|
|
|
|
Version.prototype.fromBuffer = function(payload) {
|
|
var message = new Version();
|
|
|
|
var parser = new BufferReader(payload);
|
|
message.version = parser.readUInt32LE();
|
|
message.services = parser.readUInt64LEBN();
|
|
message.timestamp = parser.readUInt64LEBN();
|
|
message.addr_me = parser.read(26);
|
|
message.addr_you = parser.read(26);
|
|
message.nonce = parser.read(8);
|
|
message.subversion = parser.readVarintBuf();
|
|
message.start_height = parser.readUInt32LE();
|
|
|
|
return message;
|
|
};
|
|
|
|
Version.prototype.serialize = function() {
|
|
var put = new Put();
|
|
put.word32le(this.version); // version
|
|
put.word64le(1); // services
|
|
put.word64le(Math.round(new Date().getTime() / 1000)); // timestamp
|
|
put.pad(26); // addr_me
|
|
put.pad(26); // addr_you
|
|
put.put(this.nonce);
|
|
put.varint(this.subversion.length);
|
|
put.put(new Buffer(this.subversion, 'ascii'));
|
|
put.word32le(0);
|
|
|
|
return put.buffer();
|
|
};
|
|
|
|
module.exports.Version = Version;
|
|
|
|
// ====== INV MESSAGE ======
|
|
function Inventory(inventory) {
|
|
this.command = 'inv';
|
|
this.inventory = inventory || [];
|
|
}
|
|
|
|
Inventory.prototype.fromBuffer = function(payload) {
|
|
var message = new Inventory();
|
|
|
|
var parser = new BufferReader(payload);
|
|
var count = parser.readVarintNum();
|
|
for (var i = 0; i < count; i++) {
|
|
message.inventory.push({
|
|
type: parser.readUInt32LE(),
|
|
hash: parser.read(32)
|
|
});
|
|
}
|
|
|
|
return message;
|
|
};
|
|
|
|
Inventory.prototype.serialize = function() {
|
|
var put = new Put();
|
|
|
|
put.varint(this.inventory.length);
|
|
this.inventory.forEach(function(value) {
|
|
value instanceof Block ? put.word32le(2) : put.word32le(1);
|
|
put.put(value.getHash());
|
|
});
|
|
|
|
return put.buffer();
|
|
};
|
|
|
|
module.exports.Inventory = Inventory;
|
|
|
|
// ====== GETDATA MESSAGE ======
|
|
function GetData(inventory) {
|
|
this.command = 'getdata';
|
|
this.inventory = inventory || [];
|
|
}
|
|
|
|
util.inherits(GetData, Inventory);
|
|
module.exports.GetData = GetData;
|
|
|
|
// ====== PING MESSAGE ======
|
|
function Ping(nonce) {
|
|
this.command = 'ping';
|
|
this.nonce = nonce || CONNECTION_NONCE;
|
|
}
|
|
|
|
Ping.prototype.fromBuffer = function(payload) {
|
|
var nonce = new BufferReader(payload).read(8);
|
|
return new Ping(nonce);
|
|
};
|
|
|
|
Ping.prototype.serialize = function() {
|
|
return this.nonce;
|
|
};
|
|
|
|
module.exports.Ping = Ping;
|
|
|
|
// ====== PONG MESSAGE ======
|
|
function Pong(nonce) {
|
|
this.command = 'pong';
|
|
this.nonce = nonce || CONNECTION_NONCE;
|
|
}
|
|
|
|
util.inherits(Pong, Ping);
|
|
module.exports.Pong = Pong;
|
|
|
|
// ====== ADDR MESSAGE ======
|
|
function Addresses(nonce) {
|
|
this.command = 'addr';
|
|
this.addresses = [];
|
|
}
|
|
|
|
Address.prototype.fromBuffer = function(payload) {
|
|
var message = new Address();
|
|
|
|
var parser = new BufferReader(payload);
|
|
var addrCount = Math.min(parser.readVarintNum(), 1000);
|
|
|
|
message.addresses = [];
|
|
for (var i = 0; i < addrCount; i++) {
|
|
// TODO: Time actually depends on the version of the other peer (>=31402)
|
|
message.addresses.push({
|
|
time: parser.readUInt32LE(),
|
|
services: parser.readUInt64LEBN(),
|
|
ip: parser.read(16),
|
|
port: parser.readUInt16BE()
|
|
});
|
|
}
|
|
|
|
return message;
|
|
};
|
|
|
|
Address.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER; // TODO
|
|
};
|
|
|
|
module.exports.Address = Address;
|
|
|
|
// ====== GETADDR MESSAGE ======
|
|
function GetAddresses() {
|
|
this.command = 'getaddr';
|
|
}
|
|
|
|
GetAddresses.prototype.fromBuffer = function() {
|
|
return new GetAddresses();
|
|
};
|
|
|
|
GetAddresses.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER;
|
|
};
|
|
|
|
module.exports.GetAddresses = GetAddresses;
|
|
|
|
// ====== VERACK MESSAGE ======
|
|
function VerAck() {
|
|
this.command = 'verack';
|
|
}
|
|
|
|
VerAck.prototype.fromBuffer = function() {
|
|
return new VerAck();
|
|
};
|
|
|
|
VerAck.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER;
|
|
};
|
|
|
|
module.exports.VerAck = VerAck;
|
|
|
|
// ====== REJECT MESSAGE ======
|
|
// TODO: Parse REJECT message
|
|
function Reject() {
|
|
this.command = 'reject';
|
|
}
|
|
|
|
Reject.prototype.fromBuffer = function() {
|
|
return new Reject();
|
|
};
|
|
|
|
Reject.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER;
|
|
};
|
|
|
|
module.exports.Reject = Reject;
|
|
|
|
// ====== ALERT MESSAGE ======
|
|
function Alert(payload) {
|
|
this.command = 'reject';
|
|
}
|
|
|
|
Alert.prototype.fromBuffer = function() {
|
|
var message = new Alert();
|
|
|
|
var parser = new BufferReader(payload);
|
|
message.payload = parser.readVarintBuf(); // TODO: Use current format
|
|
message.signature = parser.readVarintBuf();
|
|
return message;
|
|
};
|
|
|
|
Alert.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER; // TODO: Serialize
|
|
};
|
|
|
|
module.exports.Alert = Alert;
|
|
|
|
// ====== HEADERS MESSAGE ======
|
|
function Headers(blockheaders) {
|
|
this.command = 'headers';
|
|
this.headers = blockheaders || [];
|
|
}
|
|
|
|
Headers.prototype.fromBuffer = function() {
|
|
var message = new Headers();
|
|
|
|
var parser = new BufferReader(payload);
|
|
var count = parser.readVarintNum();
|
|
|
|
message.headers = [];
|
|
for (i = 0; i < count; i++) {
|
|
var header = Block().fromBufferReader(parser);
|
|
message.headers.push(header);
|
|
}
|
|
|
|
return message;
|
|
};
|
|
|
|
Headers.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER; // TODO: Serialize
|
|
};
|
|
|
|
module.exports.Headers = Headers;
|
|
|
|
// ====== BLOCK MESSAGE ======
|
|
function Block(block) {
|
|
this.command = 'block';
|
|
this.block = block;
|
|
}
|
|
|
|
Block.prototype.fromBuffer = function() {
|
|
var parser = new BufferReader(payload);
|
|
var block = Block().fromBufferReader(parser);
|
|
return new Block(block);
|
|
};
|
|
|
|
Block.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER; // TODO: Serialize
|
|
};
|
|
|
|
module.exports.Block = Block;
|
|
|
|
// ====== TX MESSAGE ======
|
|
function Transaction(transaction) {
|
|
this.command = 'tx';
|
|
this.transaction = transaction;
|
|
}
|
|
|
|
Transaction.prototype.fromBuffer = function() {
|
|
var parser = new BufferReader(payload);
|
|
var transaction = Transaction().fromBufferReader(parser);
|
|
return new Transaction(transaction);
|
|
};
|
|
|
|
Transaction.prototype.serialize = function() {
|
|
return BufferUtil.EMPTY_BUFFER; // TODO: Serialize
|
|
};
|
|
|
|
module.exports.Transaction = Transaction;
|
|
|
|
// ====== GETBLOCKS MESSAGE ======
|
|
function GetBlocks(starts, stop) {
|
|
this.command = 'getblocks';
|
|
this.version = PROTOCOL_VERSION;
|
|
this.starts = starts || [];
|
|
this.stop = stop || BufferUtil.NULL_HASH;
|
|
}
|
|
|
|
GetBlocks.prototype.fromBuffer = function() {
|
|
var message = new GetBlocks();
|
|
|
|
var parser = new BufferReader(payload);
|
|
message.version = parser.readUInt32LE();
|
|
|
|
var startCount = Math.min(parser.readVarintNum(), 500);
|
|
message.starts = [];
|
|
for (var i = 0; i < startCount; i++) {
|
|
message.starts.push(parser.read(32));
|
|
}
|
|
message.stop = parser.read(32);
|
|
};
|
|
|
|
GetBlocks.prototype.serialize = function() {
|
|
var put = new Put();
|
|
put.word32le(this.version);
|
|
put.varint(this.starts.length);
|
|
|
|
for (var i = 0; i < starts.length; i++) {
|
|
if (this.starts[i].length != 32) {
|
|
throw new Error('Invalid hash length');
|
|
}
|
|
put.put(this.starts[i]);
|
|
}
|
|
|
|
if (this.stop.length != 32) {
|
|
throw new Error('Invalid hash length');
|
|
}
|
|
put.put(this.stop);
|
|
|
|
return put.buffer();
|
|
};
|
|
|
|
module.exports.GetBlocks = GetBlocks;
|
|
|
|
// ====== GETHEADERS MESSAGE ======
|
|
function GetHeaders(starts, stop) {
|
|
this.command = 'getheaders';
|
|
this.version = PROTOCOL_VERSION;
|
|
this.starts = starts || [];
|
|
this.stop = stop || BufferUtil.NULL_HASH;
|
|
}
|
|
|
|
util.inherits(GetHeaders, GetBlocks);
|
|
module.exports.GetHeaders = GetHeaders;
|
|
|