/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ const lazy = {}; ChromeUtils.defineESModuleGetters(lazy, { AppInfo: "chrome://remote/content/shared/AppInfo.sys.mjs", error: "chrome://remote/content/shared/webdriver/Errors.sys.mjs", pprint: "chrome://remote/content/shared/Format.sys.mjs", }); /** * Shorthands for common assertions made in WebDriver. * * @namespace */ export const assert = {}; /** * Asserts that WebDriver has an active session. * * @param {WebDriverSession} session * WebDriver session instance. * @param {string=} msg * Custom error message. * * @throws {InvalidSessionIDError} * If session does not exist, or has an invalid id. */ assert.session = function (session, msg = "") { msg = msg || "WebDriver session does not exist, or is not active"; assert.that( session => session && typeof session.id == "string", msg, lazy.error.InvalidSessionIDError )(session); }; /** * Asserts that the current browser is Firefox Desktop. * * @param {string=} msg * Custom error message. * * @throws {UnsupportedOperationError} * If current browser is not Firefox. */ assert.firefox = function (msg = "") { msg = msg || "Only supported in Firefox"; assert.that( isFirefox => isFirefox, msg, lazy.error.UnsupportedOperationError )(lazy.AppInfo.isFirefox); }; /** * Asserts that the current application is Firefox Desktop or Thunderbird. * * @param {string=} msg * Custom error message. * * @throws {UnsupportedOperationError} * If current application is not running on desktop. */ assert.desktop = function (msg = "") { msg = msg || "Only supported in desktop applications"; assert.that( isDesktop => isDesktop, msg, lazy.error.UnsupportedOperationError )(!lazy.AppInfo.isAndroid); }; /** * Asserts that the current application runs on Android. * * @param {string=} msg * Custom error message. * * @throws {UnsupportedOperationError} * If current application is not running on Android. */ assert.mobile = function (msg = "") { msg = msg || "Only supported on Android"; assert.that( isAndroid => isAndroid, msg, lazy.error.UnsupportedOperationError )(lazy.AppInfo.isAndroid); }; /** * Asserts that the current context is content. * * @param {string} context * Context to test. * @param {string=} msg * Custom error message. * * @returns {string} * context is returned unaltered. * * @throws {UnsupportedOperationError} * If context is not content. */ assert.content = function (context, msg = "") { msg = msg || "Only supported in content context"; assert.that( c => c.toString() == "content", msg, lazy.error.UnsupportedOperationError )(context); }; /** * Asserts that the {@link CanonicalBrowsingContext} is open. * * @param {CanonicalBrowsingContext} browsingContext * Canonical browsing context to check. * @param {string=} msg * Custom error message. * * @returns {CanonicalBrowsingContext} * browsingContext is returned unaltered. * * @throws {NoSuchWindowError} * If browsingContext is no longer open. */ assert.open = function (browsingContext, msg = "") { msg = msg || "Browsing context has been discarded"; return assert.that( browsingContext => { if (!browsingContext?.currentWindowGlobal) { return false; } if (browsingContext.isContent && !browsingContext.top.embedderElement) { return false; } return true; }, msg, lazy.error.NoSuchWindowError )(browsingContext); }; /** * Asserts that there is no current user prompt. * * @param {modal.Dialog} dialog * Reference to current dialogue. * @param {string=} msg * Custom error message. * * @throws {UnexpectedAlertOpenError} * If there is a user prompt. */ assert.noUserPrompt = function (dialog, msg = "") { assert.that( d => d === null || typeof d == "undefined", msg, lazy.error.UnexpectedAlertOpenError )(dialog); }; /** * Asserts that obj is defined. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {?} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not defined. */ assert.defined = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be defined`; return assert.that(o => typeof o != "undefined", msg)(obj); }; /** * Asserts that obj is a finite number. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a number. */ assert.number = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be finite number`; return assert.that(Number.isFinite, msg)(obj); }; /** * Asserts that obj is a positive number. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a positive integer. */ assert.positiveNumber = function (obj, msg = "") { assert.number(obj, msg); msg = msg || lazy.pprint`Expected ${obj} to be >= 0`; return assert.that(n => n >= 0, msg)(obj); }; /** * Asserts that obj is a number in the inclusive range lower to upper. * * @param {?} obj * Value to test. * @param {Array} range * Array range [lower, upper] * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a number in the specified range. */ assert.numberInRange = function (obj, range, msg = "") { const [lower, upper] = range; assert.number(obj, msg); msg = msg || lazy.pprint`Expected ${obj} to be >= ${lower} and <= ${upper}`; return assert.that(n => n >= lower && n <= upper, msg)(obj); }; /** * Asserts that obj is callable. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {Function} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not callable. */ assert.callable = function (obj, msg = "") { msg = msg || lazy.pprint`${obj} is not callable`; return assert.that(o => typeof o == "function", msg)(obj); }; /** * Asserts that obj is an unsigned short number. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not an unsigned short. */ assert.unsignedShort = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be >= 0 and < 65536`; return assert.that(n => n >= 0 && n < 65536, msg)(obj); }; /** * Asserts that obj is an integer. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not an integer. */ assert.integer = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be an integer`; return assert.that(Number.isSafeInteger, msg)(obj); }; /** * Asserts that obj is a positive integer. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a positive integer. */ assert.positiveInteger = function (obj, msg = "") { assert.integer(obj, msg); msg = msg || lazy.pprint`Expected ${obj} to be >= 0`; return assert.that(n => n >= 0, msg)(obj); }; /** * Asserts that obj is an integer in the inclusive range lower to upper. * * @param {?} obj * Value to test. * @param {Array} range * Array range [lower, upper] * @param {string=} msg * Custom error message. * * @returns {number} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a number in the specified range. */ assert.integerInRange = function (obj, range, msg = "") { const [lower, upper] = range; assert.integer(obj, msg); msg = msg || lazy.pprint`Expected ${obj} to be >= ${lower} and <= ${upper}`; return assert.that(n => n >= lower && n <= upper, msg)(obj); }; /** * Asserts that obj is a boolean. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {boolean} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a boolean. */ assert.boolean = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be boolean`; return assert.that(b => typeof b == "boolean", msg)(obj); }; /** * Asserts that obj is a string. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {string} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not a string. */ assert.string = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be a string`; return assert.that(s => typeof s == "string", msg)(obj); }; /** * Asserts that obj is an object. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {object} * obj| is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not an object. */ assert.object = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be an object`; return assert.that(o => { // unable to use instanceof because LHS and RHS may come from // different globals let s = Object.prototype.toString.call(o); return s == "[object Object]" || s == "[object nsJSIID]"; }, msg)(obj); }; /** * Asserts that prop is in obj. * * @param {?} prop * An array element or own property to test if is in obj. * @param {?} obj * An array or an Object that is being tested. * @param {string=} msg * Custom error message. * * @returns {?} * The array element, or the value of obj's own property * prop. * * @throws {InvalidArgumentError} * If the obj was an array and did not contain prop. * Otherwise if prop is not in obj, or obj * is not an object. */ assert.in = function (prop, obj, msg = "") { if (Array.isArray(obj)) { assert.that(p => obj.includes(p), msg)(prop); return prop; } assert.object(obj, msg); msg = msg || lazy.pprint`Expected ${prop} in ${obj}`; assert.that(p => obj.hasOwnProperty(p), msg)(prop); return obj[prop]; }; /** * Asserts that obj is an Array. * * @param {?} obj * Value to test. * @param {string=} msg * Custom error message. * * @returns {object} * obj is returned unaltered. * * @throws {InvalidArgumentError} * If obj is not an Array. */ assert.array = function (obj, msg = "") { msg = msg || lazy.pprint`Expected ${obj} to be an Array`; return assert.that(Array.isArray, msg)(obj); }; /** * Returns a function that is used to assert the |predicate|. * * @param {function(?): boolean} predicate * Evaluated on calling the return value of this function. If its * return value of the inner function is false, error * is thrown with message. * @param {string=} message * Custom error message. * @param {Error=} err * Custom error type by its class. * * @returns {function(?): ?} * Function that takes and returns the passed in value unaltered, * and which may throw error with message * if predicate evaluates to false. */ assert.that = function ( predicate, message = "", err = lazy.error.InvalidArgumentError ) { return obj => { if (!predicate(obj)) { throw new err(message); } return obj; }; };