diff --git a/lib/buffer.js b/lib/buffer.js index 91fd67706c4..25dfaab042c 100644 --- a/lib/buffer.js +++ b/lib/buffer.js @@ -196,6 +196,15 @@ SlowBuffer.prototype.slice = function(start, end) { }; +function coerce(length) { + // Coerce length to a number (possibly NaN), round up + // in case it's fractional (e.g. 123.456) then do a + // double negate to coerce a NaN to 0. Easy, right? + length = ~~Math.ceil(+length); + return length < 0 ? 0 : length; +} + + // Buffer function Buffer(subject, encoding, offset) { @@ -207,14 +216,14 @@ function Buffer(subject, encoding, offset) { // Are we slicing? if (typeof offset === 'number') { - this.length = encoding; + this.length = coerce(encoding); this.parent = subject; this.offset = offset; } else { // Find the length switch (type = typeof subject) { case 'number': - this.length = subject; + this.length = coerce(subject); break; case 'string': @@ -222,7 +231,7 @@ function Buffer(subject, encoding, offset) { break; case 'object': // Assume object is an array - this.length = subject.length; + this.length = coerce(subject.length); break; default: diff --git a/test/simple/test-buffer.js b/test/simple/test-buffer.js index 4152c28b569..fdd3a365e2b 100644 --- a/test/simple/test-buffer.js +++ b/test/simple/test-buffer.js @@ -689,3 +689,16 @@ buf.write('123456', 'base64'); assert.equal(Buffer._charsWritten, 6); buf.write('00010203040506070809', 'hex'); assert.equal(Buffer._charsWritten, 18); + +// Check for fractional length args, junk length args, etc. +// https://github.com/joyent/node/issues/1758 +Buffer(3.3).toString(); // throws bad argument error in commit 43cb4ec +assert.equal(Buffer(-1).length, 0); +assert.equal(Buffer(NaN).length, 0); +assert.equal(Buffer(3.3).length, 4); +assert.equal(Buffer({length:3.3}).length, 4); +assert.equal(Buffer({length:"BAM"}).length, 0); + +// Make sure that strings are not coerced to numbers. +assert.equal(Buffer("99").length, 2); +assert.equal(Buffer("13.37").length, 5); \ No newline at end of file