fcoin/lib/workers/parser.js

184 lines
3.9 KiB
JavaScript

/*!
* workers.js - worker processes for bcoin
* Copyright (c) 2014-2015, Fedor Indutny (MIT License)
* Copyright (c) 2014-2017, Christopher Jeffrey (MIT License).
* https://github.com/bcoin-org/bcoin
*/
'use strict';
var assert = require('assert');
var EventEmitter = require('events').EventEmitter;
var util = require('../utils/util');
var packets = require('./packets');
/**
* Parser
* @alias module:workers.Parser
* @constructor
*/
function Parser() {
if (!(this instanceof Parser))
return new Parser();
EventEmitter.call(this);
this.waiting = 9;
this.header = null;
this.pending = [];
this.total = 0;
}
util.inherits(Parser, EventEmitter);
Parser.prototype.feed = function feed(data) {
var chunk;
this.total += data.length;
this.pending.push(data);
while (this.total >= this.waiting) {
chunk = this.read(this.waiting);
this.parse(chunk);
}
};
Parser.prototype.read = function read(size) {
var pending, chunk, off, len;
assert(this.total >= size, 'Reading too much.');
if (size === 0)
return Buffer.alloc(0);
pending = this.pending[0];
if (pending.length > size) {
chunk = pending.slice(0, size);
this.pending[0] = pending.slice(size);
this.total -= chunk.length;
return chunk;
}
if (pending.length === size) {
chunk = this.pending.shift();
this.total -= chunk.length;
return chunk;
}
chunk = Buffer.allocUnsafe(size);
off = 0;
len = 0;
while (off < chunk.length) {
pending = this.pending[0];
len = pending.copy(chunk, off);
if (len === pending.length)
this.pending.shift();
else
this.pending[0] = pending.slice(len);
off += len;
}
assert.equal(off, chunk.length);
this.total -= chunk.length;
return chunk;
};
Parser.prototype.parse = function parse(data) {
var header = this.header;
var packet;
if (!header) {
try {
header = this.parseHeader(data);
} catch (e) {
this.emit('error', e);
return;
}
this.header = header;
this.waiting = header.size + 1;
return;
}
this.waiting = 9;
this.header = null;
try {
packet = this.parsePacket(header, data);
} catch (e) {
this.emit('error', e);
return;
}
if (data[data.length - 1] !== 0x0a) {
this.emit('error', new Error('No trailing newline.'));
return;
}
packet.id = header.id;
this.emit('packet', packet);
};
Parser.prototype.parseHeader = function parseHeader(data) {
var id = data.readUInt32LE(0, true);
var cmd = data.readUInt8(4, true);
var size = data.readUInt32LE(5, true);
return new Header(id, cmd, size);
};
Parser.prototype.parsePacket = function parsePacket(header, data) {
switch (header.cmd) {
case packets.types.EVENT:
return packets.EventPacket.fromRaw(data);
case packets.types.LOG:
return packets.LogPacket.fromRaw(data);
case packets.types.ERROR:
return packets.ErrorPacket.fromRaw(data);
case packets.types.ERRORRESULT:
return packets.ErrorResultPacket.fromRaw(data);
case packets.types.VERIFYRESULT:
return packets.VerifyResultPacket.fromRaw(data);
case packets.types.SIGNRESULT:
return packets.SignResultPacket.fromRaw(data);
case packets.types.VERIFYINPUTRESULT:
return packets.VerifyInputResultPacket.fromRaw(data);
case packets.types.SIGNINPUTRESULT:
return packets.SignInputResultPacket.fromRaw(data);
case packets.types.ECVERIFYRESULT:
return packets.ECVerifyResultPacket.fromRaw(data);
case packets.types.ECSIGNRESULT:
return packets.ECSignResultPacket.fromRaw(data);
case packets.types.MINERESULT:
return packets.MineResultPacket.fromRaw(data);
case packets.types.SCRYPTRESULT:
return packets.ScryptResultPacket.fromRaw(data);
default:
throw new Error('Unknown packet.');
}
};
/**
* Header
* @constructor
* @ignore
*/
function Header(id, cmd, size) {
this.id = id;
this.cmd = cmd;
this.size = size;
}
/*
* Expose
*/
module.exports = Parser;