chromium/third_party/google-closure-library/closure/goog/testing/asserts.js

/**
 * @license
 * Copyright The Closure Library Authors.
 * SPDX-License-Identifier: Apache-2.0
 */

goog.provide('goog.testing.asserts');
goog.setTestOnly();

goog.require('goog.dom.safe');
goog.require('goog.html.uncheckedconversions');
goog.require('goog.string');
goog.require('goog.string.Const');
goog.require('goog.testing.JsUnitException');

var DOUBLE_EQUALITY_PREDICATE = function(var1, var2) {
  'use strict';
  return var1 == var2;
};
var JSUNIT_UNDEFINED_VALUE = void 0;
var TO_STRING_EQUALITY_PREDICATE = function(var1, var2) {
  'use strict';
  return var1.toString() === var2.toString();
};
var OUTPUT_NEW_LINE_THRESHOLD = 40;


/** @typedef {function(?, ?):boolean} */
var PredicateFunctionType;


/**
 * An associative array of constructors corresponding to primitive and
 * well-known JS types.
 * @const {!Array<string>}
 */
const PRIMITIVE_TRUE_TYPES =
    ['String', 'Boolean', 'Number', 'Array', 'RegExp', 'Date', 'Function'];

if (typeof ArrayBuffer === 'function') {
  PRIMITIVE_TRUE_TYPES.push('ArrayBuffer');
}


/**
 * @const {{
 *   String : !PredicateFunctionType,
 *   Number : !PredicateFunctionType,
 *   Boolean : !PredicateFunctionType,
 *   Date : !PredicateFunctionType,
 *   RegExp : !PredicateFunctionType,
 *   Function : !PredicateFunctionType,
 *   TrustedHTML : !PredicateFunctionType,
 *   TrustedScript : !PredicateFunctionType,
 *   TrustedScriptURL : !PredicateFunctionType
 * }}
 */
const EQUALITY_PREDICATES = {
  'String': DOUBLE_EQUALITY_PREDICATE,
  'Number': DOUBLE_EQUALITY_PREDICATE,
  'Bigint': DOUBLE_EQUALITY_PREDICATE,
  'Boolean': DOUBLE_EQUALITY_PREDICATE,
  'Date': function(date1, date2) {
    'use strict';
    return date1.getTime() == date2.getTime();
  },
  'RegExp': TO_STRING_EQUALITY_PREDICATE,
  'Function': TO_STRING_EQUALITY_PREDICATE,
  'TrustedHTML': TO_STRING_EQUALITY_PREDICATE,
  'TrustedScript': TO_STRING_EQUALITY_PREDICATE,
  'TrustedScriptURL': TO_STRING_EQUALITY_PREDICATE
};


/**
 * Compares equality of two numbers, allowing them to differ up to a given
 * tolerance.
 * @param {number} var1 A number.
 * @param {number} var2 A number.
 * @param {number} tolerance the maximum allowed difference.
 * @return {boolean} Whether the two variables are sufficiently close.
 * @private
 */
goog.testing.asserts.numberRoughEqualityPredicate_ = function(
    var1, var2, tolerance) {
  'use strict';
  return Math.abs(var1 - var2) <= tolerance;
};


/**
 * @type {!Object<string, function(?, ?, number): boolean>}
 * @private
 */
goog.testing.asserts.primitiveRoughEqualityPredicates_ = {
  'Number': goog.testing.asserts.numberRoughEqualityPredicate_
};


var _trueTypeOf = function(something) {
  'use strict';
  let result = typeof something;
  try {
    switch (result) {
      case 'string':
        break;
      case 'boolean':
        break;
      case 'number':
        break;
      case 'object':
        if (something == null) {
          result = 'null';
          break;
        }
      case 'function':
        let foundConstructor = false;
        for (let i = 0; i < PRIMITIVE_TRUE_TYPES.length; i++) {
          // NOTE: this cannot be a for-of loop because it's used from Rhino
          // without the necessary Array.prototype[Symbol.iterator] polyfill.
          const trueType = PRIMITIVE_TRUE_TYPES[i];
          if (something.constructor === goog.global[trueType]) {
            result = trueType;
            foundConstructor = true;
            break;
          }
        }
        // Constructor doesn't match any of the known "primitive" constructors.
        if (!foundConstructor) {
          const m =
              something.constructor.toString().match(/function\s*([^( ]+)\(/);
          if (m) {
            result = m[1];
          }
        }
        break;
    }
  } catch (e) {
  } finally {
    result = result.substr(0, 1).toUpperCase() + result.substr(1);
  }
  return result;
};

var _displayStringForValue = function(aVar) {
  'use strict';
  var result;
  try {
    result = '<' + String(aVar) + '>';
  } catch (ex) {
    result = '<toString failed: ' + ex.message + '>';
    // toString does not work on this object :-(
  }
  if (!(aVar === null || aVar === JSUNIT_UNDEFINED_VALUE)) {
    result += ' (' + _trueTypeOf(aVar) + ')';
  }
  return result;
};

/** @param {?} failureMessage */
goog.testing.asserts.fail = function(failureMessage) {
  'use strict';
  _assert('Call to fail()', false, failureMessage);
};
/**
 * @const
 * @suppress {duplicate,checkTypes} Test frameworks like Jasmine may also
 * define global fail functions.
 */
var fail = goog.testing.asserts.fail;

var argumentsIncludeComments = function(expectedNumberOfNonCommentArgs, args) {
  'use strict';
  return args.length == expectedNumberOfNonCommentArgs + 1;
};

var commentArg = function(expectedNumberOfNonCommentArgs, args) {
  'use strict';
  if (argumentsIncludeComments(expectedNumberOfNonCommentArgs, args)) {
    return args[0];
  }

  return null;
};

var nonCommentArg = function(
    desiredNonCommentArgIndex, expectedNumberOfNonCommentArgs, args) {
  'use strict';
  return argumentsIncludeComments(expectedNumberOfNonCommentArgs, args) ?
      args[desiredNonCommentArgIndex] :
      args[desiredNonCommentArgIndex - 1];
};

var _validateArguments = function(expectedNumberOfNonCommentArgs, args) {
  'use strict';
  var valid = args.length == expectedNumberOfNonCommentArgs ||
      args.length == expectedNumberOfNonCommentArgs + 1 &&
          typeof args[0] === 'string';
  if (!valid) {
    goog.testing.asserts.raiseException(
        'Incorrect arguments passed to assert function.\n' +
        'Expected ' + expectedNumberOfNonCommentArgs + ' argument(s) plus ' +
        'optional comment; got ' + args.length + '.');
  }
};

/**
 * @return {?} goog.testing.TestCase or null
 * We suppress the lint error and we explicitly do not goog.require()
 * goog.testing.TestCase to avoid a build time dependency cycle.
 * @suppress {missingRequire|undefinedNames|undefinedVars|missingProperties}
 * @private
 */
var _getCurrentTestCase = function() {
  'use strict';
  // Some users of goog.testing.asserts do not use goog.testing.TestRunner and
  // they do not include goog.testing.TestCase. Exceptions will not be
  // completely correct for these users.
  if (!goog.testing.TestCase) {
    if (goog.global.console) {
      goog.global.console.error(
          'Missing goog.testing.TestCase, ' +
          'add /* @suppress {extraRequire} */' +
          'goog.require(\'goog.testing.TestCase\')');
    }
    return null;
  }
  return goog.testing.TestCase.getActiveTestCase();
};

var _assert = function(comment, booleanValue, failureMessage) {
  'use strict';
  // If another framework has installed an adapter, tell it about the assertion.
  var adapter =
      typeof window !== 'undefined' && window['Closure assert adapter'];
  if (adapter) {
    adapter['assertWithMessage'](
        booleanValue,
        goog.testing.JsUnitException.generateMessage(comment, failureMessage));
    // Also throw an error, for callers that assume that asserts throw. We don't
    // include error details to avoid duplicate failure messages.
    if (!booleanValue) throw new Error('goog.testing assertion failed');
  }
  if (!booleanValue) {
    goog.testing.asserts.raiseException(comment, failureMessage);
  }
};


/**
 * @param {*} expected The expected value.
 * @param {*} actual The actual value.
 * @return {string} A failure message of the values don't match.
 * @private
 */
goog.testing.asserts.getDefaultErrorMsg_ = function(expected, actual) {
  'use strict';
  var expectedDisplayString = _displayStringForValue(expected);
  var actualDisplayString = _displayStringForValue(actual);
  var shouldUseNewLines =
      expectedDisplayString.length > OUTPUT_NEW_LINE_THRESHOLD ||
      actualDisplayString.length > OUTPUT_NEW_LINE_THRESHOLD;
  var msg = [
    'Expected', expectedDisplayString, 'but was', actualDisplayString
  ].join(shouldUseNewLines ? '\n' : ' ');

  if ((typeof expected == 'string') && (typeof actual == 'string')) {
    // Try to find a human-readable difference.
    var limit = Math.min(expected.length, actual.length);
    var commonPrefix = 0;
    while (commonPrefix < limit &&
           expected.charAt(commonPrefix) == actual.charAt(commonPrefix)) {
      commonPrefix++;
    }

    var commonSuffix = 0;
    while (commonSuffix < limit &&
           expected.charAt(expected.length - commonSuffix - 1) ==
               actual.charAt(actual.length - commonSuffix - 1)) {
      commonSuffix++;
    }

    if (commonPrefix + commonSuffix > limit) {
      commonSuffix = 0;
    }

    if (commonPrefix > 2 || commonSuffix > 2) {
      var printString = function(str) {
        'use strict';
        var startIndex = Math.max(0, commonPrefix - 2);
        var endIndex = Math.min(str.length, str.length - (commonSuffix - 2));
        return (startIndex > 0 ? '...' : '') +
            str.substring(startIndex, endIndex) +
            (endIndex < str.length ? '...' : '');
      };

      var expectedPrinted = printString(expected);
      var expectedActual = printString(actual);
      var shouldUseNewLinesInDiff =
          expectedPrinted.length > OUTPUT_NEW_LINE_THRESHOLD ||
          expectedActual.length > OUTPUT_NEW_LINE_THRESHOLD;
      msg += '\nDifference was at position ' + commonPrefix + '. ' + [
        'Expected', '[' + expectedPrinted + ']', 'vs. actual',
        '[' + expectedActual + ']'
      ].join(shouldUseNewLinesInDiff ? '\n' : ' ');
    }
  }
  return msg;
};


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assert = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var comment = commentArg(1, arguments);
  var booleanValue = nonCommentArg(1, 1, arguments);

  _assert(
      comment, typeof booleanValue === 'boolean',
      'Bad argument to assert(boolean): ' +
          _displayStringForValue(booleanValue));
  _assert(comment, booleanValue, 'Call to assert(boolean) with false');
};
/** @const */
var assert = goog.testing.asserts.assert;


/**
 * Asserts that the function throws an error.
 *
 * @param {!(string|Function)} a The assertion comment or the function to call.
 * @param {!Function=} opt_b The function to call (if the first argument of
 *     `assertThrows` was the comment).
 * @return {!Error} The error thrown by the function. Beware that code may throw
 *     other types in strange scenarios.
 * @throws {goog.testing.JsUnitException} If the assertion failed.
 */
goog.testing.asserts.assertThrows = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var func = nonCommentArg(1, 1, arguments);
  var comment = commentArg(1, arguments);
  _assert(
      comment, typeof func == 'function',
      'Argument passed to assertThrows is not a function');

  try {
    func();
  } catch (e) {
    goog.testing.asserts.removeOperaStacktrace_(e);

    var testCase = _getCurrentTestCase();
    if (e && e['isJsUnitException'] && testCase) {
      goog.testing.asserts.raiseException(
          comment,
          'Function passed to assertThrows caught a JsUnitException (usually ' +
              'from an assert or call to fail()). If this is expected, use ' +
              'assertThrowsJsUnitException instead.');
    }

    return e;
  }
  goog.testing.asserts.raiseException(
      comment, 'No exception thrown from function passed to assertThrows');
  throw new Error('Should have thrown an error.');  // Make the compiler happy.
};
/** @const */
var assertThrows = goog.testing.asserts.assertThrows;


/**
 * Removes a stacktrace from an Error object for Opera 10.0.
 * @param {*} e
 * @private
 */
goog.testing.asserts.removeOperaStacktrace_ = function(e) {
  'use strict';
  if (goog.isObject(e) && typeof e['stacktrace'] === 'string' &&
      typeof e['message'] === 'string') {
    var startIndex = e['message'].length - e['stacktrace'].length;
    if (e['message'].indexOf(e['stacktrace'], startIndex) == startIndex) {
      e['message'] = e['message'].substr(0, startIndex - 14);
    }
  }
};


/**
 * Asserts that the function does not throw an error.
 *
 * @param {!(string|Function)} a The assertion comment or the function to call.
 * @param {!Function=} opt_b The function to call (if the first argument of
 *     `assertNotThrows` was the comment).
 * @return {*} The return value of the function.
 * @throws {goog.testing.JsUnitException} If the assertion failed.
 */
goog.testing.asserts.assertNotThrows = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var comment = commentArg(1, arguments);
  var func = nonCommentArg(1, 1, arguments);
  _assert(
      comment, typeof func == 'function',
      'Argument passed to assertNotThrows is not a function');

  try {
    return func();
  } catch (e) {
    comment = comment ? (comment + '\n') : '';
    comment += 'A non expected exception was thrown from function passed to ' +
        'assertNotThrows';
    // Some browsers don't have a stack trace so at least have the error
    // description.
    var stackTrace = e['stack'] || e['stacktrace'] || e.toString();
    goog.testing.asserts.raiseException(comment, stackTrace);
  }
};
/** @const */
var assertNotThrows = goog.testing.asserts.assertNotThrows;


/**
 * Asserts that the given callback function results in a JsUnitException when
 * called, and that the resulting failure message matches the given expected
 * message.
 * @param {function() : void} callback Function to be run expected to result
 *     in a JsUnitException (usually contains a call to an assert).
 * @param {string=} opt_expectedMessage Failure message expected to be given
 *     with the exception.
 * @return {!goog.testing.JsUnitException} The error thrown by the function.
 * @throws {goog.testing.JsUnitException} If the function did not throw a
 *     JsUnitException.
 */
goog.testing.asserts.assertThrowsJsUnitException = function(
    callback, opt_expectedMessage) {
  'use strict';
  try {
    callback();
  } catch (e) {
    var testCase = _getCurrentTestCase();
    if (testCase) {
      testCase.invalidateAssertionException(e);
    } else {
      goog.global.console.error(
          'Failed to remove expected exception: no test case is installed.');
    }

    if (!e.isJsUnitException) {
      goog.testing.asserts.fail(
          'Expected a JsUnitException, got \'' + e + '\' instead');
    }

    if (typeof opt_expectedMessage != 'undefined' &&
        e.message != opt_expectedMessage) {
      goog.testing.asserts.fail(
          'Expected message [' + opt_expectedMessage + '] but got [' +
          e.message + ']');
    }

    return e;
  }

  var msg = 'Expected a failure';
  if (typeof opt_expectedMessage != 'undefined') {
    msg += ': ' + opt_expectedMessage;
  }
  throw new goog.testing.JsUnitException(msg);
};
/** @const */
var assertThrowsJsUnitException =
    goog.testing.asserts.assertThrowsJsUnitException;


/**
 * Asserts that the IThenable rejects.
 *
 * This is useful for asserting that async functions throw, like an asynchronous
 * assertThrows. Example:
 *
 * ```
 *   async function shouldThrow() { throw new Error('error!'); }
 *   async function testShouldThrow() {
 *     const error = await assertRejects(shouldThrow());
 *     assertEquals('error!', error.message);
 *   }
 * ```
 *
 * @param {!(string|IThenable)} a The assertion comment or the IThenable.
 * @param {!IThenable=} opt_b The IThenable (if the first argument of
 *     `assertRejects` was the comment).
 * @return {!IThenable<*>} A child IThenable which resolves with the error that
 *     the passed in IThenable rejects with. This IThenable will reject if the
 *     passed in IThenable does not reject.
 */
goog.testing.asserts.assertRejects = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var thenable = /** @type {!IThenable<*>} */ (nonCommentArg(1, 1, arguments));
  var comment = commentArg(1, arguments);
  _assert(
      comment, goog.isObject(thenable) && typeof thenable.then === 'function',
      'Argument passed to assertRejects is not an IThenable');

  return thenable.then(
      function() {
        'use strict';
        goog.testing.asserts.raiseException(
            comment, 'IThenable passed into assertRejects did not reject');
      },
      function(e) {
        'use strict';
        goog.testing.asserts.removeOperaStacktrace_(e);
        return e;
      });
};
/** @const */
var assertRejects = goog.testing.asserts.assertRejects;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertTrue = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var comment = commentArg(1, arguments);
  var booleanValue = nonCommentArg(1, 1, arguments);

  _assert(
      comment, typeof booleanValue === 'boolean',
      'Bad argument to assertTrue(boolean): ' +
          _displayStringForValue(booleanValue));
  _assert(comment, booleanValue, 'Call to assertTrue(boolean) with false');
};
/** @const */
var assertTrue = goog.testing.asserts.assertTrue;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertFalse = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var comment = commentArg(1, arguments);
  var booleanValue = nonCommentArg(1, 1, arguments);

  _assert(
      comment, typeof booleanValue === 'boolean',
      'Bad argument to assertFalse(boolean): ' +
          _displayStringForValue(booleanValue));
  _assert(comment, !booleanValue, 'Call to assertFalse(boolean) with true');
};
/** @const */
var assertFalse = goog.testing.asserts.assertFalse;


/**
 * @param {*} a The expected value (2 args) or the debug message (3 args).
 * @param {*} b The actual value (2 args) or the expected value (3 args).
 * @param {*=} opt_c The actual value (3 args only).
 */
goog.testing.asserts.assertEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var var1 = nonCommentArg(1, 2, arguments);
  var var2 = nonCommentArg(2, 2, arguments);
  _assert(
      commentArg(2, arguments), var1 === var2,
      goog.testing.asserts.getDefaultErrorMsg_(var1, var2));
};
/** @const */
var assertEquals = goog.testing.asserts.assertEquals;


/**
 * @param {*} a The expected value (2 args) or the debug message (3 args).
 * @param {*} b The actual value (2 args) or the expected value (3 args).
 * @param {*=} opt_c The actual value (3 args only).
 */
goog.testing.asserts.assertNotEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var var1 = nonCommentArg(1, 2, arguments);
  var var2 = nonCommentArg(2, 2, arguments);
  _assert(
      commentArg(2, arguments), var1 !== var2,
      'Expected not to be ' + _displayStringForValue(var2));
};
/** @const */
var assertNotEquals = goog.testing.asserts.assertNotEquals;

/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNull = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar === null,
      goog.testing.asserts.getDefaultErrorMsg_(null, aVar));
};
/** @const */
var assertNull = goog.testing.asserts.assertNull;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNotNull = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar !== null,
      'Expected not to be ' + _displayStringForValue(null));
};
/** @const */
var assertNotNull = goog.testing.asserts.assertNotNull;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertUndefined = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar === JSUNIT_UNDEFINED_VALUE,
      goog.testing.asserts.getDefaultErrorMsg_(JSUNIT_UNDEFINED_VALUE, aVar));
};
/** @const */
var assertUndefined = goog.testing.asserts.assertUndefined;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNotUndefined = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar !== JSUNIT_UNDEFINED_VALUE,
      'Expected not to be ' + _displayStringForValue(JSUNIT_UNDEFINED_VALUE));
};
/** @const */
var assertNotUndefined = goog.testing.asserts.assertNotUndefined;

/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNullOrUndefined = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar == null,
      'Expected ' + _displayStringForValue(null) + ' or ' +
          _displayStringForValue(JSUNIT_UNDEFINED_VALUE) + ' but was ' +
          _displayStringForValue(aVar));
};
/** @const */
var assertNullOrUndefined = goog.testing.asserts.assertNullOrUndefined;

/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNotNullNorUndefined = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  goog.testing.asserts.assertNotNull.apply(null, arguments);
  goog.testing.asserts.assertNotUndefined.apply(null, arguments);
};
/** @const */
var assertNotNullNorUndefined = goog.testing.asserts.assertNotNullNorUndefined;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNonEmptyString = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar !== JSUNIT_UNDEFINED_VALUE &&
          aVar !== null && typeof aVar == 'string' && aVar !== '',
      'Expected non-empty string but was ' + _displayStringForValue(aVar));
};
/** @const */
var assertNonEmptyString = goog.testing.asserts.assertNonEmptyString;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNaN = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(
      commentArg(1, arguments), aVar !== aVar,
      'Expected NaN but was ' + _displayStringForValue(aVar));
};
/** @const */
var assertNaN = goog.testing.asserts.assertNaN;


/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertNotNaN = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var aVar = nonCommentArg(1, 1, arguments);
  _assert(commentArg(1, arguments), !isNaN(aVar), 'Expected not NaN');
};
/** @const */
var assertNotNaN = goog.testing.asserts.assertNotNaN;


/**
 * The return value of the equality predicate passed to findDifferences below,
 * in cases where the predicate can't test the input variables for equality.
 * @type {?string}
 */
goog.testing.asserts.EQUALITY_PREDICATE_CANT_PROCESS = null;


/**
 * The return value of the equality predicate passed to findDifferences below,
 * in cases where the input vriables are equal.
 * @type {?string}
 */
goog.testing.asserts.EQUALITY_PREDICATE_VARS_ARE_EQUAL = '';


/**
 * @const {!Object<string, boolean>}
 */
goog.testing.asserts.ARRAY_TYPES = {
  'Array': true,
  'Float32Array': true,
  'Float64Array': true,
  'Int8Array': true,
  'Int16Array': true,
  'Int32Array': true,
  'Uint8Array': true,
  'Uint8ClampedArray': true,
  'Uint16Array': true,
  'Uint32Array': true,
  'BigInt64Array': true,
  'BigUint64Array': true
};

/**
 * Determines if two items of any type match, and formulates an error message
 * if not.
 * @param {*} expected Expected argument to match.
 * @param {*} actual Argument as a result of performing the test.
 * @param {(function(string, *, *): ?string)=} opt_equalityPredicate An optional
 *     function that can be used to check equality of variables. It accepts 3
 *     arguments: type-of-variables, var1, var2 (in that order) and returns an
 *     error message if the variables are not equal,
 *     goog.testing.asserts.EQUALITY_PREDICATE_VARS_ARE_EQUAL if the variables
 *     are equal, or
 *     goog.testing.asserts.EQUALITY_PREDICATE_CANT_PROCESS if the predicate
 *     couldn't check the input variables. The function will be called only if
 *     the types of var1 and var2 are identical.
 * @return {?string} Null on success, error message on failure.
 */
goog.testing.asserts.findDifferences = function(
    expected, actual, opt_equalityPredicate) {
  'use strict';
  var failures = [];
  // True if there a generic error at the root (with no path).  If so, we should
  // fail, but not add to the failures array (because it will be included at the
  // top anyway).
  var rootFailed = false;
  var seen1 = [];
  var seen2 = [];

  // To avoid infinite recursion when the two parameters are self-referential
  // along the same path of properties, keep track of the object pairs already
  // seen in this call subtree, and abort when a cycle is detected.
  function innerAssertWithCycleCheck(var1, var2, path) {
    // This is used for testing, so we can afford to be slow (but more
    // accurate). So we just check whether var1 is in seen1. If we
    // found var1 in index i, we simply need to check whether var2 is
    // in seen2[i]. If it is, we do not recurse to check var1/var2. If
    // it isn't, we know that the structures of the two objects must be
    // different.
    //
    // This is based on the fact that values at index i in seen1 and
    // seen2 will be checked for equality eventually (when
    // innerAssertImplementation(seen1[i], seen2[i], path) finishes).
    for (var i = 0; i < seen1.length; ++i) {
      var match1 = seen1[i] === var1;
      var match2 = seen2[i] === var2;
      if (match1 || match2) {
        if (!match1 || !match2) {
          // Asymmetric cycles, so the objects have different structure.
          failures.push('Asymmetric cycle detected at ' + path);
        }
        return;
      }
    }

    seen1.push(var1);
    seen2.push(var2);
    innerAssertImplementation(var1, var2, path);
    seen1.pop();
    seen2.pop();
  }

  const equalityPredicate =
      opt_equalityPredicate || function(type, var1, var2) {
        'use strict';
        const typedPredicate = EQUALITY_PREDICATES[type];
        if (!typedPredicate) {
          return goog.testing.asserts.EQUALITY_PREDICATE_CANT_PROCESS;
        }
        const equal = typedPredicate(var1, var2);
        return equal ? goog.testing.asserts.EQUALITY_PREDICATE_VARS_ARE_EQUAL :
                       goog.testing.asserts.getDefaultErrorMsg_(var1, var2);
      };

  /**
   * @param {*} var1 An item in the expected object.
   * @param {*} var2 The corresponding item in the actual object.
   * @param {string} path Their path in the objects.
   * @suppress {missingProperties} The map_ property is unknown to the compiler
   *     unless goog.structs.Map is loaded.
   */
  function innerAssertImplementation(var1, var2, path) {
    if (var1 === var2) {
      return;
    }

    var typeOfVar1 = _trueTypeOf(var1);
    var typeOfVar2 = _trueTypeOf(var2);

    if (typeOfVar1 === typeOfVar2) {
      const isArrayBuffer = typeOfVar1 === 'ArrayBuffer';
      if (isArrayBuffer) {
        // Since ArrayBuffer instances can't themselves be iterated through,
        // compare 1-byte-per-element views of them.
        var1 = new Uint8Array(/** @type {!ArrayBuffer} */ (var1));
        var2 = new Uint8Array(/** @type {!ArrayBuffer} */ (var2));
      }
      const isArray =
          isArrayBuffer || goog.testing.asserts.ARRAY_TYPES[typeOfVar1];
      var errorMessage = equalityPredicate(typeOfVar1, var1, var2);
      if (errorMessage !=
          goog.testing.asserts.EQUALITY_PREDICATE_CANT_PROCESS) {
        if (errorMessage !=
            goog.testing.asserts.EQUALITY_PREDICATE_VARS_ARE_EQUAL) {
          if (path) {
            failures.push(path + ': ' + errorMessage);
          } else {
            rootFailed = true;
          }
        }
      } else if (isArray && var1.length != var2.length) {
        failures.push(
            (path ? path + ': ' : '') + 'Expected ' + var1.length +
            '-element array ' +
            'but got a ' + var2.length + '-element array');
      } else if (typeOfVar1 == 'String') {
        // If the comparer cannot process strings (eg, roughlyEquals).
        if (var1 != var2) {
          if (path) {
            failures.push(
                path + ': ' +
                goog.testing.asserts.getDefaultErrorMsg_(var1, var2));
          } else {
            rootFailed = true;
          }
        }
      } else {
        var childPath = path + (isArray ? '[%s]' : (path ? '.%s' : '%s'));
        // These type checks do not use _trueTypeOf because that does not work
        // for polyfilled Map/Set. Note that these checks may potentially fail
        // if var1 comes from a different window.
        if ((typeof Map != 'undefined' && var1 instanceof Map) ||
            (typeof Set != 'undefined' && var1 instanceof Set)) {
          var1.forEach(function(value, key) {
            'use strict';
            if (var2.has(key)) {
              // For a map, the values must be compared, but with Set, checking
              // that the second set contains the first set's "keys" is
              // sufficient.
              if (var2.get) {
                innerAssertWithCycleCheck(
                    // NOTE: replace will call functions, so stringify eagerly.
                    value, var2.get(key), childPath.replace('%s', String(key)));
              }
            } else {
              failures.push(
                  key + ' not present in actual ' + (path || typeOfVar2));
            }
          });

          var2.forEach(function(value, key) {
            'use strict';
            if (!var1.has(key)) {
              failures.push(
                  key + ' not present in expected ' + (path || typeOfVar1));
            }
          });
        } else if (!var1['__iterator__']) {
          // if an object has an __iterator__ property, we have no way of
          // actually inspecting its raw properties, and JS 1.7 doesn't
          // overload [] to make it possible for someone to generically
          // use what the iterator returns to compare the object-managed
          // properties. This gets us into deep poo with things like
          // goog.structs.Map, at least on systems that support iteration.
          for (var prop in var1) {
            if (isArray && goog.testing.asserts.isArrayIndexProp_(prop)) {
              // Skip array indices for now. We'll handle them later.
              continue;
            }

            if (prop in var2) {
              innerAssertWithCycleCheck(
                  var1[prop], var2[prop], childPath.replace('%s', prop));
            } else {
              failures.push(
                  'property ' + prop + ' not present in actual ' +
                  (path || typeOfVar2));
            }
          }
          // make sure there aren't properties in var2 that are missing
          // from var1. if there are, then by definition they don't
          // match.
          for (var prop in var2) {
            if (isArray && goog.testing.asserts.isArrayIndexProp_(prop)) {
              // Skip array indices for now. We'll handle them later.
              continue;
            }

            if (!(prop in var1)) {
              failures.push(
                  'property ' + prop + ' not present in expected ' +
                  (path || typeOfVar1));
            }
          }

          // Handle array indices by iterating from 0 to arr.length.
          //
          // Although all browsers allow holes in arrays, browsers
          // are inconsistent in what they consider a hole. For example,
          // "[0,undefined,2]" has a hole on IE but not on Firefox.
          //
          // Because our style guide bans for...in iteration over arrays,
          // we assume that most users don't care about holes in arrays,
          // and that it is ok to say that a hole is equivalent to a slot
          // populated with 'undefined'.
          if (isArray) {
            for (prop = 0; prop < var1.length; prop++) {
              innerAssertWithCycleCheck(
                  var1[prop], var2[prop],
                  childPath.replace('%s', String(prop)));
            }
          }
        } else {
          // special-case for closure objects that have iterators
          if (typeof var1.equals === 'function') {
            // use the object's own equals function, assuming it accepts an
            // object and returns a boolean
            if (!var1.equals(var2)) {
              failures.push(
                  'equals() returned false for ' + (path || typeOfVar1));
            }
          } else if (var1.map_) {
            // assume goog.structs.Map or goog.structs.Set, where comparing
            // their private map_ field is sufficient
            innerAssertWithCycleCheck(
                var1.map_, var2.map_, childPath.replace('%s', 'map_'));
          } else {
            // else die, so user knows we can't do anything
            failures.push(
                'unable to check ' + (path || typeOfVar1) +
                ' for equality: it has an iterator we do not ' +
                'know how to handle. please add an equals method');
          }
        }
      }
    } else if (path) {
      failures.push(
          path + ': ' + goog.testing.asserts.getDefaultErrorMsg_(var1, var2));
    } else {
      rootFailed = true;
    }
  }

  innerAssertWithCycleCheck(expected, actual, '');

  if (rootFailed) {
    return goog.testing.asserts.getDefaultErrorMsg_(expected, actual);
  }
  return failures.length == 0 ? null : goog.testing.asserts.getDefaultErrorMsg_(
                                           expected, actual) +
          '\n   ' + failures.join('\n   ');
};


/**
 * Notes:
 * Object equality has some nasty browser quirks, and this implementation is
 * not 100% correct. For example,
 *
 * <code>
 * var a = [0, 1, 2];
 * var b = [0, 1, 2];
 * delete a[1];
 * b[1] = undefined;
 * assertObjectEquals(a, b); // should fail, but currently passes
 * </code>
 *
 * See asserts_test.html for more interesting edge cases.
 *
 * The first comparison object provided is the expected value, the second is
 * the actual.
 *
 * @param {*} a Assertion message or comparison object.
 * @param {*} b Comparison object.
 * @param {*=} opt_c Comparison object, if an assertion message was provided.
 */
goog.testing.asserts.assertObjectEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var v1 = nonCommentArg(1, 2, arguments);
  var v2 = nonCommentArg(2, 2, arguments);
  var failureMessage = commentArg(2, arguments) ? commentArg(2, arguments) : '';
  var differences = goog.testing.asserts.findDifferences(v1, v2);

  _assert(failureMessage, !differences, differences);
};
/** @const */
var assertObjectEquals = goog.testing.asserts.assertObjectEquals;


/**
 * Similar to assertObjectEquals above, but accepts a tolerance margin.
 *
 * @param {*} a Assertion message or comparison object.
 * @param {*} b Comparison object.
 * @param {*} c Comparison object or tolerance.
 * @param {*=} opt_d Tolerance, if an assertion message was provided.
 */
goog.testing.asserts.assertObjectRoughlyEquals = function(a, b, c, opt_d) {
  'use strict';
  _validateArguments(3, arguments);
  var v1 = nonCommentArg(1, 3, arguments);
  var v2 = nonCommentArg(2, 3, arguments);
  var tolerance = nonCommentArg(3, 3, arguments);
  var failureMessage = commentArg(3, arguments) ? commentArg(3, arguments) : '';
  var equalityPredicate = function(type, var1, var2) {
    'use strict';
    var typedPredicate =
        goog.testing.asserts.primitiveRoughEqualityPredicates_[type];
    if (!typedPredicate) {
      return goog.testing.asserts.EQUALITY_PREDICATE_CANT_PROCESS;
    }
    var equal = typedPredicate(var1, var2, tolerance);
    return equal ? goog.testing.asserts.EQUALITY_PREDICATE_VARS_ARE_EQUAL :
                   goog.testing.asserts.getDefaultErrorMsg_(var1, var2) +
            ' which was more than ' + tolerance + ' away';
  };
  var differences =
      goog.testing.asserts.findDifferences(v1, v2, equalityPredicate);

  _assert(failureMessage, !differences, differences);
};
/** @const */
var assertObjectRoughlyEquals = goog.testing.asserts.assertObjectRoughlyEquals;

/**
 * Compares two arbitrary objects for non-equalness.
 *
 * All the same caveats as for assertObjectEquals apply here:
 * Undefined values may be confused for missing values, or vice versa.
 *
 * @param {*} a Assertion message or comparison object.
 * @param {*} b Comparison object.
 * @param {*=} opt_c Comparison object, if an assertion message was provided.
 */
goog.testing.asserts.assertObjectNotEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var v1 = nonCommentArg(1, 2, arguments);
  var v2 = nonCommentArg(2, 2, arguments);
  var failureMessage = commentArg(2, arguments) ? commentArg(2, arguments) : '';
  var differences = goog.testing.asserts.findDifferences(v1, v2);

  _assert(failureMessage, differences, 'Objects should not be equal');
};
/** @const */
var assertObjectNotEquals = goog.testing.asserts.assertObjectNotEquals;


/**
 * Compares two arrays ignoring negative indexes and extra properties on the
 * array objects. Use case: Internet Explorer adds the index, lastIndex and
 * input enumerable fields to the result of string.match(/regexp/g), which makes
 * assertObjectEquals fail.
 * @param {*} a The expected array (2 args) or the debug message (3 args).
 * @param {*} b The actual array (2 args) or the expected array (3 args).
 * @param {*=} opt_c The actual array (3 args only).
 */
goog.testing.asserts.assertArrayEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var v1 = nonCommentArg(1, 2, arguments);
  var v2 = nonCommentArg(2, 2, arguments);
  var failureMessage = commentArg(2, arguments) ? commentArg(2, arguments) : '';

  var typeOfVar1 = _trueTypeOf(v1);
  _assert(
      failureMessage, typeOfVar1 == 'Array',
      'Expected an array for assertArrayEquals but found a ' + typeOfVar1);

  var typeOfVar2 = _trueTypeOf(v2);
  _assert(
      failureMessage, typeOfVar2 == 'Array',
      'Expected an array for assertArrayEquals but found a ' + typeOfVar2);

  goog.testing.asserts.assertObjectEquals(
      failureMessage, Array.prototype.concat.call(v1),
      Array.prototype.concat.call(v2));
};
/** @const */
var assertArrayEquals = goog.testing.asserts.assertArrayEquals;


/**
 * Compares two objects that can be accessed like an array and assert that
 * each element is equal.
 * @param {string|Object} a Failure message (3 arguments)
 *     or object #1 (2 arguments).
 * @param {Object} b Object #2 (2 arguments) or object #1 (3 arguments).
 * @param {Object=} opt_c Object #2 (3 arguments).
 */
goog.testing.asserts.assertElementsEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);

  var v1 = nonCommentArg(1, 2, arguments);
  var v2 = nonCommentArg(2, 2, arguments);
  var failureMessage = commentArg(2, arguments) ? commentArg(2, arguments) : '';

  if (!v1) {
    goog.testing.asserts.assert(failureMessage, !v2);
  } else {
    goog.testing.asserts.assertEquals(
        'length mismatch: ' + failureMessage, v1.length, v2.length);
    for (var i = 0; i < v1.length; ++i) {
      goog.testing.asserts.assertEquals(
          'mismatch at index ' + i + ': ' + failureMessage, v1[i], v2[i]);
    }
  }
};
/** @const */
var assertElementsEquals = goog.testing.asserts.assertElementsEquals;


/**
 * Compares two objects that can be accessed like an array and assert that
 * each element is roughly equal.
 * @param {string|Object} a Failure message (4 arguments)
 *     or object #1 (3 arguments).
 * @param {Object} b Object #1 (4 arguments) or object #2 (3 arguments).
 * @param {Object|number} c Object #2 (4 arguments) or tolerance (3 arguments).
 * @param {number=} opt_d tolerance (4 arguments).
 */
goog.testing.asserts.assertElementsRoughlyEqual = function(a, b, c, opt_d) {
  'use strict';
  _validateArguments(3, arguments);

  var v1 = nonCommentArg(1, 3, arguments);
  var v2 = nonCommentArg(2, 3, arguments);
  var tolerance = nonCommentArg(3, 3, arguments);
  var failureMessage = commentArg(3, arguments) ? commentArg(3, arguments) : '';

  if (!v1) {
    goog.testing.asserts.assert(failureMessage, !v2);
  } else {
    goog.testing.asserts.assertEquals(
        'length mismatch: ' + failureMessage, v1.length, v2.length);
    for (var i = 0; i < v1.length; ++i) {
      goog.testing.asserts.assertRoughlyEquals(
          failureMessage, v1[i], v2[i], tolerance);
    }
  }
};
/** @const */
var assertElementsRoughlyEqual =
    goog.testing.asserts.assertElementsRoughlyEqual;

/**
 * Compares elements of two array-like or iterable objects using strict equality
 * without taking their order into account.
 * @param {string|!IArrayLike|!Iterable} a Assertion message or the
 *     expected elements.
 * @param {!IArrayLike|!Iterable} b Expected elements or the actual
 *     elements.
 * @param {!IArrayLike|!Iterable=} opt_c Actual elements.
 */
goog.testing.asserts.assertSameElements = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var expected = nonCommentArg(1, 2, arguments);
  var actual = nonCommentArg(2, 2, arguments);
  var message = commentArg(2, arguments);

  goog.testing.asserts.assertTrue(
      'Value of \'expected\' should be array-like or iterable',
      goog.testing.asserts.isArrayLikeOrIterable_(expected));

  goog.testing.asserts.assertTrue(
      'Value of \'actual\' should be array-like or iterable',
      goog.testing.asserts.isArrayLikeOrIterable_(actual));

  // Clones expected and actual and converts them to real arrays.
  expected = goog.testing.asserts.toArray_(expected);
  actual = goog.testing.asserts.toArray_(actual);
  // TODO(user): It would be great to show only the difference
  // between the expected and actual elements.
  _assert(
      message, expected.length == actual.length, 'Expected ' + expected.length +
          ' elements: [' + expected + '], ' +
          'got ' + actual.length + ' elements: [' + actual + ']');

  var toFind = goog.testing.asserts.toArray_(expected);
  for (var i = 0; i < actual.length; i++) {
    var index = goog.testing.asserts.indexOf_(toFind, actual[i]);
    _assert(
        message, index != -1,
        'Expected [' + expected + '], got [' + actual + ']');
    toFind.splice(index, 1);
  }
};
/** @const */
var assertSameElements = goog.testing.asserts.assertSameElements;

/**
 * @param {*} obj Object to test.
 * @return {boolean} Whether given object is array-like or iterable.
 * @private
 */
goog.testing.asserts.isArrayLikeOrIterable_ = function(obj) {
  'use strict';
  return goog.isArrayLike(obj) || goog.testing.asserts.isIterable_(obj);
};

/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertEvaluatesToTrue = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var value = nonCommentArg(1, 1, arguments);
  if (!value) {
    _assert(commentArg(1, arguments), false, 'Expected to evaluate to true');
  }
};
/** @const */
var assertEvaluatesToTrue = goog.testing.asserts.assertEvaluatesToTrue;

/**
 * @param {*} a The value to assert (1 arg) or debug message (2 args).
 * @param {*=} opt_b The value to assert (2 args only).
 */
goog.testing.asserts.assertEvaluatesToFalse = function(a, opt_b) {
  'use strict';
  _validateArguments(1, arguments);
  var value = nonCommentArg(1, 1, arguments);
  if (value) {
    _assert(commentArg(1, arguments), false, 'Expected to evaluate to false');
  }
};
/** @const */
var assertEvaluatesToFalse = goog.testing.asserts.assertEvaluatesToFalse;

/**
 * Compares two HTML snippets.
 *
 * Take extra care if attributes are involved. `assertHTMLEquals`'s
 * implementation isn't prepared for complex cases. For example, the following
 * comparisons erroneously fail:
 * <pre>
 * assertHTMLEquals('<a href="x" target="y">', '<a target="y" href="x">');
 * assertHTMLEquals('<div class="a b">', '<div class="b a">');
 * assertHTMLEquals('<input disabled>', '<input disabled="disabled">');
 * </pre>
 *
 * When in doubt, use `goog.testing.dom.assertHtmlMatches`.
 *
 * @param {*} a The expected value (2 args) or the debug message (3 args).
 * @param {*} b The actual value (2 args) or the expected value (3 args).
 * @param {*=} opt_c The actual value (3 args only).
 */
goog.testing.asserts.assertHTMLEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var var1 = nonCommentArg(1, 2, arguments);
  var var2 = nonCommentArg(2, 2, arguments);
  var var1Standardized = standardizeHTML(var1);
  var var2Standardized = standardizeHTML(var2);

  _assert(
      commentArg(2, arguments), var1Standardized === var2Standardized,
      goog.testing.asserts.getDefaultErrorMsg_(
          var1Standardized, var2Standardized));
};
/** @const */
var assertHTMLEquals = goog.testing.asserts.assertHTMLEquals;


/**
 * Compares two CSS property values to make sure that they represent the same
 * things. This will normalize values in the browser. For example, in Firefox,
 * this assertion will consider "rgb(0, 0, 255)" and "#0000ff" to be identical
 * values for the "color" property. This function won't normalize everything --
 * for example, in most browsers, "blue" will not match "#0000ff". It is
 * intended only to compensate for unexpected normalizations performed by
 * the browser that should also affect your expected value.
 * @param {string} a Assertion message, or the CSS property name.
 * @param {string} b CSS property name, or the expected value.
 * @param {string} c The expected value, or the actual value.
 * @param {string=} opt_d The actual value.
 */
goog.testing.asserts.assertCSSValueEquals = function(a, b, c, opt_d) {
  'use strict';
  _validateArguments(3, arguments);
  var propertyName = nonCommentArg(1, 3, arguments);
  var expectedValue = nonCommentArg(2, 3, arguments);
  var actualValue = nonCommentArg(3, 3, arguments);
  var expectedValueStandardized =
      standardizeCSSValue(propertyName, expectedValue);
  var actualValueStandardized = standardizeCSSValue(propertyName, actualValue);

  _assert(
      commentArg(3, arguments),
      expectedValueStandardized == actualValueStandardized,
      goog.testing.asserts.getDefaultErrorMsg_(
          expectedValueStandardized, actualValueStandardized));
};
/** @const */
var assertCSSValueEquals = goog.testing.asserts.assertCSSValueEquals;


/**
 * @param {*} a The expected value (2 args) or the debug message (3 args).
 * @param {*} b The actual value (2 args) or the expected value (3 args).
 * @param {*=} opt_c The actual value (3 args only).
 */
goog.testing.asserts.assertHashEquals = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var var1 = nonCommentArg(1, 2, arguments);
  var var2 = nonCommentArg(2, 2, arguments);
  var message = commentArg(2, arguments);
  for (var key in var1) {
    _assert(
        message, key in var2,
        'Expected hash had key ' + key + ' that was not found');
    _assert(
        message, var1[key] == var2[key], 'Value for key ' + key +
            ' mismatch - expected = ' + var1[key] + ', actual = ' + var2[key]);
  }

  for (var key in var2) {
    _assert(
        message, key in var1,
        'Actual hash had key ' + key + ' that was not expected');
  }
};
/** @const */
var assertHashEquals = goog.testing.asserts.assertHashEquals;


/**
 * @param {*} a The expected value (3 args) or the debug message (4 args).
 * @param {*} b The actual value (3 args) or the expected value (4 args).
 * @param {*} c The tolerance (3 args) or the actual value (4 args).
 * @param {*=} opt_d The tolerance (4 args only).
 */
goog.testing.asserts.assertRoughlyEquals = function(a, b, c, opt_d) {
  'use strict';
  _validateArguments(3, arguments);
  var expected = nonCommentArg(1, 3, arguments);
  var actual = nonCommentArg(2, 3, arguments);
  var tolerance = nonCommentArg(3, 3, arguments);
  _assert(
      commentArg(3, arguments),
      goog.testing.asserts.numberRoughEqualityPredicate_(
          expected, actual, tolerance),
      'Expected ' + expected + ', but got ' + actual + ' which was more than ' +
          tolerance + ' away');
};
/** @const */
var assertRoughlyEquals = goog.testing.asserts.assertRoughlyEquals;


/**
 * Checks if the test value is included in the given container. The container
 * can be a string (where "included" means a substring), an array or any
 *  `IArrayLike` (where "included" means a member), or any type implementing
 * `indexOf` with similar semantics (returning -1 for not included).
 *
 * @param {*} a Failure message (3 arguments) or the test value
 *     (2 arguments).
 * @param {*} b The test value (3 arguments) or the container
 *     (2 arguments).
 * @param {*=} opt_c The container.
 */
goog.testing.asserts.assertContains = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var contained = nonCommentArg(1, 2, arguments);
  var container = nonCommentArg(2, 2, arguments);
  _assert(
      commentArg(2, arguments),
      goog.testing.asserts.contains_(container, contained),
      'Expected \'' + container + '\' to contain \'' + contained + '\'');
};
/** @const */
var assertContains = goog.testing.asserts.assertContains;

/**
 * Checks if the test value is not included in the given container. The
 * container can be a string (where "included" means a substring), an array or
 * any `IArrayLike` (where "included" means a member), or any type implementing
 * `indexOf` with similar semantics (returning -1 for not included).
 * @param {*} a Failure message (3 arguments) or the contained element
 *     (2 arguments).
 * @param {*} b The contained element (3 arguments) or the container
 *     (2 arguments).
 * @param {*=} opt_c The container.
 */
goog.testing.asserts.assertNotContains = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var contained = nonCommentArg(1, 2, arguments);
  var container = nonCommentArg(2, 2, arguments);
  _assert(
      commentArg(2, arguments),
      !goog.testing.asserts.contains_(container, contained),
      'Expected \'' + container + '\' not to contain \'' + contained + '\'');
};
/** @const */
var assertNotContains = goog.testing.asserts.assertNotContains;


/**
 * Checks if the given string matches the given regular expression.
 * @param {*} a Failure message (3 arguments) or the expected regular
 *     expression as a string or RegExp (2 arguments).
 * @param {*} b The regular expression (3 arguments) or the string to test
 *     (2 arguments).
 * @param {*=} opt_c The string to test.
 */
goog.testing.asserts.assertRegExp = function(a, b, opt_c) {
  'use strict';
  _validateArguments(2, arguments);
  var regexp = nonCommentArg(1, 2, arguments);
  var string = nonCommentArg(2, 2, arguments);
  if (typeof(regexp) == 'string') {
    regexp = new RegExp(regexp);
  }
  _assert(
      commentArg(2, arguments), regexp.test(string),
      'Expected \'' + string + '\' to match RegExp ' + regexp.toString());
};
/** @const */
var assertRegExp = goog.testing.asserts.assertRegExp;


/**
 * Converts an array-like or iterable object to an array (clones it if it's
 * already an array).
 * @param {!Iterable|!IArrayLike} obj The collection object.
 * @return {!Array<?>} Copy of the collection as array.
 * @private
 */
goog.testing.asserts.toArray_ = function(obj) {
  'use strict';
  var ret = [];
  if (goog.testing.asserts.isIterable_(obj)) {
    var iterator =
        goog.testing.asserts.getIterator_(/** @type {!Iterable} */ (obj));

    // Cannot use for..of syntax here as ES6 syntax is not available in Closure.
    // See b/117231092
    while (true) {
      var result = iterator.next();
      if (result.done) {
        return ret;
      }
      ret.push(result.value);
    }
  }

  for (var i = 0; i < obj.length; i++) {
    ret[i] = obj[i];
  }
  return ret;
};

// TODO(nnaze): Consider moving isIterable_ and getIterator_ functionality
// into goog.iter.es6. See discussion in cl/217356297.

/**
 * @param {*} obj
 * @return {boolean} Whether the object is iterable (JS iterator protocol).
 * @private
 */
goog.testing.asserts.isIterable_ = function(obj) {
  'use strict';
  return !!(
      typeof Symbol !== 'undefined' && Symbol.iterator && obj[Symbol.iterator]);
};

/**
 * @param {!Iterable} iterable
 * @return {!Iterator} An iterator for obj.
 * @throws {!goog.testing.JsUnitException} If the given object is not iterable.
 * @private
 */
goog.testing.asserts.getIterator_ = function(iterable) {
  'use strict';
  if (!goog.testing.asserts.isIterable_(iterable)) {
    goog.testing.asserts.raiseException('parameter iterable is not iterable');
  }

  return iterable[Symbol.iterator]();
};


/**
 * Finds the position of the first occurrence of an element in a container.
 * @param {IArrayLike<?>|{indexOf: function(*): number}} container
 *     The array to find the element in.
 * @param {*} contained Element to find.
 * @return {number} Index of the first occurrence or -1 if not found.
 * @private
 */
goog.testing.asserts.indexOf_ = function(container, contained) {
  'use strict';
  if (typeof container.indexOf == 'function') {
    return container.indexOf(contained);
  } else {
    // IE6/7 do not have indexOf so do a search.
    for (var i = 0; i < container.length; i++) {
      if (container[i] === contained) {
        return i;
      }
    }
    return -1;
  }
};


/**
 * Tells whether the array contains the given element.
 * @param {IArrayLike<?>|{indexOf: function(*): number}} container The array to
 *     find the element in.
 * @param {*} contained Element to find.
 * @return {boolean} Whether the element is in the array.
 * @private
 */
goog.testing.asserts.contains_ = function(container, contained) {
  'use strict';
  // TODO(user): Can we check for container.contains as well?
  // That would give us support for most goog.structs (though weird results
  // with anything else with a contains method, like goog.math.Range). Falling
  // back with container.some would catch all iterables, too.
  return goog.testing.asserts.indexOf_(container, contained) != -1;
};

var standardizeHTML = function(html) {
  'use strict';
  var translator = document.createElement('div');

  goog.dom.safe.setInnerHtml(
      translator,
      goog.html.uncheckedconversions
          .safeHtmlFromStringKnownToSatisfyTypeContract(
              goog.string.Const.from('HTML is never attached to DOM'), html));

  // Trim whitespace from result (without relying on goog.string)
  return translator.innerHTML.replace(/^\s+|\s+$/g, '');
};


/**
 * Standardizes a CSS value for a given property by applying it to an element
 * and then reading it back.
 * @param {string} propertyName CSS property name.
 * @param {string} value CSS value.
 * @return {string} Normalized CSS value.
 */
var standardizeCSSValue = function(propertyName, value) {
  'use strict';
  var styleDeclaration = document.createElement('div').style;
  styleDeclaration[propertyName] = value;
  return styleDeclaration[propertyName];
};


/**
 * Raises a JsUnit exception with the given comment. If the exception is
 * unexpectedly caught during a unit test, it will be rethrown so that it is
 * seen by the test framework.
 * @param {string} comment A summary for the exception.
 * @param {string=} opt_message A description of the exception.
 */
goog.testing.asserts.raiseException = function(comment, opt_message) {
  'use strict';
  var e = new goog.testing.JsUnitException(comment, opt_message);

  var testCase = _getCurrentTestCase();
  if (testCase) {
    testCase.raiseAssertionException(e);
  } else {
    goog.global.console.error(
        'Failed to save thrown exception: no test case is installed.');
    throw e;
  }
};


/**
 * Helper function for assertObjectEquals.
 * @param {string} prop A property name.
 * @return {boolean} If the property name is an array index.
 * @private
 */
goog.testing.asserts.isArrayIndexProp_ = function(prop) {
  'use strict';
  return prop === '0' || /^[1-9][0-9]*$/.test(prop);
};

/** @define {boolean} */
goog.EXPORT_ASSERTIONS = goog.define('goog.EXPORT_ASSERTIONS', true);
/*
 * These symbols are both exported in the global namespace (for legacy
 * reasons) and as part of the goog.testing.asserts namespace. Although they
 * can be used globally in tests, these symbols are allowed to be imported for
 * cleaner typing.
 */
if (goog.EXPORT_ASSERTIONS) {
  goog.exportSymbol('fail', fail);
  goog.exportSymbol('assert', assert);
  goog.exportSymbol('assertThrows', assertThrows);
  goog.exportSymbol('assertNotThrows', assertNotThrows);
  goog.exportSymbol('assertThrowsJsUnitException', assertThrowsJsUnitException);
  goog.exportSymbol('assertRejects', assertRejects);
  goog.exportSymbol('assertTrue', assertTrue);
  goog.exportSymbol('assertFalse', assertFalse);
  goog.exportSymbol('assertEquals', assertEquals);
  goog.exportSymbol('assertNotEquals', assertNotEquals);
  goog.exportSymbol('assertNull', assertNull);
  goog.exportSymbol('assertNotNull', assertNotNull);
  goog.exportSymbol('assertUndefined', assertUndefined);
  goog.exportSymbol('assertNotUndefined', assertNotUndefined);
  goog.exportSymbol('assertNullOrUndefined', assertNullOrUndefined);
  goog.exportSymbol('assertNotNullNorUndefined', assertNotNullNorUndefined);
  goog.exportSymbol('assertNonEmptyString', assertNonEmptyString);
  goog.exportSymbol('assertNaN', assertNaN);
  goog.exportSymbol('assertNotNaN', assertNotNaN);
  goog.exportSymbol('assertObjectEquals', assertObjectEquals);
  goog.exportSymbol('assertObjectRoughlyEquals', assertObjectRoughlyEquals);
  goog.exportSymbol('assertObjectNotEquals', assertObjectNotEquals);
  goog.exportSymbol('assertArrayEquals', assertArrayEquals);
  goog.exportSymbol('assertElementsEquals', assertElementsEquals);
  goog.exportSymbol('assertElementsRoughlyEqual', assertElementsRoughlyEqual);
  goog.exportSymbol('assertSameElements', assertSameElements);
  goog.exportSymbol('assertEvaluatesToTrue', assertEvaluatesToTrue);
  goog.exportSymbol('assertEvaluatesToFalse', assertEvaluatesToFalse);
  goog.exportSymbol('assertHTMLEquals', assertHTMLEquals);
  goog.exportSymbol('assertHashEquals', assertHashEquals);
  goog.exportSymbol('assertRoughlyEquals', assertRoughlyEquals);
  goog.exportSymbol('assertContains', assertContains);
  goog.exportSymbol('assertNotContains', assertNotContains);
  goog.exportSymbol('assertRegExp', assertRegExp);
}