itns-sidechain/lib/net/parser.js

194 lines
3.7 KiB
JavaScript
Raw Normal View History

/*!
2015-12-18 22:53:31 -08:00
* parser.js - packet parser for bcoin
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
2017-02-03 22:47:26 -08:00
* Copyright (c) 2014-2017, Christopher Jeffrey (MIT License).
2016-06-09 16:18:50 -07:00
* https://github.com/bcoin-org/bcoin
2015-12-18 22:53:31 -08:00
*/
2016-06-13 01:06:01 -07:00
'use strict';
2017-01-21 03:01:18 -08:00
var assert = require('assert');
2014-04-28 17:43:13 +04:00
var EventEmitter = require('events').EventEmitter;
2017-01-21 03:01:18 -08:00
var Network = require('../protocol/network');
2016-11-19 10:45:31 -08:00
var util = require('../utils/util');
var crypto = require('../crypto/crypto');
2017-01-06 09:57:55 -08:00
var common = require('./common');
2016-08-25 01:35:20 -07:00
var packets = require('./packets');
2014-04-28 17:12:26 +04:00
2015-12-18 22:53:31 -08:00
/**
* Protocol packet parser
2017-02-03 22:47:26 -08:00
* @alias module:net.Parser
* @constructor
2017-01-21 03:01:18 -08:00
* @param {Network} network
* @emits Parser#error
* @emits Parser#packet
2015-12-18 22:53:31 -08:00
*/
2017-01-21 03:01:18 -08:00
function Parser(network) {
2014-04-28 17:43:13 +04:00
if (!(this instanceof Parser))
2017-01-21 03:01:18 -08:00
return new Parser(network);
2014-04-28 17:12:26 +04:00
2014-04-28 17:43:13 +04:00
EventEmitter.call(this);
2014-04-28 17:12:26 +04:00
2017-01-21 03:01:18 -08:00
this.network = Network.get(network);
2016-07-26 12:23:40 -07:00
2014-04-28 17:12:26 +04:00
this.pending = [];
2016-07-27 04:10:08 -07:00
this.total = 0;
2014-04-28 17:12:26 +04:00
this.waiting = 24;
2016-08-26 16:49:38 -07:00
this.header = null;
2014-04-28 17:12:26 +04:00
}
2015-12-18 22:37:02 -08:00
2016-11-19 10:45:31 -08:00
util.inherits(Parser, EventEmitter);
2014-04-28 17:12:26 +04:00
/**
* Emit an error.
* @private
2016-09-16 18:18:34 -07:00
* @param {...String} msg
*/
2016-09-16 18:18:34 -07:00
Parser.prototype.error = function error() {
2016-11-19 14:33:18 -08:00
var msg = util.fmt.apply(util, arguments);
2016-09-16 18:18:34 -07:00
this.emit('error', new Error(msg));
2014-05-05 16:52:17 +04:00
};
/**
* Feed data to the parser.
* @param {Buffer} data
*/
2014-04-30 12:44:59 +04:00
Parser.prototype.feed = function feed(data) {
2016-03-14 13:25:02 -07:00
var chunk, off, len;
2015-12-18 22:37:02 -08:00
2016-07-27 04:10:08 -07:00
this.total += data.length;
2014-04-30 12:44:59 +04:00
this.pending.push(data);
2015-12-18 22:37:02 -08:00
2016-07-27 04:10:08 -07:00
while (this.total >= this.waiting) {
chunk = new Buffer(this.waiting);
2015-12-18 22:37:02 -08:00
off = 0;
len = 0;
2016-03-14 13:25:02 -07:00
while (off < chunk.length) {
2016-04-18 00:36:03 -07:00
len = this.pending[0].copy(chunk, off);
2014-04-28 19:57:40 +04:00
if (len === this.pending[0].length)
this.pending.shift();
else
this.pending[0] = this.pending[0].slice(len);
2014-04-28 17:12:26 +04:00
off += len;
}
2015-12-18 22:37:02 -08:00
2014-04-28 17:12:26 +04:00
assert.equal(off, chunk.length);
2016-07-27 04:10:08 -07:00
this.total -= chunk.length;
2014-04-28 17:12:26 +04:00
this.parse(chunk);
}
};
/**
* Parse a fully-buffered chunk.
* @param {Buffer} chunk
*/
2016-08-26 16:49:38 -07:00
Parser.prototype.parse = function parse(data) {
var payload, checksum;
2016-05-18 07:09:16 -07:00
2017-01-06 09:57:55 -08:00
assert(data.length <= common.MAX_MESSAGE);
2016-03-04 14:51:37 -08:00
2016-08-26 16:49:38 -07:00
if (!this.header) {
this.header = this.parseHeader(data);
2015-12-18 22:37:02 -08:00
return;
2014-04-28 17:12:26 +04:00
}
2015-12-18 22:37:02 -08:00
2016-09-16 19:32:48 -07:00
checksum = crypto.hash256(data).readUInt32LE(0, true);
2015-12-18 22:37:02 -08:00
2016-08-26 16:49:38 -07:00
if (checksum !== this.header.checksum) {
2016-05-18 18:54:46 -07:00
this.waiting = 24;
2016-08-26 16:49:38 -07:00
this.header = null;
2017-01-26 01:48:54 -08:00
this.error('Invalid checksum: %s.', util.hex32(checksum));
2017-01-21 03:01:18 -08:00
return;
2016-05-18 18:54:46 -07:00
}
2015-12-18 22:37:02 -08:00
2016-02-24 20:48:08 -08:00
try {
2016-08-26 16:49:38 -07:00
payload = this.parsePayload(this.header.cmd, data);
2016-02-24 20:48:08 -08:00
} catch (e) {
2016-05-18 17:32:35 -07:00
this.waiting = 24;
2016-08-26 16:49:38 -07:00
this.header = null;
2016-09-16 19:35:04 -07:00
this.emit('error', e);
2016-05-18 17:32:35 -07:00
return;
2016-02-24 20:48:08 -08:00
}
2015-12-18 22:37:02 -08:00
this.waiting = 24;
2016-08-26 16:49:38 -07:00
this.header = null;
2016-09-16 19:35:04 -07:00
this.emit('packet', payload);
2014-04-28 17:12:26 +04:00
};
/**
* Parse buffered packet header.
2016-09-16 17:34:34 -07:00
* @param {Buffer} data - Header.
* @returns {Header}
*/
2016-09-16 17:34:34 -07:00
Parser.prototype.parseHeader = function parseHeader(data) {
2016-08-26 16:49:38 -07:00
var i, magic, cmd, size, checksum;
2015-12-18 22:37:02 -08:00
2016-09-16 17:34:34 -07:00
magic = data.readUInt32LE(0, true);
2015-12-18 22:37:02 -08:00
2017-01-21 03:56:30 -08:00
if (magic !== this.network.magic) {
this.error('Invalid magic value: %s.', util.hex32(magic));
return null;
}
2014-04-28 17:12:26 +04:00
// Count length of the cmd
2016-09-16 17:34:34 -07:00
for (i = 0; data[i + 4] !== 0 && i < 12; i++);
2015-12-18 22:37:02 -08:00
2017-01-21 03:56:30 -08:00
if (i === 12) {
this.error('Non NULL-terminated command.');
return null;
}
2014-04-28 17:12:26 +04:00
2016-09-16 17:34:34 -07:00
cmd = data.toString('ascii', 4, 4 + i);
2016-05-18 18:54:46 -07:00
2016-09-16 17:34:34 -07:00
size = data.readUInt32LE(16, true);
2014-04-28 17:12:26 +04:00
2017-01-06 09:57:55 -08:00
if (size > common.MAX_MESSAGE) {
2016-05-18 18:54:46 -07:00
this.waiting = 24;
2017-01-21 03:01:18 -08:00
this.error('Packet length too large: %dmb.', util.mb(size));
2017-01-21 03:56:30 -08:00
return null;
2016-05-18 18:54:46 -07:00
}
2016-03-04 14:51:37 -08:00
2016-09-16 19:35:04 -07:00
this.waiting = size;
2016-09-16 17:34:34 -07:00
checksum = data.readUInt32LE(20, true);
2016-06-22 11:44:43 -07:00
2016-08-26 16:49:38 -07:00
return new Header(cmd, size, checksum);
2014-04-28 17:43:13 +04:00
};
/**
* Parse a payload.
* @param {String} cmd - Packet type.
2016-08-25 01:35:20 -07:00
* @param {Buffer} data - Payload.
* @returns {Object}
*/
2016-08-25 01:35:20 -07:00
Parser.prototype.parsePayload = function parsePayload(cmd, data) {
2016-08-26 16:49:38 -07:00
return packets.fromRaw(cmd, data);
};
2015-12-18 22:53:31 -08:00
2016-06-22 11:44:43 -07:00
/**
2016-09-16 17:34:34 -07:00
* Packet Header
2016-06-22 11:44:43 -07:00
* @constructor
2017-02-03 22:47:26 -08:00
* @ignore
2016-06-22 11:44:43 -07:00
*/
2016-08-26 16:49:38 -07:00
function Header(cmd, size, checksum) {
2016-06-22 11:44:43 -07:00
this.cmd = cmd;
this.size = size;
this.checksum = checksum;
}
2016-05-15 18:07:06 -07:00
/*
* Expose
*/
2016-05-13 09:23:57 -07:00
module.exports = Parser;