optimize parsers.

This commit is contained in:
Christopher Jeffrey 2016-07-27 04:56:07 -07:00
parent 0fa972dcb3
commit 20d69b7cbc
No known key found for this signature in database
GPG Key ID: 8962AB9DE6666BBD
3 changed files with 142 additions and 122 deletions

View File

@ -80,7 +80,8 @@ function BIP151Stream(cipher, key) {
this.pending = [];
this.total = 0;
this.waiting = 0;
this.waiting = 4;
this.hasSize = false;
}
utils.inherits(BIP151Stream, EventEmitter);
@ -178,7 +179,7 @@ BIP151Stream.prototype.sequence = function sequence() {
*/
BIP151Stream.prototype.update = function update() {
this.iv.writeUInt32LE(this.seq, 4, true);
this.iv.writeUInt32LE(this.seq, 0, true);
return this.iv;
};
@ -267,89 +268,101 @@ BIP151Stream.prototype.verify = function verify(tag) {
};
/**
* Parse a ciphertext payload chunk.
* Parse ciphertext data and split into chunks.
* Potentially emits a `packet` event.
* @param {Buffer} data
*/
BIP151Stream.prototype.feed = function feed(data) {
var chunk, size, payload, tag, p, cmd, body;
var chunk, off, len;
while (data.length) {
if (!this.waiting) {
this.total += data.length;
this.pending.push(data);
this.total += data.length;
this.pending.push(data);
if (this.total < 4)
break;
while (this.total >= this.waiting) {
chunk = new Buffer(this.waiting);
off = 0;
len = 0;
chunk = concat(this.pending);
this.total = 0;
this.pending.length = 0;
size = this.decryptSize(chunk);
data = chunk.slice(4);
// Allow 3 batched packets of max message size (12mb).
// Not technically standard, but this protects us
// from buffering tons of data due to either an
// potential dos'er or a cipher state mismatch.
// Note that 6 is the minimum size:
// cmd=varint(1) string(1) length(4) data(0)
if (size < 6 || size > constants.MAX_MESSAGE * 3) {
this.emit('error', new Error('Bad packet size.'));
continue;
}
this.waiting = size + 16;
continue;
while (off < chunk.length) {
len = this.pending[0].copy(chunk, off);
if (len === this.pending[0].length)
this.pending.shift();
else
this.pending[0] = this.pending[0].slice(len);
off += len;
}
this.total += data.length;
this.pending.push(data);
assert.equal(off, chunk.length);
if (this.total < this.waiting)
break;
this.total -= chunk.length;
this.parse(chunk);
}
};
chunk = concat(this.pending);
payload = chunk.slice(0, this.waiting - 16);
tag = chunk.slice(this.waiting - 16, this.waiting);
data = chunk.slice(this.waiting);
/**
* Parse a ciphertext payload chunk.
* Potentially emits a `packet` event.
* @param {Buffer} data
*/
this.total = 0;
this.pending.length = 0;
this.waiting = 0;
BIP151Stream.prototype.parse = function parse(data) {
var size, payload, tag, p, cmd, body;
// Authenticate payload before decrypting.
// This ensures the cipher state isn't altered
// if the payload integrity has been compromised.
this.auth(payload);
this.finish();
if (!this.hasSize) {
size = this.decryptSize(data);
if (!this.verify(tag)) {
this.sequence();
this.emit('error', new Error('Bad tag.'));
continue;
// Allow 3 batched packets of max message size (12mb).
// Not technically standard, but this protects us
// from buffering tons of data due to either an
// potential dos'er or a cipher state mismatch.
// Note that 6 is the minimum size:
// cmd=varint(1) string(1) length(4) data(0)
if (size < 6 || size > constants.MAX_MESSAGE * 3) {
this.waiting = 4;
this.emit('error', new Error('Bad packet size.'));
return;
}
this.decrypt(payload);
this.hasSize = true;
this.waiting = size + 16;
return;
}
payload = data.slice(0, this.waiting - 16);
tag = data.slice(this.waiting - 16, this.waiting);
this.hasSize = false;
this.waiting = 4;
// Authenticate payload before decrypting.
// This ensures the cipher state isn't altered
// if the payload integrity has been compromised.
this.auth(payload);
this.finish();
if (!this.verify(tag)) {
this.sequence();
this.emit('error', new Error('Bad tag.'));
return;
}
p = bcoin.reader(payload, true);
this.decrypt(payload);
this.sequence();
while (p.left()) {
try {
cmd = p.readVarString('ascii');
body = p.readBytes(p.readU32());
} catch (e) {
this.emit('error', e);
break;
}
p = bcoin.reader(payload, true);
this.emit('packet', cmd, body);
while (p.left()) {
try {
cmd = p.readVarString('ascii');
body = p.readBytes(p.readU32());
} catch (e) {
this.emit('error', e);
return;
}
this.emit('packet', cmd, body);
}
};
@ -626,16 +639,6 @@ BIP151.prototype.packet = function packet(cmd, body) {
return this.output.packet(cmd, body);
};
/*
* Helpers
*/
function concat(buffers) {
return buffers.length > 1
? Buffer.concat(buffers)
: buffers[0];
}
/*
* Expose
*/

View File

@ -93,9 +93,7 @@ Parser.prototype.feed = function feed(data) {
this.pending.push(data);
while (this.total >= this.waiting) {
// Concat chunks
chunk = new Buffer(this.waiting);
off = 0;
len = 0;
@ -110,7 +108,6 @@ Parser.prototype.feed = function feed(data) {
assert.equal(off, chunk.length);
// Slice buffers
this.total -= chunk.length;
this.parse(chunk);
}

View File

@ -951,57 +951,83 @@ function Parser() {
utils.inherits(Parser, EventEmitter);
Parser.prototype.feed = function feed(data) {
var chunk, header, body;
var chunk, off, len;
while (data) {
this.total += data.length;
this.pending.push(data);
data = null;
if (this.total < this.waiting)
break;
chunk = concat(this.pending);
if (chunk.length > this.waiting) {
data = chunk.slice(this.waiting);
chunk = chunk.slice(0, this.waiting);
}
if (!this.header) {
this.header = this.parseHeader(chunk);
this.waiting = this.header.size;
this.pending.length = 0;
this.total = 0;
if (this.header.magic !== 0xdeadbeef) {
this.header = null;
this.waiting = 12;
this.emit('error', new Error('Bad magic number.'));
continue;
}
this.total += data.length;
this.pending.push(data);
while (this.total >= this.waiting) {
if (this.waiting === 0) {
this.parse(new Buffer(0));
continue;
}
header = this.header;
this.pending.length = 0;
this.total = 0;
this.waiting = 12;
this.header = null;
try {
body = this.parseBody(chunk);
} catch (e) {
this.emit('error', e);
if (this.pending[0].length > this.waiting) {
chunk = this.pending[0].slice(0, this.waiting);
this.pending[0] = this.pending[0].slice(this.waiting);
this.total -= chunk.length;
this.parse(chunk);
continue;
}
this.emit('packet', header.job, body);
if (this.pending[0].length === this.waiting) {
chunk = this.pending.shift();
this.total -= chunk.length;
this.parse(chunk);
continue;
}
chunk = new Buffer(this.waiting);
off = 0;
len = 0;
while (off < chunk.length) {
len = this.pending[0].copy(chunk, off);
if (len === this.pending[0].length)
this.pending.shift();
else
this.pending[0] = this.pending[0].slice(len);
off += len;
}
assert.equal(off, chunk.length);
this.total -= chunk.length;
this.parse(chunk);
}
};
Parser.prototype.parse = function parse(data) {
var header, body;
if (!this.header) {
header = this.parseHeader(data);
if (header.magic !== 0xdeadbeef) {
this.emit('error', new Error('Bad magic number.'));
return;
}
this.header = header;
this.waiting = header.size;
return;
}
header = this.header;
this.waiting = 12;
this.header = null;
try {
body = this.parseBody(data);
} catch (e) {
this.emit('error', e);
return;
}
this.emit('packet', header.job, body);
};
Parser.prototype.parseHeader = function parseHeader(data) {
return {
magic: data.readUInt32LE(0, true),
@ -1107,12 +1133,6 @@ function fromError(err) {
return [err.message, err.stack + '', err.type];
}
function concat(buffers) {
return buffers.length > 1
? Buffer.concat(buffers)
: buffers[0];
}
/*
* Expose
*/