| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421 | // Copyright 2008 The Closure Library Authors. All Rights Reserved.//// Licensed under the Apache License, Version 2.0 (the "License");// you may not use this file except in compliance with the License.// You may obtain a copy of the License at////      http://www.apache.org/licenses/LICENSE-2.0//// Unless required by applicable law or agreed to in writing, software// distributed under the License is distributed on an "AS-IS" BASIS,// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.// See the License for the specific language governing permissions and// limitations under the License./** * @fileoverview Performance timer. * * {@see goog.testing.benchmark} for an easy way to use this functionality. * * @author attila@google.com (Attila Bodis) */goog.setTestOnly('goog.testing.PerformanceTimer');goog.provide('goog.testing.PerformanceTimer');goog.provide('goog.testing.PerformanceTimer.Task');goog.require('goog.array');goog.require('goog.async.Deferred');goog.require('goog.math');/** * Creates a performance timer that runs test functions a number of times to * generate timing samples, and provides performance statistics (minimum, * maximum, average, and standard deviation). * @param {number=} opt_numSamples Number of times to run the test function; *     defaults to 10. * @param {number=} opt_timeoutInterval Number of milliseconds after which the *     test is to be aborted; defaults to 5 seconds (5,000ms). * @constructor */goog.testing.PerformanceTimer = function(opt_numSamples, opt_timeoutInterval) {  /**   * Number of times the test function is to be run; defaults to 10.   * @private {number}   */  this.numSamples_ = opt_numSamples || 10;  /**   * Number of milliseconds after which the test is to be aborted; defaults to   * 5,000ms.   * @private {number}   */  this.timeoutInterval_ = opt_timeoutInterval || 5000;  /**   * Whether to discard outliers (i.e. the smallest and the largest values)   * from the sample set before computing statistics.  Defaults to false.   * @private {boolean}   */  this.discardOutliers_ = false;};/** * A function whose subsequent calls differ in milliseconds. Used to calculate * the start and stop checkpoint times for runs. Note that high performance * timers do not necessarily return the current time in milliseconds. * @return {number} * @private */goog.testing.PerformanceTimer.now_ = function() {  // goog.now is used in DEBUG mode to make the class easier to test.  return !goog.DEBUG && window.performance && window.performance.now ?      window.performance.now() :      goog.now();};/** * @return {number} The number of times the test function will be run. */goog.testing.PerformanceTimer.prototype.getNumSamples = function() {  return this.numSamples_;};/** * Sets the number of times the test function will be run. * @param {number} numSamples Number of times to run the test function. */goog.testing.PerformanceTimer.prototype.setNumSamples = function(numSamples) {  this.numSamples_ = numSamples;};/** * @return {number} The number of milliseconds after which the test times out. */goog.testing.PerformanceTimer.prototype.getTimeoutInterval = function() {  return this.timeoutInterval_;};/** * Sets the number of milliseconds after which the test times out. * @param {number} timeoutInterval Timeout interval in ms. */goog.testing.PerformanceTimer.prototype.setTimeoutInterval = function(    timeoutInterval) {  this.timeoutInterval_ = timeoutInterval;};/** * Sets whether to ignore the smallest and the largest values when computing * stats. * @param {boolean} discard Whether to discard outlier values. */goog.testing.PerformanceTimer.prototype.setDiscardOutliers = function(discard) {  this.discardOutliers_ = discard;};/** * @return {boolean} Whether outlier values are discarded prior to computing *     stats. */goog.testing.PerformanceTimer.prototype.isDiscardOutliers = function() {  return this.discardOutliers_;};/** * Executes the test function the required number of times (or until the * test run exceeds the timeout interval, whichever comes first).  Returns * an object containing the following: * <pre> *   { *     'average': average execution time (ms) *     'count': number of executions (may be fewer than expected due to timeout) *     'maximum': longest execution time (ms) *     'minimum': shortest execution time (ms) *     'standardDeviation': sample standard deviation (ms) *     'total': total execution time (ms) *   } * </pre> * * @param {Function} testFn Test function whose performance is to *     be measured. * @return {!Object} Object containing performance stats. */goog.testing.PerformanceTimer.prototype.run = function(testFn) {  return this.runTask(      new goog.testing.PerformanceTimer.Task(          /** @type {goog.testing.PerformanceTimer.TestFunction} */ (testFn)));};/** * Executes the test function of the specified task as described in * {@code run}. In addition, if specified, the set up and tear down functions of * the task are invoked before and after each invocation of the test function. * @see goog.testing.PerformanceTimer#run * @param {goog.testing.PerformanceTimer.Task} task A task describing the test *     function to invoke. * @return {!Object} Object containing performance stats. */goog.testing.PerformanceTimer.prototype.runTask = function(task) {  var samples = [];  var testStart = goog.testing.PerformanceTimer.now_();  var totalRunTime = 0;  var testFn = task.getTest();  var setUpFn = task.getSetUp();  var tearDownFn = task.getTearDown();  for (var i = 0; i < this.numSamples_ && totalRunTime <= this.timeoutInterval_;       i++) {    setUpFn();    var sampleStart = goog.testing.PerformanceTimer.now_();    testFn();    var sampleEnd = goog.testing.PerformanceTimer.now_();    tearDownFn();    samples[i] = sampleEnd - sampleStart;    totalRunTime = sampleEnd - testStart;  }  return this.finishTask_(samples);};/** * Finishes the run of a task by creating a result object from samples, in the * format described in {@code run}. * @see goog.testing.PerformanceTimer#run * @param {!Array<number>} samples The samples to analyze. * @return {!Object} Object containing performance stats. * @private */goog.testing.PerformanceTimer.prototype.finishTask_ = function(samples) {  if (this.discardOutliers_ && samples.length > 2) {    goog.array.remove(samples, Math.min.apply(null, samples));    goog.array.remove(samples, Math.max.apply(null, samples));  }  return goog.testing.PerformanceTimer.createResults(samples);};/** * Executes the test function of the specified task asynchronously. The test * function is expected to take a callback as input and has to call it to signal * that it's done. In addition, if specified, the setUp and tearDown functions * of the task are invoked before and after each invocation of the test * function. Note that setUp/tearDown functions take a callback as input and * must call this callback when they are done. * @see goog.testing.PerformanceTimer#run * @param {goog.testing.PerformanceTimer.Task} task A task describing the test *     function to invoke. * @return {!goog.async.Deferred} The deferred result, eventually an object *     containing performance stats. */goog.testing.PerformanceTimer.prototype.runAsyncTask = function(task) {  var samples = [];  var testStart = goog.testing.PerformanceTimer.now_();  var testFn = task.getTest();  var setUpFn = task.getSetUp();  var tearDownFn = task.getTearDown();  // Note that this uses a separate code path from runTask() because  // implementing runTask() in terms of runAsyncTask() could easily cause  // a stack overflow if there are many iterations.  var result = new goog.async.Deferred();  this.runAsyncTaskSample_(      testFn, setUpFn, tearDownFn, result, samples, testStart);  return result;};/** * Runs a task once, waits for the test function to complete asynchronously * and starts another run if not enough samples have been collected. Otherwise * finishes this task. * @param {goog.testing.PerformanceTimer.TestFunction} testFn The test function. * @param {goog.testing.PerformanceTimer.TestFunction} setUpFn The set up *     function that will be called once before the test function is run. * @param {goog.testing.PerformanceTimer.TestFunction} tearDownFn The set up *     function that will be called once after the test function completed. * @param {!goog.async.Deferred} result The deferred result, eventually an *     object containing performance stats. * @param {!Array<number>} samples The time samples from all runs of the test *     function so far. * @param {number} testStart The timestamp when the first sample was started. * @private */goog.testing.PerformanceTimer.prototype.runAsyncTaskSample_ = function(    testFn, setUpFn, tearDownFn, result, samples, testStart) {  var timer = this;  timer.handleOptionalDeferred_(setUpFn, function() {    var sampleStart = goog.testing.PerformanceTimer.now_();    timer.handleOptionalDeferred_(testFn, function() {      var sampleEnd = goog.testing.PerformanceTimer.now_();      timer.handleOptionalDeferred_(tearDownFn, function() {        samples.push(sampleEnd - sampleStart);        var totalRunTime = sampleEnd - testStart;        if (samples.length < timer.numSamples_ &&            totalRunTime <= timer.timeoutInterval_) {          timer.runAsyncTaskSample_(              testFn, setUpFn, tearDownFn, result, samples, testStart);        } else {          result.callback(timer.finishTask_(samples));        }      });    });  });};/** * Execute a function that optionally returns a deferred object and continue * with the given continuation function only once the deferred object has a * result. * @param {goog.testing.PerformanceTimer.TestFunction} deferredFactory The *     function that optionally returns a deferred object. * @param {function()} continuationFunction The function that should be called *     after the optional deferred has a result. * @private */goog.testing.PerformanceTimer.prototype.handleOptionalDeferred_ = function(    deferredFactory, continuationFunction) {  var deferred = deferredFactory();  if (deferred) {    deferred.addCallback(continuationFunction);  } else {    continuationFunction();  }};/** * Creates a performance timer results object by analyzing a given array of * sample timings. * @param {!Array<number>} samples The samples to analyze. * @return {!Object} Object containing performance stats. */goog.testing.PerformanceTimer.createResults = function(samples) {  return {    'average': goog.math.average.apply(null, samples),    'count': samples.length,    'maximum': Math.max.apply(null, samples),    'minimum': Math.min.apply(null, samples),    'standardDeviation': goog.math.standardDeviation.apply(null, samples),    'total': goog.math.sum.apply(null, samples)  };};/** * A test function whose performance should be measured or a setUp/tearDown * function. It may optionally return a deferred object. If it does so, the * test harness will assume the function is asynchronous and it must signal * that it's done by setting an (empty) result on the deferred object. If the * function doesn't return anything, the test harness will assume it's * synchronous. * @typedef {function():(goog.async.Deferred|undefined)} */goog.testing.PerformanceTimer.TestFunction;/** * A task for the performance timer to measure. Callers can specify optional * setUp and tearDown methods to control state before and after each run of the * test function. * @param {goog.testing.PerformanceTimer.TestFunction} test Test function whose *     performance is to be measured. * @constructor * @final */goog.testing.PerformanceTimer.Task = function(test) {  /**   * The test function to time.   * @type {goog.testing.PerformanceTimer.TestFunction}   * @private   */  this.test_ = test;};/** * An optional set up function to run before each invocation of the test * function. * @type {goog.testing.PerformanceTimer.TestFunction} * @private */goog.testing.PerformanceTimer.Task.prototype.setUp_ = goog.nullFunction;/** * An optional tear down function to run after each invocation of the test * function. * @type {goog.testing.PerformanceTimer.TestFunction} * @private */goog.testing.PerformanceTimer.Task.prototype.tearDown_ = goog.nullFunction;/** * @return {goog.testing.PerformanceTimer.TestFunction} The test function to *     time. */goog.testing.PerformanceTimer.Task.prototype.getTest = function() {  return this.test_;};/** * Specifies a set up function to be invoked before each invocation of the test * function. * @param {goog.testing.PerformanceTimer.TestFunction} setUp The set up *     function. * @return {!goog.testing.PerformanceTimer.Task} This task. */goog.testing.PerformanceTimer.Task.prototype.withSetUp = function(setUp) {  this.setUp_ = setUp;  return this;};/** * @return {goog.testing.PerformanceTimer.TestFunction} The set up function or *     the default no-op function if none was specified. */goog.testing.PerformanceTimer.Task.prototype.getSetUp = function() {  return this.setUp_;};/** * Specifies a tear down function to be invoked after each invocation of the * test function. * @param {goog.testing.PerformanceTimer.TestFunction} tearDown The tear down *     function. * @return {!goog.testing.PerformanceTimer.Task} This task. */goog.testing.PerformanceTimer.Task.prototype.withTearDown = function(tearDown) {  this.tearDown_ = tearDown;  return this;};/** * @return {goog.testing.PerformanceTimer.TestFunction} The tear down function *     or the default no-op function if none was specified. */goog.testing.PerformanceTimer.Task.prototype.getTearDown = function() {  return this.tearDown_;};
 |