1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969 |
- // Copyright 2007 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 A class representing a set of test functions to be run.
- *
- * Testing code should not have dependencies outside of goog.testing so as to
- * reduce the chance of masking missing dependencies.
- *
- * This file does not compile correctly with --collapse_properties. Use
- * --property_renaming=ALL_UNQUOTED instead.
- *
- */
- goog.setTestOnly('goog.testing.TestCase');
- goog.provide('goog.testing.TestCase');
- goog.provide('goog.testing.TestCase.Error');
- goog.provide('goog.testing.TestCase.Order');
- goog.provide('goog.testing.TestCase.Result');
- goog.provide('goog.testing.TestCase.Test');
- goog.require('goog.Promise');
- goog.require('goog.Thenable');
- goog.require('goog.array');
- goog.require('goog.asserts');
- goog.require('goog.dom');
- goog.require('goog.dom.TagName');
- goog.require('goog.json');
- goog.require('goog.object');
- goog.require('goog.testing.JsUnitException');
- goog.require('goog.testing.asserts');
- /**
- * A class representing a JsUnit test case. A TestCase is made up of a number
- * of test functions which can be run. Individual test cases can override the
- * following functions to set up their test environment:
- * - runTests - completely override the test's runner
- * - setUpPage - called before any of the test functions are run
- * - tearDownPage - called after all tests are finished
- * - setUp - called before each of the test functions
- * - tearDown - called after each of the test functions
- * - shouldRunTests - called before a test run, all tests are skipped if it
- * returns false. Can be used to disable tests on browsers
- * where they aren't expected to pass.
- * <p>
- * TestCase objects are usually constructed by inspecting the global environment
- * to discover functions that begin with the prefix <code>test</code>.
- * (See {@link #autoDiscoverLifecycle} and {@link #autoDiscoverTests}.)
- * </p>
- *
- * <h2>Testing asychronous code with promises</h2>
- *
- * <p>
- * In the simplest cases, the behavior that the developer wants to test
- * is synchronous, and the test functions exercising the behavior execute
- * synchronously. But TestCase can also be used to exercise asynchronous code
- * through the use of <a
- * href="https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise">
- * promises</a>. If a test function returns an object that has a
- * <code>then</code> method defined on it, the test framework switches to an
- * asynchronous execution strategy: the next test function will not begin
- * execution until the returned promise is resolved or rejected. Instead of
- * writing test assertions at the top level inside a test function, the test
- * author chains them on the end of the returned promise. For example:
- * </p>
- * <pre>
- * function testPromiseBasedAPI() {
- * return promiseBasedAPI().then(function(value) {
- * // Will run when the promise resolves, and before the next
- * // test function begins execution.
- * assertEquals('foo', value.bar);
- * });
- * }
- * </pre>
- * <p>
- * Synchronous and asynchronous tests can be mixed in the same TestCase.
- * Test functions that return an object with a <code>then</code> method are
- * executed asynchronously, and all other test functions are executed
- * synchronously. While this is convenient for test authors (since it doesn't
- * require any explicit configuration for asynchronous tests), it can lead to
- * confusion if the test author forgets to return the promise from the test
- * function. For example:
- * </p>
- * <pre>
- * function testPromiseBasedAPI() {
- * // This test should never succeed.
- * promiseBasedAPI().then(fail, fail);
- * // Oops! The promise isn't returned to the framework,
- * // so this test actually does succeed.
- * }
- * </pre>
- * <p>
- * Since the test framework knows nothing about the promise created
- * in the test function, it will run the function synchronously, record
- * a success, and proceed immediately to the next test function.
- * </p>
- * <p>
- * Promises returned from test functions can time out. If a returned promise
- * is not resolved or rejected within {@link promiseTimeout} milliseconds,
- * the test framework rejects the promise without a timeout error message.
- * Test cases can configure the value of {@code promiseTimeout} by setting
- * <pre>
- * goog.testing.TestCase.getActiveTestCase().promiseTimeout = ...
- * </pre>
- * in their {@code setUpPage} methods.
- * </p>
- *
- * @param {string=} opt_name The name of the test case, defaults to
- * 'Untitled Test Case'.
- * @constructor
- */
- goog.testing.TestCase = function(opt_name) {
- /**
- * A name for the test case.
- * @type {string}
- * @private
- */
- this.name_ = opt_name || 'Untitled Test Case';
- /**
- * Array of test functions that can be executed.
- * @type {!Array<!goog.testing.TestCase.Test>}
- * @private
- */
- this.tests_ = [];
- /**
- * Set of test names and/or indices to execute, or null if all tests should
- * be executed.
- *
- * Indices are included to allow automation tools to run a subset of the
- * tests without knowing the exact contents of the test file.
- *
- * Indices should only be used with SORTED ordering.
- *
- * Example valid values:
- * <ul>
- * <li>[testName]
- * <li>[testName1, testName2]
- * <li>[2] - will run the 3rd test in the order specified
- * <li>[1,3,5]
- * <li>[testName1, testName2, 3, 5] - will work
- * <ul>
- * @type {Object}
- * @private
- */
- this.testsToRun_ = null;
- /**
- * A call back for each test.
- * @private {?function(goog.testing.TestCase.Test, !Array<string>)}
- */
- this.testDone_ = null;
- /**
- * The order to run the auto-discovered tests in.
- * @type {string}
- */
- this.order = goog.testing.TestCase.Order.SORTED;
- /** @private {function(!goog.testing.TestCase.Result)} */
- this.runNextTestCallback_ = goog.nullFunction;
- /** @private {goog.testing.TestCase.Test} */
- this.curTest_ = null;
- /**
- * Object used to encapsulate the test results.
- * @type {!goog.testing.TestCase.Result}
- * @protected
- * @suppress {underscore|visibility}
- */
- this.result_ = new goog.testing.TestCase.Result(this);
- /**
- * An array of exceptions generated by {@code assert} statements.
- * @private {!Array<!goog.testing.JsUnitException>}
- */
- this.thrownAssertionExceptions_ = [];
- /**
- * Whether the test should fail if exceptions arising from an assert statement
- * never bubbled up to the testing framework.
- * @type {boolean}
- */
- this.failOnUnreportedAsserts = true;
- /**
- * The maximum time in milliseconds a promise returned from a test function
- * may remain pending before the test fails due to timeout.
- * @type {number}
- */
- this.promiseTimeout = 1000; // 1s
- };
- /**
- * The order to run the auto-discovered tests.
- * @enum {string}
- */
- goog.testing.TestCase.Order = {
- /**
- * This is browser dependent and known to be different in FF and Safari
- * compared to others.
- */
- NATURAL: 'natural',
- /** Random order. */
- RANDOM: 'random',
- /** Sorted based on the name. */
- SORTED: 'sorted'
- };
- /**
- * @return {string} The name of the test.
- */
- goog.testing.TestCase.prototype.getName = function() {
- return this.name_;
- };
- /**
- * The maximum amount of time in milliseconds that the test case can take
- * before it is forced to yield and reschedule. This prevents the test runner
- * from blocking the browser and potentially hurting the test harness.
- * @type {number}
- */
- goog.testing.TestCase.maxRunTime = 200;
- /**
- * Save a reference to {@code window.setTimeout}, so any code that overrides the
- * default behavior (the MockClock, for example) doesn't affect our runner.
- * @type {function((Function|string), number=, *=): number}
- * @private
- */
- goog.testing.TestCase.protectedSetTimeout_ = goog.global.setTimeout;
- /**
- * Save a reference to {@code window.clearTimeout}, so any code that overrides
- * the default behavior (e.g. MockClock) doesn't affect our runner.
- * @type {function((null|number|undefined)): void}
- * @private
- */
- goog.testing.TestCase.protectedClearTimeout_ = goog.global.clearTimeout;
- /**
- * Save a reference to {@code window.Date}, so any code that overrides
- * the default behavior doesn't affect our runner.
- * @type {function(new: Date)}
- * @private
- */
- goog.testing.TestCase.protectedDate_ = Date;
- /**
- * Save a reference to {@code window.performance}, so any code that overrides
- * the default behavior doesn't affect our runner.
- * @type {?Performance}
- * @private
- */
- goog.testing.TestCase.protectedPerformance_ =
- window.performance && window.performance.now ? performance : null;
- /**
- * TODO(user) replace this with prototype.currentTest.
- * Name of the current test that is running, or null if none is running.
- * @type {?string}
- */
- goog.testing.TestCase.currentTestName = null;
- /**
- * Avoid a dependency on goog.userAgent and keep our own reference of whether
- * the browser is IE.
- * @type {boolean}
- */
- goog.testing.TestCase.IS_IE = typeof opera == 'undefined' &&
- !!goog.global.navigator &&
- goog.global.navigator.userAgent.indexOf('MSIE') != -1;
- /**
- * Exception object that was detected before a test runs.
- * @type {*}
- * @protected
- */
- goog.testing.TestCase.prototype.exceptionBeforeTest;
- /**
- * Whether the test case has ever tried to execute.
- * @type {boolean}
- */
- goog.testing.TestCase.prototype.started = false;
- /**
- * Whether the test case is running.
- * @type {boolean}
- */
- goog.testing.TestCase.prototype.running = false;
- /**
- * Timestamp for when the test was started.
- * @type {number}
- * @private
- */
- goog.testing.TestCase.prototype.startTime_ = 0;
- /**
- * Time since the last batch of tests was started, if batchTime exceeds
- * {@link #maxRunTime} a timeout will be used to stop the tests blocking the
- * browser and a new batch will be started.
- * @type {number}
- * @private
- */
- goog.testing.TestCase.prototype.batchTime_ = 0;
- /**
- * Pointer to the current test.
- * @type {number}
- * @private
- */
- goog.testing.TestCase.prototype.currentTestPointer_ = 0;
- /**
- * Optional callback that will be executed when the test has finalized.
- * @type {Function}
- * @private
- */
- goog.testing.TestCase.prototype.onCompleteCallback_ = null;
- /**
- * Adds a new test to the test case.
- * @param {!goog.testing.TestCase.Test} test The test to add.
- */
- goog.testing.TestCase.prototype.add = function(test) {
- goog.asserts.assert(test);
- if (this.started) {
- throw Error(
- 'Tests cannot be added after execute() has been called. ' +
- 'Test: ' + test.name);
- }
- this.tests_.push(test);
- };
- /**
- * Creates and adds a new test.
- *
- * Convenience function to make syntax less awkward when not using automatic
- * test discovery.
- *
- * @param {string} name The test name.
- * @param {!Function} ref Reference to the test function.
- * @param {!Object=} opt_scope Optional scope that the test function should be
- * called in.
- */
- goog.testing.TestCase.prototype.addNewTest = function(name, ref, opt_scope) {
- var test = new goog.testing.TestCase.Test(name, ref, opt_scope || this);
- this.add(test);
- };
- /**
- * Sets the tests.
- * @param {!Array<goog.testing.TestCase.Test>} tests A new test array.
- * @protected
- */
- goog.testing.TestCase.prototype.setTests = function(tests) {
- this.tests_ = tests;
- };
- /**
- * Gets the tests.
- * @return {!Array<goog.testing.TestCase.Test>} The test array.
- */
- goog.testing.TestCase.prototype.getTests = function() {
- return this.tests_;
- };
- /**
- * Returns the number of tests contained in the test case.
- * @return {number} The number of tests.
- */
- goog.testing.TestCase.prototype.getCount = function() {
- return this.tests_.length;
- };
- /**
- * Returns the number of tests actually run in the test case, i.e. subtracting
- * any which are skipped.
- * @return {number} The number of un-ignored tests.
- */
- goog.testing.TestCase.prototype.getActuallyRunCount = function() {
- return this.testsToRun_ ? goog.object.getCount(this.testsToRun_) : 0;
- };
- /**
- * Returns the current test and increments the pointer.
- * @return {goog.testing.TestCase.Test} The current test case.
- */
- goog.testing.TestCase.prototype.next = function() {
- var test;
- while ((test = this.tests_[this.currentTestPointer_++])) {
- if (!this.testsToRun_ || this.testsToRun_[test.name] ||
- this.testsToRun_[this.currentTestPointer_ - 1]) {
- return test;
- }
- }
- return null;
- };
- /**
- * Resets the test case pointer, so that next returns the first test.
- */
- goog.testing.TestCase.prototype.reset = function() {
- this.currentTestPointer_ = 0;
- this.result_ = new goog.testing.TestCase.Result(this);
- };
- /**
- * Sets the callback function that should be executed when the tests have
- * completed.
- * @param {Function} fn The callback function.
- */
- goog.testing.TestCase.prototype.setCompletedCallback = function(fn) {
- this.onCompleteCallback_ = fn;
- };
- /**
- * @param {goog.testing.TestCase.Order} order The sort order for running tests.
- */
- goog.testing.TestCase.prototype.setOrder = function(order) {
- this.order = order;
- };
- /**
- * @param {Object<string, boolean>} testsToRun Set of tests to run. Entries in
- * the set may be test names, like "testFoo", or numeric indicies. Only
- * tests identified by name or by index will be executed.
- */
- goog.testing.TestCase.prototype.setTestsToRun = function(testsToRun) {
- this.testsToRun_ = testsToRun;
- };
- /**
- * Can be overridden in test classes to indicate whether the tests in a case
- * should be run in that particular situation. For example, this could be used
- * to stop tests running in a particular browser, where browser support for
- * the class under test was absent.
- * @return {boolean} Whether any of the tests in the case should be run.
- */
- goog.testing.TestCase.prototype.shouldRunTests = function() {
- return true;
- };
- /**
- * Executes the tests, yielding asynchronously if execution time exceeds
- * {@link maxRunTime}. There is no guarantee that the test case has finished
- * once this method has returned. To be notified when the test case
- * has finished, use {@link #setCompletedCallback} or
- * {@link #runTestsReturningPromise}.
- */
- goog.testing.TestCase.prototype.execute = function() {
- if (!this.prepareForRun_()) {
- return;
- }
- this.log('Starting tests: ' + this.name_);
- this.cycleTests();
- };
- /**
- * Sets up the internal state of the test case for a run.
- * @return {boolean} If false, preparation failed because the test case
- * is not supposed to run in the present environment.
- * @private
- */
- goog.testing.TestCase.prototype.prepareForRun_ = function() {
- this.started = true;
- this.reset();
- this.startTime_ = this.now();
- this.running = true;
- this.result_.totalCount = this.getCount();
- if (!this.shouldRunTests()) {
- this.log('shouldRunTests() returned false, skipping these tests.');
- this.result_.testSuppressed = true;
- this.finalize();
- return false;
- }
- return true;
- };
- /**
- * Finalizes the test case, called when the tests have finished executing.
- */
- goog.testing.TestCase.prototype.finalize = function() {
- this.saveMessage('Done');
- this.tearDownPage();
- this.endTime_ = this.now();
- this.running = false;
- this.result_.runTime = this.endTime_ - this.startTime_;
- this.result_.numFilesLoaded = this.countNumFilesLoaded_();
- this.result_.complete = true;
- this.log(this.result_.getSummary());
- if (this.result_.isSuccess()) {
- this.log('Tests complete');
- } else {
- this.log('Tests Failed');
- }
- if (this.onCompleteCallback_) {
- var fn = this.onCompleteCallback_;
- // Execute's the completed callback in the context of the global object.
- fn();
- this.onCompleteCallback_ = null;
- }
- };
- /**
- * Saves a message to the result set.
- * @param {string} message The message to save.
- */
- goog.testing.TestCase.prototype.saveMessage = function(message) {
- this.result_.messages.push(this.getTimeStamp_() + ' ' + message);
- };
- /**
- * @return {boolean} Whether the test case is running inside the multi test
- * runner.
- */
- goog.testing.TestCase.prototype.isInsideMultiTestRunner = function() {
- var top = goog.global['top'];
- return top && typeof top['_allTests'] != 'undefined';
- };
- /**
- * Logs an object to the console, if available.
- * @param {*} val The value to log. Will be ToString'd.
- */
- goog.testing.TestCase.prototype.log = function(val) {
- if (!this.isInsideMultiTestRunner() && goog.global.console) {
- if (typeof val == 'string') {
- val = this.getTimeStamp_() + ' : ' + val;
- }
- if (val instanceof Error && val.stack) {
- // Chrome does console.log asynchronously in a different process
- // (http://code.google.com/p/chromium/issues/detail?id=50316).
- // This is an acute problem for Errors, which almost never survive.
- // Grab references to the immutable strings so they survive.
- goog.global.console.log(val, val.message, val.stack);
- // TODO(gboyer): Consider for Chrome cloning any object if we can ensure
- // there are no circular references.
- } else {
- goog.global.console.log(val);
- }
- }
- };
- /**
- * @return {boolean} Whether the test was a success.
- */
- goog.testing.TestCase.prototype.isSuccess = function() {
- return !!this.result_ && this.result_.isSuccess();
- };
- /**
- * Returns a string detailing the results from the test.
- * @param {boolean=} opt_verbose If true results will include data about all
- * tests, not just what failed.
- * @return {string} The results from the test.
- */
- goog.testing.TestCase.prototype.getReport = function(opt_verbose) {
- var rv = [];
- if (this.running) {
- rv.push(this.name_ + ' [RUNNING]');
- } else if (this.result_.runCount == 0) {
- rv.push(this.name_ + ' [NO TESTS RUN]');
- } else {
- var label = this.result_.isSuccess() ? 'PASSED' : 'FAILED';
- rv.push(this.name_ + ' [' + label + ']');
- }
- if (goog.global.location) {
- rv.push(this.trimPath_(goog.global.location.href));
- }
- rv.push(this.result_.getSummary());
- if (opt_verbose) {
- rv.push('.', this.result_.messages.join('\n'));
- } else if (!this.result_.isSuccess()) {
- rv.push(this.result_.errors.join('\n'));
- }
- rv.push(' ');
- return rv.join('\n');
- };
- /**
- * Returns the test results.
- * @return {!goog.testing.TestCase.Result}
- * @package
- */
- goog.testing.TestCase.prototype.getResult = function() {
- return this.result_;
- };
- /**
- * Returns the amount of time it took for the test to run.
- * @return {number} The run time, in milliseconds.
- */
- goog.testing.TestCase.prototype.getRunTime = function() {
- return this.result_.runTime;
- };
- /**
- * Returns the number of script files that were loaded in order to run the test.
- * @return {number} The number of script files.
- */
- goog.testing.TestCase.prototype.getNumFilesLoaded = function() {
- return this.result_.numFilesLoaded;
- };
- /**
- * Represents a test result.
- * @typedef {{
- * 'source': string,
- * 'message': string,
- * 'stacktrace': string
- * }}
- */
- goog.testing.TestCase.IResult;
- /**
- * Returns the test results object: a map from test names to a list of test
- * failures (if any exist).
- * @return {!Object<string, !Array<goog.testing.TestCase.IResult>>} Test
- * results object.
- */
- goog.testing.TestCase.prototype.getTestResults = function() {
- var map = {};
- goog.object.forEach(this.result_.resultsByName, function(resultArray, key) {
- // Make sure we only use properties on the actual map
- if (!Object.prototype.hasOwnProperty.call(
- this.result_.resultsByName, key)) {
- return;
- }
- map[key] = [];
- for (var j = 0; j < resultArray.length; j++) {
- map[key].push(resultArray[j].toObject_());
- }
- }, this);
- return map;
- };
- /**
- * Returns the test results as json.
- * This is called by the testing infrastructure through G_testrunner.
- * @return {string} Tests results object.
- */
- goog.testing.TestCase.prototype.getTestResultsAsJson = function() {
- return goog.json.serialize(this.getTestResults());
- };
- /**
- * Executes each of the tests, yielding asynchronously if execution time
- * exceeds {@link #maxRunTime}. There is no guarantee that the test case
- * has finished execution once this method has returned.
- * To be notified when the test case has finished execution, use
- * {@link #setCompletedCallback} or {@link #runTestsReturningPromise}.
- *
- * Overridable by the individual test case. This allows test cases to defer
- * when the test is actually started. If overridden, finalize must be called
- * by the test to indicate it has finished.
- */
- goog.testing.TestCase.prototype.runTests = function() {
- goog.testing.Continuation_.run(this.runSetUpPage_(this.execute));
- };
- /**
- * Executes each of the tests, returning a promise that resolves with the
- * test results once they are done running.
- * @return {!IThenable<!goog.testing.TestCase.Result>}
- * @final
- * @package
- */
- goog.testing.TestCase.prototype.runTestsReturningPromise = function() {
- return new goog.Promise(function(resolve) {
- goog.testing.Continuation_.run(this.runSetUpPage_(function() {
- if (!this.prepareForRun_()) {
- resolve(this.result_);
- return;
- }
- this.log('Starting tests: ' + this.name_);
- this.saveMessage('Start');
- this.batchTime_ = this.now();
- this.runNextTestCallback_ = resolve;
- goog.testing.Continuation_.run(this.runNextTest_());
- }));
- }, this);
- };
- /**
- * Runs the setUpPage methods.
- * @param {function(this:goog.testing.TestCase)} runTestsFn Callback to invoke
- * after setUpPage has completed.
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.runSetUpPage_ = function(runTestsFn) {
- return this.invokeTestFunction_(this.setUpPage, runTestsFn, function(e) {
- this.exceptionBeforeTest = e;
- runTestsFn.call(this);
- }, 'setUpPage');
- };
- /**
- * Executes the next test method synchronously or with promises, depending on
- * the test method's return value.
- *
- * If the test method returns a promise, the next test method will run once
- * the promise is resolved or rejected. If the test method does not
- * return a promise, it is assumed to be synchronous, and execution proceeds
- * immediately to the next test method. This means that test cases can run
- * partially synchronously and partially asynchronously, depending on
- * the return values of their test methods. In particular, a test case
- * executes synchronously until the first promise is returned from a
- * test method (or until a resource limit is reached; see
- * {@link finishTestInvocation_}).
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.runNextTest_ = function() {
- this.curTest_ = this.next();
- if (!this.curTest_ || !this.running) {
- this.finalize();
- return new goog.testing.Continuation_(
- goog.bind(this.runNextTestCallback_, this, this.result_));
- }
- this.curTest_.started();
- this.result_.runCount++;
- this.log('Running test: ' + this.curTest_.name);
- if (this.maybeFailTestEarly(this.curTest_)) {
- return new goog.testing.Continuation_(
- goog.bind(this.finishTestInvocation_, this));
- }
- goog.testing.TestCase.currentTestName = this.curTest_.name;
- return this.invokeTestFunction_(
- this.setUp, this.safeRunTest_, this.safeTearDown_, 'setUp');
- };
- /**
- * Calls the given test function, handling errors appropriately.
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.safeRunTest_ = function() {
- return this.invokeTestFunction_(
- goog.bind(this.curTest_.ref, this.curTest_.scope), this.safeTearDown_,
- this.safeTearDown_, this.curTest_.name);
- };
- /**
- * Calls {@link tearDown}, handling errors appropriately.
- * @param {*=} opt_error Error associated with the test, if any.
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.safeTearDown_ = function(opt_error) {
- if (arguments.length == 1) {
- this.doError(this.curTest_, opt_error);
- }
- return this.invokeTestFunction_(
- this.tearDown, this.finishTestInvocation_, this.finishTestInvocation_,
- 'tearDown');
- };
- /**
- * Calls the given {@code fn}, then calls either {@code onSuccess} or
- * {@code onFailure}, either synchronously or using promises, depending on
- * {@code fn}'s return value.
- *
- * If {@code fn} throws an exception, {@code onFailure} is called immediately
- * with the exception.
- *
- * If {@code fn} returns a promise, and the promise is eventually resolved,
- * {@code onSuccess} is called with no arguments. If the promise is eventually
- * rejected, {@code onFailure} is called with the rejection reason.
- *
- * Otherwise, if {@code fn} neither returns a promise nor throws an exception,
- * {@code onSuccess} is called immediately with no arguments.
- *
- * {@code fn}, {@code onSuccess}, and {@code onFailure} are all called with
- * the TestCase instance as the method receiver.
- *
- * @param {function()} fn The function to call.
- * @param {function(): (?goog.testing.Continuation_|undefined)} onSuccess
- * @param {function(*): (?goog.testing.Continuation_|undefined)} onFailure
- * @param {string} fnName Name of the function being invoked e.g. 'setUp'.
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.invokeTestFunction_ = function(
- fn, onSuccess, onFailure, fnName) {
- var testCase = this;
- this.thrownAssertionExceptions_ = [];
- try {
- var retval = fn.call(this);
- if (goog.Thenable.isImplementedBy(retval) ||
- goog.isFunction(retval && retval['then'])) {
- // Resolve Thenable into a proper Promise to avoid hard to debug problems.
- var promise = goog.Promise.resolve(retval);
- var self = this;
- promise = this.rejectIfPromiseTimesOut_(
- promise, self.promiseTimeout,
- 'Timed out while waiting for a promise returned from ' + fnName +
- ' to resolve. Set goog.testing.TestCase.getActiveTestCase()' +
- '.promiseTimeout to adjust the timeout.');
- promise.then(
- function() {
- self.resetBatchTimeAfterPromise_();
- if (testCase.thrownAssertionExceptions_.length == 0) {
- goog.testing.Continuation_.run(onSuccess.call(self));
- } else {
- goog.testing.Continuation_.run(onFailure.call(
- self,
- testCase.reportUnpropagatedAssertionExceptions_(fnName)));
- }
- },
- function(e) {
- self.resetBatchTimeAfterPromise_();
- goog.testing.Continuation_.run(onFailure.call(self, e));
- });
- return null;
- } else {
- if (this.thrownAssertionExceptions_.length == 0) {
- return new goog.testing.Continuation_(goog.bind(onSuccess, this));
- } else {
- return new goog.testing.Continuation_(goog.bind(
- onFailure, this,
- this.reportUnpropagatedAssertionExceptions_(fnName)));
- }
- }
- } catch (e) {
- return new goog.testing.Continuation_(goog.bind(onFailure, this, e));
- }
- };
- /**
- * Logs all of the exceptions generated from failing assertions, and returns a
- * generic exception informing the user that one or more exceptions were not
- * propagated, causing the test to erroneously pass.
- * @param {string} testName The test function's name.
- * @return {!goog.testing.JsUnitException}
- * @private
- */
- goog.testing.TestCase.prototype.reportUnpropagatedAssertionExceptions_ =
- function(testName) {
- var numExceptions = this.thrownAssertionExceptions_.length;
- for (var i = 0; i < numExceptions; i++) {
- this.recordError_(testName, this.thrownAssertionExceptions_[i]);
- }
- return new goog.testing.JsUnitException(
- 'One or more assertions were raised but not caught by the testing ' +
- 'framework. These assertions may have been unintentionally captured ' +
- 'by a catch block or a thenCatch resolution of a Promise.');
- };
- /**
- * Resets the batch run timer. This should only be called after resolving a
- * promise since Promise.then() has an implicit yield.
- * @private
- */
- goog.testing.TestCase.prototype.resetBatchTimeAfterPromise_ = function() {
- this.batchTime_ = this.now();
- };
- /**
- * Finishes up bookkeeping for the current test function, and schedules
- * the next test function to run, either immediately or asychronously.
- * @param {*=} opt_error Optional error resulting from the test invocation.
- * @return {?goog.testing.Continuation_}
- * @private
- */
- goog.testing.TestCase.prototype.finishTestInvocation_ = function(opt_error) {
- if (arguments.length == 1) {
- this.doError(this.curTest_, opt_error);
- }
- // If no errors have been recorded for the test, it is a success.
- if (!(this.curTest_.name in this.result_.resultsByName) ||
- !this.result_.resultsByName[this.curTest_.name].length) {
- this.doSuccess(this.curTest_);
- }
- goog.testing.TestCase.currentTestName = null;
- // If the test case has consumed too much time or stack space,
- // yield to avoid blocking the browser. Otherwise, proceed to the next test.
- if (this.now() - this.batchTime_ > goog.testing.TestCase.maxRunTime) {
- this.saveMessage('Breaking async');
- this.timeout(goog.bind(this.startNextBatch_, this), 0);
- return null;
- } else {
- return new goog.testing.Continuation_(goog.bind(this.runNextTest_, this));
- }
- };
- /**
- * Start a new batch to tests after yielding, resetting batchTime and depth.
- * @private
- */
- goog.testing.TestCase.prototype.startNextBatch_ = function() {
- this.batchTime_ = this.now();
- goog.testing.Continuation_.run(this.runNextTest_());
- };
- /**
- * Reorders the tests depending on the {@code order} field.
- * @private
- */
- goog.testing.TestCase.prototype.orderTests_ = function() {
- switch (this.order) {
- case goog.testing.TestCase.Order.RANDOM:
- // Fisher-Yates shuffle
- var i = this.tests_.length;
- while (i > 1) {
- // goog.math.randomInt is inlined to reduce dependencies.
- var j = Math.floor(Math.random() * i); // exclusive
- i--;
- var tmp = this.tests_[i];
- this.tests_[i] = this.tests_[j];
- this.tests_[j] = tmp;
- }
- break;
- case goog.testing.TestCase.Order.SORTED:
- this.tests_.sort(function(t1, t2) {
- if (t1.name == t2.name) {
- return 0;
- }
- return t1.name < t2.name ? -1 : 1;
- });
- break;
- // Do nothing for NATURAL.
- }
- };
- /**
- * Gets list of objects that potentially contain test cases. For IE 8 and below,
- * this is the global "this" (for properties set directly on the global this or
- * window) and the RuntimeObject (for global variables and functions). For all
- * other browsers, the array simply contains the global this.
- *
- * @param {string=} opt_prefix An optional prefix. If specified, only get things
- * under this prefix. Note that the prefix is only honored in IE, since it
- * supports the RuntimeObject:
- * http://msdn.microsoft.com/en-us/library/ff521039%28VS.85%29.aspx
- * TODO: Remove this option.
- * @return {!Array<!Object>} A list of objects that should be inspected.
- */
- goog.testing.TestCase.prototype.getGlobals = function(opt_prefix) {
- return goog.testing.TestCase.getGlobals(opt_prefix);
- };
- /**
- * Gets list of objects that potentially contain test cases. For IE 8 and below,
- * this is the global "this" (for properties set directly on the global this or
- * window) and the RuntimeObject (for global variables and functions). For all
- * other browsers, the array simply contains the global this.
- *
- * @param {string=} opt_prefix An optional prefix. If specified, only get things
- * under this prefix. Note that the prefix is only honored in IE, since it
- * supports the RuntimeObject:
- * http://msdn.microsoft.com/en-us/library/ff521039%28VS.85%29.aspx
- * TODO: Remove this option.
- * @return {!Array<!Object>} A list of objects that should be inspected.
- */
- goog.testing.TestCase.getGlobals = function(opt_prefix) {
- // Look in the global scope for most browsers, on IE we use the little known
- // RuntimeObject which holds references to all globals. We reference this
- // via goog.global so that there isn't an aliasing that throws an exception
- // in Firefox.
- return typeof goog.global['RuntimeObject'] != 'undefined' ?
- [goog.global['RuntimeObject']((opt_prefix || '') + '*'), goog.global] :
- [goog.global];
- };
- /**
- * @private {?goog.testing.TestCase}
- */
- goog.testing.TestCase.activeTestCase_ = null;
- /**
- * @return {?goog.testing.TestCase} currently active test case or null if not
- * test is currently running. Tries the G_testRunner first then the stored
- * value (when run outside of G_testRunner.
- */
- goog.testing.TestCase.getActiveTestCase = function() {
- var gTestRunner = goog.global['G_testRunner'];
- if (gTestRunner && gTestRunner.testCase) {
- return gTestRunner.testCase;
- } else {
- return goog.testing.TestCase.activeTestCase_;
- }
- };
- /**
- * Calls {@link goog.testing.TestCase.prototype.invalidateAssertionException} on
- * the active test case if it is installed, and logs an error otherwise.
- * @param {!goog.testing.JsUnitException} e The exception object to invalidate.
- * @package
- */
- goog.testing.TestCase.invalidateAssertionException = function(e) {
- var testCase = goog.testing.TestCase.getActiveTestCase();
- if (testCase) {
- testCase.invalidateAssertionException(e);
- } else {
- goog.global.console.error(
- 'Failed to remove expected exception: no test case is installed.');
- }
- };
- /**
- * Gets called before any tests are executed. Can be overridden to set up the
- * environment for the whole test case.
- * @return {!Thenable|undefined}
- */
- goog.testing.TestCase.prototype.setUpPage = function() {};
- /**
- * Gets called after all tests have been executed. Can be overridden to tear
- * down the entire test case.
- */
- goog.testing.TestCase.prototype.tearDownPage = function() {};
- /**
- * Gets called before every goog.testing.TestCase.Test is been executed. Can be
- * overridden to add set up functionality to each test.
- * @return {!Thenable|undefined}
- */
- goog.testing.TestCase.prototype.setUp = function() {};
- /**
- * Gets called after every goog.testing.TestCase.Test has been executed. Can be
- * overriden to add tear down functionality to each test.
- * @return {!Thenable|undefined}
- */
- goog.testing.TestCase.prototype.tearDown = function() {};
- /**
- * @return {string} The function name prefix used to auto-discover tests.
- */
- goog.testing.TestCase.prototype.getAutoDiscoveryPrefix = function() {
- return 'test';
- };
- /**
- * @return {number} Time since the last batch of tests was started.
- * @protected
- */
- goog.testing.TestCase.prototype.getBatchTime = function() {
- return this.batchTime_;
- };
- /**
- * @param {number} batchTime Time since the last batch of tests was started.
- * @protected
- */
- goog.testing.TestCase.prototype.setBatchTime = function(batchTime) {
- this.batchTime_ = batchTime;
- };
- /**
- * Creates a {@code goog.testing.TestCase.Test} from an auto-discovered
- * function.
- * @param {string} name The name of the function.
- * @param {function() : void} ref The auto-discovered function.
- * @return {!goog.testing.TestCase.Test} The newly created test.
- * @protected
- */
- goog.testing.TestCase.prototype.createTestFromAutoDiscoveredFunction = function(
- name, ref) {
- return new goog.testing.TestCase.Test(name, ref, goog.global);
- };
- /**
- * Adds any functions defined on 'obj' (the global object, by default)
- * that correspond to lifecycle events for the test case. Overrides
- * setUp, tearDown, setUpPage, tearDownPage, runTests, and shouldRunTests
- * if they are defined on 'obj'.
- * @param {!Object=} opt_obj Defaults to goog.global.
- */
- goog.testing.TestCase.prototype.autoDiscoverLifecycle = function(opt_obj) {
- var obj = opt_obj || goog.global;
- if (obj['setUp']) {
- this.setUp = goog.bind(obj['setUp'], obj);
- }
- if (obj['tearDown']) {
- this.tearDown = goog.bind(obj['tearDown'], obj);
- }
- if (obj['setUpPage']) {
- this.setUpPage = goog.bind(obj['setUpPage'], obj);
- }
- if (obj['tearDownPage']) {
- this.tearDownPage = goog.bind(obj['tearDownPage'], obj);
- }
- if (obj['runTests']) {
- this.runTests = goog.bind(obj['runTests'], obj);
- }
- if (obj['shouldRunTests']) {
- this.shouldRunTests = goog.bind(obj['shouldRunTests'], obj);
- }
- };
- // TODO(johnlenz): make this package private
- /**
- * @param {!Object} obj An object from which to extract test and lifecycle
- * methods.
- */
- goog.testing.TestCase.prototype.setTestObj = function(obj) {
- // Drop any previously added (likely auto-discovered) tests, only one source
- // of discovered test and life-cycle methods is allowed.
- goog.asserts.assert(
- this.tests_.length == 0, 'Test methods have already been configured.');
- var regex = new RegExp('^' + this.getAutoDiscoveryPrefix());
- var properties = goog.object.getAllPropertyNames(obj);
- for (var i = 0; i < properties.length; i++) {
- var name = properties[i];
- if (regex.test(name)) {
- var testMethod = obj[name];
- if (goog.isFunction(testMethod)) {
- this.addNewTest(name, testMethod, obj);
- }
- }
- }
- if (obj['getTestName']) {
- this.name_ = obj['getTestName']();
- }
- this.autoDiscoverLifecycle(obj);
- };
- /**
- * Adds any functions defined in the global scope that are prefixed with "test"
- * to the test case.
- */
- goog.testing.TestCase.prototype.autoDiscoverTests = function() {
- var prefix = this.getAutoDiscoveryPrefix();
- var testSources = this.getGlobals(prefix);
- var foundTests = [];
- for (var i = 0; i < testSources.length; i++) {
- var testSource = testSources[i];
- for (var name in testSource) {
- if ((new RegExp('^' + prefix)).test(name)) {
- var ref;
- try {
- ref = testSource[name];
- } catch (ex) {
- // NOTE(brenneman): When running tests from a file:// URL on Firefox
- // 3.5 for Windows, any reference to goog.global.sessionStorage raises
- // an "Operation is not supported" exception. Ignore any exceptions
- // raised by simply accessing global properties.
- ref = undefined;
- }
- if (goog.isFunction(ref)) {
- foundTests.push(this.createTestFromAutoDiscoveredFunction(name, ref));
- }
- }
- }
- }
- for (var i = 0; i < foundTests.length; i++) {
- this.add(foundTests[i]);
- }
- this.orderTests_();
- this.log(this.getCount() + ' tests auto-discovered');
- // TODO(user): Do this as a separate call. Unfortunately, a lot of projects
- // currently override autoDiscoverTests and expect lifecycle events to be
- // registered as a part of this call.
- this.autoDiscoverLifecycle();
- };
- /**
- * Checks to see if the test should be marked as failed before it is run.
- *
- * If there was an error in setUpPage, we treat that as a failure for all tests
- * and mark them all as having failed.
- *
- * @param {goog.testing.TestCase.Test} testCase The current test case.
- * @return {boolean} Whether the test was marked as failed.
- * @protected
- */
- goog.testing.TestCase.prototype.maybeFailTestEarly = function(testCase) {
- if (this.exceptionBeforeTest) {
- // We just use the first error to report an error on a failed test.
- testCase.name = 'setUpPage for ' + testCase.name;
- this.doError(testCase, this.exceptionBeforeTest);
- return true;
- }
- return false;
- };
- /**
- * Cycles through the tests, yielding asynchronously if the execution time
- * exceeds {@link #maxRunTime}. In particular, there is no guarantee that
- * the test case has finished execution once this method has returned.
- * To be notified when the test case has finished execution, use
- * {@link #setCompletedCallback} or {@link #runTestsReturningPromise}.
- */
- goog.testing.TestCase.prototype.cycleTests = function() {
- this.saveMessage('Start');
- this.batchTime_ = this.now();
- if (this.running) {
- this.runNextTestCallback_ = goog.nullFunction;
- // Kick off the tests. runNextTest_ will schedule all of the tests,
- // using a mixture of synchronous and asynchronous strategies.
- goog.testing.Continuation_.run(this.runNextTest_());
- }
- };
- /**
- * Counts the number of files that were loaded for dependencies that are
- * required to run the test.
- * @return {number} The number of files loaded.
- * @private
- */
- goog.testing.TestCase.prototype.countNumFilesLoaded_ = function() {
- var scripts = goog.dom.getElementsByTagName(goog.dom.TagName.SCRIPT);
- var count = 0;
- for (var i = 0, n = scripts.length; i < n; i++) {
- if (scripts[i].src) {
- count++;
- }
- }
- return count;
- };
- /**
- * Calls a function after a delay, using the protected timeout.
- * @param {Function} fn The function to call.
- * @param {number} time Delay in milliseconds.
- * @return {number} The timeout id.
- * @protected
- */
- goog.testing.TestCase.prototype.timeout = function(fn, time) {
- // NOTE: invoking protectedSetTimeout_ as a member of goog.testing.TestCase
- // would result in an Illegal Invocation error. The method must be executed
- // with the global context.
- var protectedSetTimeout = goog.testing.TestCase.protectedSetTimeout_;
- return protectedSetTimeout(fn, time);
- };
- /**
- * Clears a timeout created by {@code this.timeout()}.
- * @param {number} id A timeout id.
- * @protected
- */
- goog.testing.TestCase.prototype.clearTimeout = function(id) {
- // NOTE: see execution note for protectedSetTimeout above.
- var protectedClearTimeout = goog.testing.TestCase.protectedClearTimeout_;
- protectedClearTimeout(id);
- };
- /**
- * @return {number} The current time in milliseconds.
- * @protected
- */
- goog.testing.TestCase.prototype.now = function() {
- return goog.testing.TestCase.now();
- };
- /**
- * @return {number} The current time in milliseconds.
- * @protected
- */
- goog.testing.TestCase.now = function() {
- // don't use goog.now as some tests override it.
- if (goog.testing.TestCase.protectedPerformance_) {
- return goog.testing.TestCase.protectedPerformance_.now();
- }
- // Fallback for IE8
- // Cannot use "new goog.testing.TestCase.protectedDate_()" due to b/8323223.
- var protectedDate = goog.testing.TestCase.protectedDate_;
- return new protectedDate().getTime();
- };
- /**
- * Returns the current time.
- * @return {string} HH:MM:SS.
- * @private
- */
- goog.testing.TestCase.prototype.getTimeStamp_ = function() {
- // Cannot use "new goog.testing.TestCase.protectedDate_()" due to b/8323223.
- var protectedDate = goog.testing.TestCase.protectedDate_;
- var d = new protectedDate();
- // Ensure millis are always 3-digits
- var millis = '00' + d.getMilliseconds();
- millis = millis.substr(millis.length - 3);
- return this.pad_(d.getHours()) + ':' + this.pad_(d.getMinutes()) + ':' +
- this.pad_(d.getSeconds()) + '.' + millis;
- };
- /**
- * Pads a number to make it have a leading zero if it's less than 10.
- * @param {number} number The number to pad.
- * @return {string} The resulting string.
- * @private
- */
- goog.testing.TestCase.prototype.pad_ = function(number) {
- return number < 10 ? '0' + number : String(number);
- };
- /**
- * Trims a path to be only that after google3.
- * @param {string} path The path to trim.
- * @return {string} The resulting string.
- * @private
- */
- goog.testing.TestCase.prototype.trimPath_ = function(path) {
- return path.substring(path.indexOf('google3') + 8);
- };
- /**
- * Handles a test that passed.
- * @param {goog.testing.TestCase.Test} test The test that passed.
- * @protected
- */
- goog.testing.TestCase.prototype.doSuccess = function(test) {
- this.result_.successCount++;
- // An empty list of error messages indicates that the test passed.
- // If we already have a failure for this test, do not set to empty list.
- if (!(test.name in this.result_.resultsByName)) {
- this.result_.resultsByName[test.name] = [];
- }
- var message = test.name + ' : PASSED';
- this.saveMessage(message);
- this.log(message);
- if (this.testDone_) {
- this.doTestDone_(test, []);
- }
- };
- /**
- * Records and logs a test failure.
- * @param {string} testName The name of the test that failed.
- * @param {*=} opt_e The exception object associated with the
- * failure or a string.
- * @private
- */
- goog.testing.TestCase.prototype.recordError_ = function(testName, opt_e) {
- var message = testName + ' : FAILED';
- this.log(message);
- this.saveMessage(message);
- var err = this.logError(testName, opt_e);
- this.result_.errors.push(err);
- if (testName in this.result_.resultsByName) {
- this.result_.resultsByName[testName].push(err);
- } else {
- this.result_.resultsByName[testName] = [err];
- }
- };
- /**
- * Handles a test that failed.
- * @param {goog.testing.TestCase.Test} test The test that failed.
- * @param {*=} opt_e The exception object associated with the
- * failure or a string.
- * @protected
- */
- goog.testing.TestCase.prototype.doError = function(test, opt_e) {
- if (!test || !test.name) {
- console.error('no name!' + opt_e);
- }
- this.recordError_(test.name, opt_e);
- if (this.testDone_) {
- var results = this.result_.resultsByName[test.name];
- var errMsgs = [];
- for (var i = 0; i < results.length; i++) {
- errMsgs.push(results[i].toString());
- }
- this.doTestDone_(test, errMsgs);
- }
- };
- /**
- * Makes note of an exception arising from an assertion, and then throws it. If
- * the test otherwise passes (i.e., because something else caught the exception
- * on its way to the test framework), it will be forced to fail.
- * @param {!goog.testing.JsUnitException} e The exception object being thrown.
- * @throws {goog.testing.JsUnitException}
- * @package
- */
- goog.testing.TestCase.prototype.raiseAssertionException = function(e) {
- if (this.failOnUnreportedAsserts) {
- this.thrownAssertionExceptions_.push(e);
- }
- throw e;
- };
- /**
- * Removes the specified exception from being tracked. This only needs to be
- * called for internal functions that intentionally catch an exception, such as
- * {@code #assertThrowsJsUnitException}.
- * @param {!goog.testing.JsUnitException} e The exception object to invalidate.
- * @package
- */
- goog.testing.TestCase.prototype.invalidateAssertionException = function(e) {
- if (this.failOnUnreportedAsserts) {
- goog.array.remove(this.thrownAssertionExceptions_, e);
- }
- };
- /**
- * @param {string} name Failed test name.
- * @param {*=} opt_e The exception object associated with the
- * failure or a string.
- * @return {!goog.testing.TestCase.Error} Error object.
- */
- goog.testing.TestCase.prototype.logError = function(name, opt_e) {
- var errMsg = null;
- var stack = null;
- if (opt_e) {
- this.log(opt_e);
- if (goog.isString(opt_e)) {
- errMsg = opt_e;
- } else {
- errMsg = opt_e.message || opt_e.description || opt_e.toString();
- stack = opt_e.stack ? opt_e.stack : opt_e['stackTrace'];
- }
- } else {
- errMsg = 'An unknown error occurred';
- }
- if (stack) {
- // Remove extra goog.testing.TestCase frames from the end.
- stack = stack.replace(
- /\n.*goog\.testing\.TestCase\.(prototype\.)?invokeTestFunction[^\0]*/m,
- '');
- }
- var err = new goog.testing.TestCase.Error(name, errMsg, stack);
- // Avoid double logging.
- if (!opt_e || !opt_e['isJsUnitException'] ||
- !opt_e['loggedJsUnitException']) {
- this.saveMessage(err.toString());
- }
- if (opt_e && opt_e['isJsUnitException']) {
- opt_e['loggedJsUnitException'] = true;
- }
- return err;
- };
- /**
- * A class representing a single test function.
- * @param {string} name The test name.
- * @param {Function} ref Reference to the test function.
- * @param {Object=} opt_scope Optional scope that the test function should be
- * called in.
- * @constructor
- */
- goog.testing.TestCase.Test = function(name, ref, opt_scope) {
- /**
- * The name of the test.
- * @type {string}
- */
- this.name = name;
- /**
- * Reference to the test function.
- * @type {Function}
- */
- this.ref = ref;
- /**
- * Scope that the test function should be called in.
- * @type {Object}
- */
- this.scope = opt_scope || null;
- /**
- * Timestamp just before the test begins execution.
- * @type {number}
- * @private
- */
- this.startTime_;
- /**
- * Timestamp just after the test ends execution.
- * @type {number}
- * @private
- */
- this.stoppedTime_;
- };
- /**
- * Executes the test function.
- * @package
- */
- goog.testing.TestCase.Test.prototype.execute = function() {
- this.ref.call(this.scope);
- };
- /**
- * Sets the start time
- */
- goog.testing.TestCase.Test.prototype.started = function() {
- this.startTime_ = goog.testing.TestCase.now();
- };
- /**
- * Sets the stop time
- */
- goog.testing.TestCase.Test.prototype.stopped = function() {
- this.stoppedTime_ = goog.testing.TestCase.now();
- };
- /**
- * Returns the runtime for this test function
- * @return {number} milliseconds takenn by the test.
- */
- goog.testing.TestCase.Test.prototype.getElapsedTime = function() {
- return this.stoppedTime_ - this.startTime_;
- };
- /**
- * A class for representing test results. A bag of public properties.
- * @param {goog.testing.TestCase} testCase The test case that owns this result.
- * @constructor
- * @final
- */
- goog.testing.TestCase.Result = function(testCase) {
- /**
- * The test case that owns this result.
- * @type {goog.testing.TestCase}
- * @private
- */
- this.testCase_ = testCase;
- /**
- * Total number of tests that should have been run.
- * @type {number}
- */
- this.totalCount = 0;
- /**
- * Total number of tests that were actually run.
- * @type {number}
- */
- this.runCount = 0;
- /**
- * Number of successful tests.
- * @type {number}
- */
- this.successCount = 0;
- /**
- * The amount of time the tests took to run.
- * @type {number}
- */
- this.runTime = 0;
- /**
- * The number of files loaded to run this test.
- * @type {number}
- */
- this.numFilesLoaded = 0;
- /**
- * Whether this test case was suppressed by shouldRunTests() returning false.
- * @type {boolean}
- */
- this.testSuppressed = false;
- /**
- * Test results for each test that was run. The test name is always added
- * as the key in the map, and the array of strings is an optional list
- * of failure messages. If the array is empty, the test passed. Otherwise,
- * the test failed.
- * @type {!Object<string, !Array<goog.testing.TestCase.Error>>}
- */
- this.resultsByName = {};
- /**
- * Errors encountered while running the test.
- * @type {!Array<goog.testing.TestCase.Error>}
- */
- this.errors = [];
- /**
- * Messages to show the user after running the test.
- * @type {!Array<string>}
- */
- this.messages = [];
- /**
- * Whether the tests have completed.
- * @type {boolean}
- */
- this.complete = false;
- };
- /**
- * @return {boolean} Whether the test was successful.
- */
- goog.testing.TestCase.Result.prototype.isSuccess = function() {
- return this.complete && this.errors.length == 0;
- };
- /**
- * @return {string} A summary of the tests, including total number of tests that
- * passed, failed, and the time taken.
- */
- goog.testing.TestCase.Result.prototype.getSummary = function() {
- var summary = this.runCount + ' of ' + this.totalCount + ' tests run in ' +
- this.runTime + 'ms.\n';
- if (this.testSuppressed) {
- summary += 'Tests not run because shouldRunTests() returned false.';
- } else {
- var failures = this.totalCount - this.successCount;
- var suppressionMessage = '';
- var countOfRunTests = this.testCase_.getActuallyRunCount();
- if (countOfRunTests) {
- failures = countOfRunTests - this.successCount;
- suppressionMessage = ', ' + (this.totalCount - countOfRunTests) +
- ' suppressed by querystring';
- }
- summary += this.successCount + ' passed, ' + failures + ' failed' +
- suppressionMessage + '.\n' + Math.round(this.runTime / this.runCount) +
- ' ms/test. ' + this.numFilesLoaded + ' files loaded.';
- }
- return summary;
- };
- /**
- * @param {function(goog.testing.TestCase.Test, !Array<string>)} testDone
- */
- goog.testing.TestCase.prototype.setTestDoneCallback = function(testDone) {
- this.testDone_ = testDone;
- };
- /**
- * @param {goog.testing.TestCase.Test} test
- * @param {!Array<string>} errMsgs
- * @private
- */
- goog.testing.TestCase.prototype.doTestDone_ = function(test, errMsgs) {
- test.stopped();
- this.testDone_(test, errMsgs);
- };
- /** Initializes the TestCase.
- * @param {goog.testing.TestCase} testCase The test case to install.
- * @param {function(goog.testing.TestCase.Test, Array<string>)=} opt_testDone
- * Called when each test completes.
- */
- goog.testing.TestCase.initializeTestCase = function(testCase, opt_testDone) {
- if (opt_testDone) {
- testCase.setTestDoneCallback(opt_testDone);
- }
- testCase.autoDiscoverTests();
- if (goog.global.location) {
- var search = goog.global.location.search;
- testCase.setOrder(
- goog.testing.TestCase.parseOrder_(search) ||
- goog.testing.TestCase.Order.SORTED);
- testCase.setTestsToRun(goog.testing.TestCase.parseRunTests_(search));
- }
- goog.testing.TestCase.activeTestCase_ = testCase;
- };
- /**
- * Initializes the given test case with the global test runner 'G_testRunner'.
- * @param {goog.testing.TestCase} testCase The test case to install.
- * @param {function(goog.testing.TestCase.Test, Array<string>)=} opt_testDone
- * Called when each test completes.
- */
- goog.testing.TestCase.initializeTestRunner = function(testCase, opt_testDone) {
- goog.testing.TestCase.initializeTestCase(testCase, opt_testDone);
- var gTestRunner = goog.global['G_testRunner'];
- if (gTestRunner) {
- gTestRunner['initialize'](testCase);
- } else {
- throw Error(
- 'G_testRunner is undefined. Please ensure goog.testing.jsunit' +
- ' is included.');
- }
- };
- /**
- * Parses URL query parameters for the 'order' parameter.
- * @param {string} search The URL query string.
- * @return {?goog.testing.TestCase.Order} The sort order for running tests.
- * @private
- */
- goog.testing.TestCase.parseOrder_ = function(search) {
- var order = null;
- var orderMatch = search.match(/(?:\?|&)order=(natural|random|sorted)/i);
- if (orderMatch) {
- order = /** @type {goog.testing.TestCase.Order} */ (
- orderMatch[1].toLowerCase());
- }
- return order;
- };
- /**
- * Parses URL query parameters for the 'runTests' parameter.
- * @param {string} search The URL query string.
- * @return {Object<string, boolean>} A set of test names or test indices to be
- * run by the test runner.
- * @private
- */
- goog.testing.TestCase.parseRunTests_ = function(search) {
- var testsToRun = null;
- var runTestsMatch = search.match(/(?:\?|&)runTests=([^?&]+)/i);
- if (runTestsMatch) {
- testsToRun = {};
- var arr = runTestsMatch[1].split(',');
- for (var i = 0, len = arr.length; i < len; i++) {
- testsToRun[arr[i]] = true;
- }
- }
- return testsToRun;
- };
- /**
- * Wraps provided promise and returns a new promise which will be rejected
- * if the original promise does not settle within the given timeout.
- * @param {!goog.Promise<T>} promise
- * @param {number} timeoutInMs Number of milliseconds to wait for the promise to
- * settle before failing it with a timeout error.
- * @param {string} errorMsg Error message to use if the promise times out.
- * @return {!goog.Promise<T>} A promise that will settle with the original
- promise unless the timeout is exceeded.
- * errror.
- * @template T
- * @private
- */
- goog.testing.TestCase.prototype.rejectIfPromiseTimesOut_ = function(
- promise, timeoutInMs, errorMsg) {
- var self = this;
- var start = this.now();
- return new goog.Promise(function(resolve, reject) {
- var timeoutId = self.timeout(function() {
- var elapsed = self.now() - start;
- reject(new Error(errorMsg + '\nElapsed time: ' + elapsed + 'ms.'));
- }, timeoutInMs);
- promise.then(resolve, reject);
- var clearTimeout = goog.bind(self.clearTimeout, self, timeoutId);
- promise.then(clearTimeout, clearTimeout);
- });
- };
- /**
- * A class representing an error thrown by the test
- * @param {string} source The name of the test which threw the error.
- * @param {string} message The error message.
- * @param {string=} opt_stack A string showing the execution stack.
- * @constructor
- * @final
- */
- goog.testing.TestCase.Error = function(source, message, opt_stack) {
- /**
- * The name of the test which threw the error.
- * @type {string}
- */
- this.source = source;
- /**
- * Reference to the test function.
- * @type {string}
- */
- this.message = message;
- /**
- * The stack.
- * @type {?string}
- */
- this.stack = null;
- if (opt_stack) {
- this.stack = opt_stack;
- } else {
- // Attempt to capture a stack trace.
- if (Error.captureStackTrace) {
- // See https://code.google.com/p/v8-wiki/wiki/JavaScriptStackTraceApi
- Error.captureStackTrace(this, goog.testing.TestCase.Error);
- } else {
- var stack = new Error().stack;
- if (stack) {
- this.stack = stack;
- }
- }
- }
- };
- /**
- * Returns a string representing the error object.
- * @return {string} A string representation of the error.
- * @override
- */
- goog.testing.TestCase.Error.prototype.toString = function() {
- return 'ERROR in ' + this.source + '\n' + this.message +
- (this.stack ? '\n' + this.stack : '');
- };
- /**
- * Returns an object representing the error suitable for JSON serialization.
- * @return {!goog.testing.TestCase.IResult} An object
- * representation of the error.
- * @private
- */
- goog.testing.TestCase.Error.prototype.toObject_ = function() {
- return {
- 'source': this.source,
- 'message': this.message,
- 'stacktrace': this.stack || ''
- };
- };
- /**
- * @constructor
- * @param {!function(): (?goog.testing.Continuation_|undefined)} fn
- * @private
- */
- goog.testing.Continuation_ = function(fn) {
- /** @private @const */
- this.fn_ = fn;
- };
- /** @param {?goog.testing.Continuation_|undefined} continuation */
- goog.testing.Continuation_.run = function(continuation) {
- var fn = continuation && continuation.fn_;
- while (fn) {
- continuation = fn();
- fn = continuation && continuation.fn_;
- }
- };
|