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

'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;