blob: d06e474271115d58d13684ec6a3455a3c428ff5b [file] [log] [blame] [edit]
// Licensed to the Software Freedom Conservancy (SFC) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The SFC licenses this file
// to you 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 Utilities for working with errors as defined by WebDriver's
* wire protocol: https://github.com/SeleniumHQ/selenium/wiki/JsonWireProtocol
*/
goog.provide('bot.Error');
goog.provide('bot.ErrorCode');
/**
* Error codes from the Selenium WebDriver protocol:
* https://github.com/SeleniumHQ/selenium/wiki/JsonWireProtocol#response-status-codes
*
* @enum {number}
* @suppress {lintChecks}
*/
bot.ErrorCode = {
SUCCESS: 0, // Included for completeness
NO_SUCH_ELEMENT: 7,
NO_SUCH_FRAME: 8,
UNKNOWN_COMMAND: 9,
UNSUPPORTED_OPERATION: 9, // Alias.
STALE_ELEMENT_REFERENCE: 10,
ELEMENT_NOT_VISIBLE: 11,
INVALID_ELEMENT_STATE: 12,
UNKNOWN_ERROR: 13,
ELEMENT_NOT_SELECTABLE: 15,
JAVASCRIPT_ERROR: 17,
XPATH_LOOKUP_ERROR: 19,
TIMEOUT: 21,
NO_SUCH_WINDOW: 23,
INVALID_COOKIE_DOMAIN: 24,
UNABLE_TO_SET_COOKIE: 25,
UNEXPECTED_ALERT_OPEN: 26,
NO_SUCH_ALERT: 27,
SCRIPT_TIMEOUT: 28,
INVALID_ELEMENT_COORDINATES: 29,
IME_NOT_AVAILABLE: 30,
IME_ENGINE_ACTIVATION_FAILED: 31,
INVALID_SELECTOR_ERROR: 32,
SESSION_NOT_CREATED: 33,
MOVE_TARGET_OUT_OF_BOUNDS: 34,
SQL_DATABASE_ERROR: 35,
INVALID_XPATH_SELECTOR: 51,
INVALID_XPATH_SELECTOR_RETURN_TYPE: 52,
INVALID_ARGUMENT: 61,
// The following error codes are derived straight from HTTP return codes.
METHOD_NOT_ALLOWED: 405
};
/**
* Represents an error returned from a WebDriver command request.
*
* @param {!bot.ErrorCode} code The error's status code.
* @param {string=} opt_message Optional error message.
* @constructor
* @extends {Error}
*/
bot.Error = function (code, opt_message) {
/**
* This error's status code.
* @type {!bot.ErrorCode}
*/
this.code = code;
/** @type {string} */
this.state =
bot.Error.CODE_TO_STATE_[code] || bot.Error.State.UNKNOWN_ERROR;
/** @override */
this.message = opt_message || '';
var name = this.state.replace(/((?:^|\s+)[a-z])/g, function (str) {
// IE<9 does not support String#trim(). Also, IE does not include 0xa0
// (the non-breaking-space) in the \s character class, so we have to
// explicitly include it.
return str.toUpperCase().replace(/^[\s\xa0]+/g, '');
});
var l = name.length - 'Error'.length;
if (l < 0 || name.indexOf('Error', l) != l) {
name += 'Error';
}
/** @override */
this.name = name;
// Generate a stacktrace for our custom error; ensure the error has our
// custom name and message so the stack prints correctly in all browsers.
var template = new Error(this.message);
template.name = this.name;
/** @override */
this.stack = template.stack || '';
};
goog.inherits(bot.Error, Error);
/**
* Status strings enumerated in the W3C WebDriver protocol.
* @enum {string}
* @see https://w3c.github.io/webdriver/webdriver-spec.html#handling-errors
*/
bot.Error.State = {
ELEMENT_NOT_SELECTABLE: 'element not selectable',
ELEMENT_NOT_VISIBLE: 'element not visible',
INVALID_ARGUMENT: 'invalid argument',
INVALID_COOKIE_DOMAIN: 'invalid cookie domain',
INVALID_ELEMENT_COORDINATES: 'invalid element coordinates',
INVALID_ELEMENT_STATE: 'invalid element state',
INVALID_SELECTOR: 'invalid selector',
INVALID_SESSION_ID: 'invalid session id',
JAVASCRIPT_ERROR: 'javascript error',
MOVE_TARGET_OUT_OF_BOUNDS: 'move target out of bounds',
NO_SUCH_ALERT: 'no such alert',
NO_SUCH_ELEMENT: 'no such element',
NO_SUCH_FRAME: 'no such frame',
NO_SUCH_WINDOW: 'no such window',
SCRIPT_TIMEOUT: 'script timeout',
SESSION_NOT_CREATED: 'session not created',
STALE_ELEMENT_REFERENCE: 'stale element reference',
TIMEOUT: 'timeout',
UNABLE_TO_SET_COOKIE: 'unable to set cookie',
UNEXPECTED_ALERT_OPEN: 'unexpected alert open',
UNKNOWN_COMMAND: 'unknown command',
UNKNOWN_ERROR: 'unknown error',
UNKNOWN_METHOD: 'unknown method',
UNSUPPORTED_OPERATION: 'unsupported operation'
};
/**
* A map of error codes to state string.
* @private {!Object.<bot.ErrorCode, bot.Error.State>}
*/
bot.Error.CODE_TO_STATE_ = {};
goog.scope(function () {
var map = bot.Error.CODE_TO_STATE_;
var code = bot.ErrorCode;
var state = bot.Error.State;
map[code.ELEMENT_NOT_SELECTABLE] = state.ELEMENT_NOT_SELECTABLE;
map[code.ELEMENT_NOT_VISIBLE] = state.ELEMENT_NOT_VISIBLE;
map[code.IME_ENGINE_ACTIVATION_FAILED] = state.UNKNOWN_ERROR;
map[code.IME_NOT_AVAILABLE] = state.UNKNOWN_ERROR;
map[code.INVALID_COOKIE_DOMAIN] = state.INVALID_COOKIE_DOMAIN;
map[code.INVALID_ELEMENT_COORDINATES] = state.INVALID_ELEMENT_COORDINATES;
map[code.INVALID_ELEMENT_STATE] = state.INVALID_ELEMENT_STATE;
map[code.INVALID_SELECTOR_ERROR] = state.INVALID_SELECTOR;
map[code.INVALID_XPATH_SELECTOR] = state.INVALID_SELECTOR;
map[code.INVALID_XPATH_SELECTOR_RETURN_TYPE] = state.INVALID_SELECTOR;
map[code.JAVASCRIPT_ERROR] = state.JAVASCRIPT_ERROR;
map[code.METHOD_NOT_ALLOWED] = state.UNSUPPORTED_OPERATION;
map[code.MOVE_TARGET_OUT_OF_BOUNDS] = state.MOVE_TARGET_OUT_OF_BOUNDS;
map[code.NO_SUCH_ALERT] = state.NO_SUCH_ALERT;
map[code.NO_SUCH_ELEMENT] = state.NO_SUCH_ELEMENT;
map[code.NO_SUCH_FRAME] = state.NO_SUCH_FRAME;
map[code.NO_SUCH_WINDOW] = state.NO_SUCH_WINDOW;
map[code.SCRIPT_TIMEOUT] = state.SCRIPT_TIMEOUT;
map[code.SESSION_NOT_CREATED] = state.SESSION_NOT_CREATED;
map[code.STALE_ELEMENT_REFERENCE] = state.STALE_ELEMENT_REFERENCE;
map[code.TIMEOUT] = state.TIMEOUT;
map[code.UNABLE_TO_SET_COOKIE] = state.UNABLE_TO_SET_COOKIE;
map[code.UNEXPECTED_ALERT_OPEN] = state.UNEXPECTED_ALERT_OPEN;
map[code.UNKNOWN_ERROR] = state.UNKNOWN_ERROR;
map[code.UNSUPPORTED_OPERATION] = state.UNKNOWN_COMMAND;
}); // goog.scope
/**
* Flag used for duck-typing when this code is embedded in a Firefox extension.
* This is required since an Error thrown in one component and then reported
* to another will fail instanceof checks in the second component.
* @type {boolean}
*/
bot.Error.prototype.isAutomationError = true;
if (goog.DEBUG) {
/**
* @override
* @return {string} The string representation of this error.
*/
bot.Error.prototype.toString = function () {
return this.name + ': ' + this.message;
};
}