From 3cbb5cdfdb621baec5dc3a2ac505be37f1718086 Mon Sep 17 00:00:00 2001 From: cjihrig Date: Thu, 22 Jan 2015 20:16:36 -0500 Subject: [PATCH] console: allow Object.prototype fields as labels Console.prototype.timeEnd() returns NaN if the timer label corresponds to a property on Object.prototype. This commit uses a Map to construct the _times object. Fixes: https://github.com/joyent/node/issues/9069 PR-URL: https://github.com/iojs/io.js/pull/563 Reviewed-By: Vladimir Kurchatkin Reviewed-By: Chris Dickinson Reviewed-By: Rod Vagg --- lib/console.js | 6 +++--- test/parallel/test-console.js | 34 +++++++++++++++++++++++++--------- 2 files changed, 28 insertions(+), 12 deletions(-) diff --git a/lib/console.js b/lib/console.js index 4163cfee621..323d23374a8 100644 --- a/lib/console.js +++ b/lib/console.js @@ -21,7 +21,7 @@ function Console(stdout, stderr) { Object.defineProperty(this, '_stdout', prop); prop.value = stderr; Object.defineProperty(this, '_stderr', prop); - prop.value = {}; + prop.value = new Map(); Object.defineProperty(this, '_times', prop); // bind the prototype functions to this Console instance @@ -56,12 +56,12 @@ Console.prototype.dir = function(object, options) { Console.prototype.time = function(label) { - this._times[label] = Date.now(); + this._times.set(label, Date.now()); }; Console.prototype.timeEnd = function(label) { - var time = this._times[label]; + var time = this._times.get(label); if (!time) { throw new Error('No such label: ' + label); } diff --git a/test/parallel/test-console.js b/test/parallel/test-console.js index 202ec6eab52..2e5060f3831 100644 --- a/test/parallel/test-console.js +++ b/test/parallel/test-console.js @@ -7,6 +7,15 @@ assert.ok(process.stderr.writable); assert.equal('number', typeof process.stdout.fd); assert.equal('number', typeof process.stderr.fd); +assert.throws(function () { + console.timeEnd('no such label'); +}); + +assert.doesNotThrow(function () { + console.time('label'); + console.timeEnd('label'); +}); + // an Object with a custom .inspect() function var custom_inspect = { foo: 'bar', inspect: function () { return 'inspect'; } }; @@ -33,6 +42,17 @@ console.dir({ foo : { bar : { baz : true } } }, { depth: 1 }); // test console.trace() console.trace('This is a %j %d', { formatted: 'trace' }, 10, 'foo'); +// test console.time() and console.timeEnd() output +console.time('label'); +console.timeEnd('label'); + +// verify that Object.prototype properties can be used as labels +console.time('__proto__'); +console.timeEnd('__proto__'); +console.time('constructor'); +console.timeEnd('constructor'); +console.time('hasOwnProperty'); +console.timeEnd('hasOwnProperty'); global.process.stdout.write = stdout_write; @@ -47,12 +67,8 @@ assert.notEqual(-1, strings.shift().indexOf('foo: [Object]')); assert.equal(-1, strings.shift().indexOf('baz')); assert.equal('Trace: This is a {"formatted":"trace"} 10 foo', strings.shift().split('\n').shift()); - -assert.throws(function () { - console.timeEnd('no such label'); -}); - -assert.doesNotThrow(function () { - console.time('label'); - console.timeEnd('label'); -}); +assert.ok(/^label: \d+ms$/.test(strings.shift().trim())); +assert.ok(/^__proto__: \d+ms$/.test(strings.shift().trim())); +assert.ok(/^constructor: \d+ms$/.test(strings.shift().trim())); +assert.ok(/^hasOwnProperty: \d+ms$/.test(strings.shift().trim())); +assert.equal(strings.length, 0);