2016-11-03 01:33:53 +01:00
|
|
|
// 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 Defines the {@linkplain Driver WebDriver} client for Firefox.
|
|
|
|
* Before using this module, you must download the latest
|
|
|
|
* [geckodriver release] and ensure it can be found on your system [PATH].
|
|
|
|
*
|
|
|
|
* Each FirefoxDriver instance will be created with an anonymous profile,
|
|
|
|
* ensuring browser historys do not share session data (cookies, history, cache,
|
|
|
|
* offline storage, etc.)
|
|
|
|
*
|
|
|
|
* __Customizing the Firefox Profile__
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* The {@linkplain Profile} class may be used to configure the browser profile
|
|
|
|
* used with WebDriver, with functions to install additional
|
2016-11-03 01:33:53 +01:00
|
|
|
* {@linkplain Profile#addExtension extensions}, configure browser
|
|
|
|
* {@linkplain Profile#setPreference preferences}, and more. For example, you
|
|
|
|
* may wish to include Firebug:
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* const {Builder} = require('selenium-webdriver');
|
|
|
|
* const firefox = require('selenium-webdriver/firefox');
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let profile = new firefox.Profile();
|
2016-11-03 01:33:53 +01:00
|
|
|
* profile.addExtension('/path/to/firebug.xpi');
|
|
|
|
* profile.setPreference('extensions.firebug.showChromeErrors', true);
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let options = new firefox.Options().setProfile(profile);
|
|
|
|
* let driver = new Builder()
|
|
|
|
* .forBrowser('firefox')
|
|
|
|
* .setFirefoxOptions(options)
|
|
|
|
* .build();
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* The {@linkplain Profile} class may also be used to configure WebDriver based
|
|
|
|
* on a pre-existing browser profile:
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let profile = new firefox.Profile(
|
2016-11-03 01:33:53 +01:00
|
|
|
* '/usr/local/home/bob/.mozilla/firefox/3fgog75h.testing');
|
2017-04-20 03:09:25 +02:00
|
|
|
* let options = new firefox.Options().setProfile(profile);
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
|
|
|
* The FirefoxDriver will _never_ modify a pre-existing profile; instead it will
|
|
|
|
* create a copy for it to modify. By extension, there are certain browser
|
|
|
|
* preferences that are required for WebDriver to function properly and they
|
|
|
|
* will always be overwritten.
|
|
|
|
*
|
|
|
|
* __Using a Custom Firefox Binary__
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* On Windows and MacOS, the FirefoxDriver will search for Firefox in its
|
2016-11-03 01:33:53 +01:00
|
|
|
* default installation location:
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* - Windows: C:\Program Files and C:\Program Files (x86).
|
|
|
|
* - MacOS: /Applications/Firefox.app
|
|
|
|
*
|
|
|
|
* For Linux, Firefox will always be located on the PATH: `$(where firefox)`.
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* Several methods are provided for starting Firefox with a custom executable.
|
|
|
|
* First, on Windows and MacOS, you may configure WebDriver to check the default
|
|
|
|
* install location for a non-release channel. If the requested channel cannot
|
|
|
|
* be found in its default location, WebDriver will fallback to searching your
|
|
|
|
* PATH. _Note:_ on Linux, Firefox is _always_ located on your path, regardless
|
|
|
|
* of the requested channel.
|
|
|
|
*
|
|
|
|
* const {Builder} = require('selenium-webdriver');
|
|
|
|
* const firefox = require('selenium-webdriver/firefox');
|
|
|
|
*
|
|
|
|
* let options = new firefox.Options().setBinary(firefox.Channel.NIGHTLY);
|
|
|
|
* let driver = new Builder()
|
|
|
|
* .forBrowser('firefox')
|
|
|
|
* .setFirefoxOptions(options)
|
|
|
|
* .build();
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* On all platforms, you may configrue WebDriver to use a Firefox specific
|
|
|
|
* executable:
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let options = new firefox.Options()
|
|
|
|
* .setBinary('/my/firefox/install/dir/firefox-bin');
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
|
|
|
* __Remote Testing__
|
|
|
|
*
|
|
|
|
* You may customize the Firefox binary and profile when running against a
|
|
|
|
* remote Selenium server. Your custom profile will be packaged as a zip and
|
|
|
|
* transfered to the remote host for use. The profile will be transferred
|
|
|
|
* _once for each new session_. The performance impact should be minimal if
|
|
|
|
* you've only configured a few extra browser preferences. If you have a large
|
|
|
|
* profile with several extensions, you should consider installing it on the
|
|
|
|
* remote host and defining its path via the {@link Options} class. Custom
|
|
|
|
* binaries are never copied to remote machines and must be referenced by
|
|
|
|
* installation path.
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* const {Builder} = require('selenium-webdriver');
|
|
|
|
* const firefox = require('selenium-webdriver/firefox');
|
|
|
|
*
|
|
|
|
* let options = new firefox.Options()
|
2016-11-03 01:33:53 +01:00
|
|
|
* .setProfile('/profile/path/on/remote/host')
|
|
|
|
* .setBinary('/install/dir/on/remote/host/firefox-bin');
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let driver = new Builder()
|
2016-11-03 01:33:53 +01:00
|
|
|
* .forBrowser('firefox')
|
|
|
|
* .usingServer('http://127.0.0.1:4444/wd/hub')
|
|
|
|
* .setFirefoxOptions(options)
|
|
|
|
* .build();
|
|
|
|
*
|
|
|
|
* __Testing Older Versions of Firefox__
|
|
|
|
*
|
|
|
|
* To test versions of Firefox prior to Firefox 47, you must disable the use of
|
|
|
|
* the geckodriver using the {@link Options} class.
|
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* let options = new firefox.Options().useGeckoDriver(false);
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
|
|
|
* Alternatively, you may disable the geckodriver at runtime by setting the
|
|
|
|
* environment variable `SELENIUM_MARIONETTE=false`.
|
|
|
|
*
|
|
|
|
* [geckodriver release]: https://github.com/mozilla/geckodriver/releases/
|
|
|
|
* [PATH]: http://en.wikipedia.org/wiki/PATH_%28variable%29
|
|
|
|
*/
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const url = require('url');
|
|
|
|
|
2017-04-20 03:09:25 +02:00
|
|
|
const {Binary, Channel} = require('./binary'),
|
2016-11-03 01:33:53 +01:00
|
|
|
Profile = require('./profile').Profile,
|
|
|
|
decodeProfile = require('./profile').decode,
|
|
|
|
http = require('../http'),
|
|
|
|
httpUtil = require('../http/util'),
|
|
|
|
io = require('../io'),
|
|
|
|
capabilities = require('../lib/capabilities'),
|
|
|
|
command = require('../lib/command'),
|
|
|
|
logging = require('../lib/logging'),
|
|
|
|
promise = require('../lib/promise'),
|
|
|
|
webdriver = require('../lib/webdriver'),
|
|
|
|
net = require('../net'),
|
|
|
|
portprober = require('../net/portprober'),
|
|
|
|
remote = require('../remote');
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Firefox-specific capability keys. Users should use the {@linkplain Options}
|
|
|
|
* class instead of referencing these keys directly. _These keys are considered
|
|
|
|
* implementation details and may be removed or changed at any time._
|
|
|
|
*
|
|
|
|
* @enum {string}
|
|
|
|
*/
|
|
|
|
const Capability = {
|
|
|
|
/**
|
|
|
|
* Defines the Firefox binary to use. May be set to either a
|
|
|
|
* {@linkplain Binary} instance, or a string path to the Firefox executable.
|
|
|
|
*/
|
|
|
|
BINARY: 'firefox_binary',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Specifies whether to use Mozilla's Marionette, or the legacy FirefoxDriver
|
|
|
|
* from the Selenium project. Defaults to false.
|
|
|
|
*/
|
|
|
|
MARIONETTE: 'marionette',
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Defines the Firefox profile to use. May be set to either a
|
|
|
|
* {@linkplain Profile} instance, or to a base-64 encoded zip of a profile
|
|
|
|
* directory.
|
|
|
|
*/
|
|
|
|
PROFILE: 'firefox_profile'
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Configuration options for the FirefoxDriver.
|
|
|
|
*/
|
|
|
|
class Options {
|
|
|
|
constructor() {
|
|
|
|
/** @private {Profile} */
|
|
|
|
this.profile_ = null;
|
|
|
|
|
|
|
|
/** @private {Binary} */
|
|
|
|
this.binary_ = null;
|
|
|
|
|
|
|
|
/** @private {logging.Preferences} */
|
|
|
|
this.logPrefs_ = null;
|
|
|
|
|
|
|
|
/** @private {?capabilities.ProxyConfig} */
|
|
|
|
this.proxy_ = null;
|
|
|
|
|
|
|
|
/** @private {boolean} */
|
|
|
|
this.marionette_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the profile to use. The profile may be specified as a
|
|
|
|
* {@link Profile} object or as the path to an existing Firefox profile to use
|
|
|
|
* as a template.
|
|
|
|
*
|
|
|
|
* @param {(string|!Profile)} profile The profile to use.
|
|
|
|
* @return {!Options} A self reference.
|
|
|
|
*/
|
|
|
|
setProfile(profile) {
|
|
|
|
if (typeof profile === 'string') {
|
|
|
|
profile = new Profile(profile);
|
|
|
|
}
|
|
|
|
this.profile_ = profile;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-04-20 03:09:25 +02:00
|
|
|
* Sets the binary to use. The binary may be specified as the path to a
|
|
|
|
* Firefox executable, a specific {@link Channel}, or as a {@link Binary}
|
|
|
|
* object.
|
2016-11-03 01:33:53 +01:00
|
|
|
*
|
2017-04-20 03:09:25 +02:00
|
|
|
* @param {(string|!Binary|!Channel)} binary The binary to use.
|
2016-11-03 01:33:53 +01:00
|
|
|
* @return {!Options} A self reference.
|
2017-04-20 03:09:25 +02:00
|
|
|
* @throws {TypeError} If `binary` is an invalid type.
|
2016-11-03 01:33:53 +01:00
|
|
|
*/
|
|
|
|
setBinary(binary) {
|
2017-04-20 03:09:25 +02:00
|
|
|
if (typeof binary === 'string' || binary instanceof Channel) {
|
2016-11-03 01:33:53 +01:00
|
|
|
binary = new Binary(binary);
|
|
|
|
}
|
2017-04-20 03:09:25 +02:00
|
|
|
|
|
|
|
if (!(binary instanceof Binary)) {
|
|
|
|
throw TypeError(
|
|
|
|
'binary must be a string path, Channel, or Binary object');
|
|
|
|
}
|
|
|
|
|
2016-11-03 01:33:53 +01:00
|
|
|
this.binary_ = binary;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the logging preferences for the new session.
|
|
|
|
* @param {logging.Preferences} prefs The logging preferences.
|
|
|
|
* @return {!Options} A self reference.
|
|
|
|
*/
|
|
|
|
setLoggingPreferences(prefs) {
|
|
|
|
this.logPrefs_ = prefs;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the proxy to use.
|
|
|
|
*
|
|
|
|
* @param {capabilities.ProxyConfig} proxy The proxy configuration to use.
|
|
|
|
* @return {!Options} A self reference.
|
|
|
|
*/
|
|
|
|
setProxy(proxy) {
|
|
|
|
this.proxy_ = proxy;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets whether to use Mozilla's geckodriver to drive the browser. This option
|
|
|
|
* is enabled by default and required for Firefox 47+.
|
|
|
|
*
|
|
|
|
* @param {boolean} enable Whether to enable the geckodriver.
|
|
|
|
* @see https://github.com/mozilla/geckodriver
|
|
|
|
*/
|
|
|
|
useGeckoDriver(enable) {
|
|
|
|
this.marionette_ = enable;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts these options to a {@link capabilities.Capabilities} instance.
|
|
|
|
*
|
|
|
|
* @return {!capabilities.Capabilities} A new capabilities object.
|
|
|
|
*/
|
|
|
|
toCapabilities() {
|
|
|
|
var caps = capabilities.Capabilities.firefox();
|
|
|
|
if (this.logPrefs_) {
|
|
|
|
caps.set(capabilities.Capability.LOGGING_PREFS, this.logPrefs_);
|
|
|
|
}
|
|
|
|
if (this.proxy_) {
|
|
|
|
caps.set(capabilities.Capability.PROXY, this.proxy_);
|
|
|
|
}
|
|
|
|
if (this.binary_) {
|
|
|
|
caps.set(Capability.BINARY, this.binary_);
|
|
|
|
}
|
|
|
|
if (this.profile_) {
|
|
|
|
caps.set(Capability.PROFILE, this.profile_);
|
|
|
|
}
|
|
|
|
caps.set(Capability.MARIONETTE, this.marionette_);
|
|
|
|
return caps;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enum of available command contexts.
|
|
|
|
*
|
|
|
|
* Command contexts are specific to Marionette, and may be used with the
|
|
|
|
* {@link #context=} method. Contexts allow you to direct all subsequent
|
|
|
|
* commands to either "content" (default) or "chrome". The latter gives
|
|
|
|
* you elevated security permissions.
|
|
|
|
*
|
|
|
|
* @enum {string}
|
|
|
|
*/
|
|
|
|
const Context = {
|
|
|
|
CONTENT: "content",
|
|
|
|
CHROME: "chrome",
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const GECKO_DRIVER_EXE =
|
|
|
|
process.platform === 'win32' ? 'geckodriver.exe' : 'geckodriver';
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {string} .
|
|
|
|
* @throws {Error}
|
|
|
|
*/
|
|
|
|
function findGeckoDriver() {
|
|
|
|
let exe = io.findInPath(GECKO_DRIVER_EXE, true);
|
|
|
|
if (!exe) {
|
|
|
|
throw Error(
|
|
|
|
'The ' + GECKO_DRIVER_EXE + ' executable could not be found on the current ' +
|
|
|
|
'PATH. Please download the latest version from ' +
|
|
|
|
'https://github.com/mozilla/geckodriver/releases/' +
|
|
|
|
'WebDriver and ensure it can be found on your PATH.');
|
|
|
|
}
|
|
|
|
return exe;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {(Profile|string)} profile The profile to prepare.
|
|
|
|
* @param {number} port The port the FirefoxDriver should listen on.
|
|
|
|
* @return {!Promise<string>} a promise for the path to the profile directory.
|
|
|
|
*/
|
|
|
|
function prepareProfile(profile, port) {
|
|
|
|
if (typeof profile === 'string') {
|
|
|
|
return decodeProfile(/** @type {string} */(profile)).then(dir => {
|
|
|
|
profile = new Profile(dir);
|
|
|
|
profile.setPreference('webdriver_firefox_port', port);
|
|
|
|
return profile.writeToDisk();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
profile = profile || new Profile;
|
|
|
|
profile.setPreference('webdriver_firefox_port', port);
|
|
|
|
return profile.writeToDisk();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function normalizeProxyConfiguration(config) {
|
|
|
|
if ('manual' === config.proxyType) {
|
|
|
|
if (config.ftpProxy && !config.ftpProxyPort) {
|
|
|
|
let hostAndPort = net.splitHostAndPort(config.ftpProxy);
|
|
|
|
config.ftpProxy = hostAndPort.host;
|
|
|
|
config.ftpProxyPort = hostAndPort.port;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.httpProxy && !config.httpProxyPort) {
|
|
|
|
let hostAndPort = net.splitHostAndPort(config.httpProxy);
|
|
|
|
config.httpProxy = hostAndPort.host;
|
|
|
|
config.httpProxyPort = hostAndPort.port;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.sslProxy && !config.sslProxyPort) {
|
|
|
|
let hostAndPort = net.splitHostAndPort(config.sslProxy);
|
|
|
|
config.sslProxy = hostAndPort.host;
|
|
|
|
config.sslProxyPort = hostAndPort.port;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.socksProxy && !config.socksProxyPort) {
|
|
|
|
let hostAndPort = net.splitHostAndPort(config.socksProxy);
|
|
|
|
config.socksProxy = hostAndPort.host;
|
|
|
|
config.socksProxyPort = hostAndPort.port;
|
|
|
|
}
|
|
|
|
} else if ('pac' === config.proxyType) {
|
|
|
|
if (config.proxyAutoconfigUrl && !config.pacUrl) {
|
|
|
|
config.pacUrl = config.proxyAutoconfigUrl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return config;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/** @enum {string} */
|
|
|
|
const ExtensionCommand = {
|
|
|
|
GET_CONTEXT: 'getContext',
|
|
|
|
SET_CONTEXT: 'setContext',
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a command executor with support for Marionette's custom commands.
|
|
|
|
* @param {!Promise<string>} serverUrl The server's URL.
|
|
|
|
* @return {!command.Executor} The new command executor.
|
|
|
|
*/
|
|
|
|
function createExecutor(serverUrl) {
|
|
|
|
let client = serverUrl.then(url => new http.HttpClient(url));
|
|
|
|
let executor = new http.Executor(client);
|
|
|
|
configureExecutor(executor);
|
|
|
|
return executor;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Configures the given executor with Firefox-specific commands.
|
|
|
|
* @param {!http.Executor} executor the executor to configure.
|
|
|
|
*/
|
|
|
|
function configureExecutor(executor) {
|
|
|
|
executor.defineCommand(
|
|
|
|
ExtensionCommand.GET_CONTEXT,
|
|
|
|
'GET',
|
|
|
|
'/session/:sessionId/moz/context');
|
|
|
|
|
|
|
|
executor.defineCommand(
|
|
|
|
ExtensionCommand.SET_CONTEXT,
|
|
|
|
'POST',
|
|
|
|
'/session/:sessionId/moz/context');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates {@link selenium-webdriver/remote.DriverService} instances that manage
|
|
|
|
* a [geckodriver](https://github.com/mozilla/geckodriver) server in a child
|
|
|
|
* process.
|
|
|
|
*/
|
|
|
|
class ServiceBuilder extends remote.DriverService.Builder {
|
|
|
|
/**
|
|
|
|
* @param {string=} opt_exe Path to the server executable to use. If omitted,
|
|
|
|
* the builder will attempt to locate the geckodriver on the system PATH.
|
|
|
|
*/
|
|
|
|
constructor(opt_exe) {
|
|
|
|
super(opt_exe || findGeckoDriver());
|
|
|
|
this.setLoopback(true); // Required.
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Enables verbose logging.
|
|
|
|
*
|
|
|
|
* @param {boolean=} opt_trace Whether to enable trace-level logging. By
|
|
|
|
* default, only debug logging is enabled.
|
|
|
|
* @return {!ServiceBuilder} A self reference.
|
|
|
|
*/
|
|
|
|
enableVerboseLogging(opt_trace) {
|
|
|
|
return this.addArguments(opt_trace ? '-vv' : '-v');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the path to the executable Firefox binary that the geckodriver should
|
|
|
|
* use. If this method is not called, this builder will attempt to locate
|
|
|
|
* Firefox in the default installation location for the current platform.
|
|
|
|
*
|
|
|
|
* @param {(string|!Binary)} binary Path to the executable Firefox binary to use.
|
|
|
|
* @return {!ServiceBuilder} A self reference.
|
|
|
|
* @see Binary#locate()
|
|
|
|
*/
|
|
|
|
setFirefoxBinary(binary) {
|
|
|
|
let exe = typeof binary === 'string'
|
|
|
|
? Promise.resolve(binary) : binary.locate();
|
|
|
|
return this.addArguments('-b', exe);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2016-11-16 01:59:39 +01:00
|
|
|
* @typedef {{executor: !command.Executor,
|
|
|
|
* capabilities: (!capabilities.Capabilities|
|
|
|
|
* {desired: (capabilities.Capabilities|undefined),
|
|
|
|
* required: (capabilities.Capabilities|undefined)}),
|
|
|
|
* onQuit: function(this: void): ?}}
|
2016-11-03 01:33:53 +01:00
|
|
|
*/
|
|
|
|
var DriverSpec;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {(http.Executor|remote.DriverService|undefined)} executor
|
|
|
|
* @param {!capabilities.Capabilities} caps
|
|
|
|
* @param {Profile} profile
|
|
|
|
* @param {Binary} binary
|
|
|
|
* @return {DriverSpec}
|
|
|
|
*/
|
2016-11-16 01:59:39 +01:00
|
|
|
function createGeckoDriver(executor, caps, profile, binary) {
|
|
|
|
let firefoxOptions = {};
|
|
|
|
caps.set('moz:firefoxOptions', firefoxOptions);
|
|
|
|
|
|
|
|
if (binary) {
|
|
|
|
if (binary.getExe()) {
|
|
|
|
firefoxOptions['binary'] = binary.getExe();
|
|
|
|
}
|
|
|
|
|
|
|
|
let args = binary.getArguments();
|
|
|
|
if (args.length) {
|
|
|
|
firefoxOptions['args'] = args;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-03 01:33:53 +01:00
|
|
|
if (profile) {
|
2016-11-16 01:59:39 +01:00
|
|
|
// If the user specified a template directory or any extensions to install,
|
|
|
|
// we need to encode the profile as a base64 string (which requires writing
|
|
|
|
// it to disk first). Otherwise, if the user just specified some custom
|
|
|
|
// preferences, we can send those directly.
|
|
|
|
if (profile.getTemplateDir() || profile.getExtensions().length) {
|
|
|
|
firefoxOptions['profile'] = profile.encode();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
let prefs = profile.getPreferences();
|
|
|
|
if (Object.keys(prefs).length) {
|
|
|
|
firefoxOptions['prefs'] = prefs;
|
|
|
|
}
|
|
|
|
}
|
2016-11-03 01:33:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
let sessionCaps = caps;
|
|
|
|
if (caps.has(capabilities.Capability.PROXY)) {
|
|
|
|
let proxy = normalizeProxyConfiguration(
|
|
|
|
caps.get(capabilities.Capability.PROXY));
|
|
|
|
|
|
|
|
// Marionette requires proxy settings to be specified as required
|
|
|
|
// capabilities. See mozilla/geckodriver#97
|
|
|
|
let required = new capabilities.Capabilities()
|
|
|
|
.set(capabilities.Capability.PROXY, proxy);
|
|
|
|
|
|
|
|
caps.delete(capabilities.Capability.PROXY);
|
|
|
|
sessionCaps = {required, desired: caps};
|
|
|
|
}
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
/** @type {!command.Executor} */
|
2016-11-03 01:33:53 +01:00
|
|
|
let cmdExecutor;
|
|
|
|
let onQuit = function() {};
|
|
|
|
|
|
|
|
if (executor instanceof http.Executor) {
|
|
|
|
configureExecutor(executor);
|
|
|
|
cmdExecutor = executor;
|
|
|
|
} else if (executor instanceof remote.DriverService) {
|
|
|
|
cmdExecutor = createExecutor(executor.start());
|
|
|
|
onQuit = () => executor.kill();
|
|
|
|
} else {
|
|
|
|
let builder = new ServiceBuilder();
|
|
|
|
if (binary) {
|
|
|
|
builder.setFirefoxBinary(binary);
|
|
|
|
}
|
|
|
|
let service = builder.build();
|
|
|
|
cmdExecutor = createExecutor(service.start());
|
|
|
|
onQuit = () => service.kill();
|
|
|
|
}
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
return {
|
|
|
|
executor: cmdExecutor,
|
|
|
|
capabilities: sessionCaps,
|
|
|
|
onQuit
|
|
|
|
};
|
2016-11-03 01:33:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {!capabilities.Capabilities} caps
|
|
|
|
* @param {Profile} profile
|
|
|
|
* @param {!Binary} binary
|
|
|
|
* @return {DriverSpec}
|
|
|
|
*/
|
|
|
|
function createLegacyDriver(caps, profile, binary, flow) {
|
|
|
|
profile = profile || new Profile;
|
|
|
|
|
|
|
|
let freePort = portprober.findFreePort();
|
|
|
|
let preparedProfile =
|
|
|
|
freePort.then(port => prepareProfile(profile, port));
|
|
|
|
let command = preparedProfile.then(dir => binary.launch(dir));
|
|
|
|
|
|
|
|
let serverUrl = command.then(() => freePort)
|
|
|
|
.then(function(/** number */port) {
|
|
|
|
let serverUrl = url.format({
|
|
|
|
protocol: 'http',
|
|
|
|
hostname: net.getLoopbackAddress(),
|
|
|
|
port: port + '',
|
|
|
|
pathname: '/hub'
|
|
|
|
});
|
|
|
|
let ready = httpUtil.waitForServer(serverUrl, 45 * 1000);
|
|
|
|
return ready.then(() => serverUrl);
|
|
|
|
});
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
return {
|
|
|
|
executor: createExecutor(serverUrl),
|
|
|
|
capabilities: caps,
|
|
|
|
onQuit: function() {
|
|
|
|
return command.then(command => {
|
|
|
|
command.kill();
|
|
|
|
return preparedProfile.then(io.rmDir)
|
|
|
|
.then(() => command.result(),
|
|
|
|
() => command.result());
|
|
|
|
});
|
|
|
|
}
|
2016-11-03 01:33:53 +01:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A WebDriver client for Firefox.
|
|
|
|
*/
|
|
|
|
class Driver extends webdriver.WebDriver {
|
|
|
|
/**
|
2016-11-16 01:59:39 +01:00
|
|
|
* Creates a new Firefox session.
|
|
|
|
*
|
2016-11-03 01:33:53 +01:00
|
|
|
* @param {(Options|capabilities.Capabilities|Object)=} opt_config The
|
|
|
|
* configuration options for this driver, specified as either an
|
|
|
|
* {@link Options} or {@link capabilities.Capabilities}, or as a raw hash
|
|
|
|
* object.
|
|
|
|
* @param {(http.Executor|remote.DriverService)=} opt_executor Either a
|
|
|
|
* pre-configured command executor to use for communicating with an
|
|
|
|
* externally managed remote end (which is assumed to already be running),
|
|
|
|
* or the `DriverService` to use to start the geckodriver in a child
|
|
|
|
* process.
|
|
|
|
*
|
|
|
|
* If an executor is provided, care should e taken not to use reuse it with
|
|
|
|
* other clients as its internal command mappings will be updated to support
|
|
|
|
* Firefox-specific commands.
|
|
|
|
*
|
|
|
|
* _This parameter may only be used with Mozilla's GeckoDriver._
|
|
|
|
*
|
|
|
|
* @param {promise.ControlFlow=} opt_flow The flow to
|
|
|
|
* schedule commands through. Defaults to the active flow object.
|
|
|
|
* @throws {Error} If a custom command executor is provided and the driver is
|
|
|
|
* configured to use the legacy FirefoxDriver from the Selenium project.
|
2016-11-16 01:59:39 +01:00
|
|
|
* @return {!Driver} A new driver instance.
|
2016-11-03 01:33:53 +01:00
|
|
|
*/
|
2016-11-16 01:59:39 +01:00
|
|
|
static createSession(opt_config, opt_executor, opt_flow) {
|
2016-11-03 01:33:53 +01:00
|
|
|
let caps;
|
|
|
|
if (opt_config instanceof Options) {
|
|
|
|
caps = opt_config.toCapabilities();
|
|
|
|
} else {
|
|
|
|
caps = new capabilities.Capabilities(opt_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
let binary = caps.get(Capability.BINARY) || new Binary();
|
|
|
|
caps.delete(Capability.BINARY);
|
|
|
|
if (typeof binary === 'string') {
|
|
|
|
binary = new Binary(binary);
|
|
|
|
}
|
|
|
|
|
|
|
|
let profile;
|
|
|
|
if (caps.has(Capability.PROFILE)) {
|
|
|
|
profile = caps.get(Capability.PROFILE);
|
|
|
|
caps.delete(Capability.PROFILE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Users must now explicitly disable marionette to use the legacy
|
|
|
|
// FirefoxDriver.
|
|
|
|
let noMarionette =
|
|
|
|
caps.get(Capability.MARIONETTE) === false
|
|
|
|
|| /^0|false$/i.test(process.env['SELENIUM_MARIONETTE']);
|
|
|
|
let useMarionette = !noMarionette;
|
|
|
|
|
|
|
|
let spec;
|
|
|
|
if (useMarionette) {
|
2016-11-16 01:59:39 +01:00
|
|
|
spec = createGeckoDriver(opt_executor, caps, profile, binary);
|
2016-11-03 01:33:53 +01:00
|
|
|
} else {
|
|
|
|
if (opt_executor) {
|
|
|
|
throw Error('You may not use a custom command executor with the legacy'
|
|
|
|
+ ' FirefoxDriver');
|
|
|
|
}
|
|
|
|
spec = createLegacyDriver(caps, profile, binary, opt_flow);
|
|
|
|
}
|
|
|
|
|
2016-11-16 01:59:39 +01:00
|
|
|
return /** @type {!Driver} */(webdriver.WebDriver.createSession(
|
|
|
|
spec.executor, spec.capabilities, opt_flow, this, spec.onQuit));
|
2016-11-03 01:33:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is a no-op as file detectors are not supported by this
|
|
|
|
* implementation.
|
|
|
|
* @override
|
|
|
|
*/
|
|
|
|
setFileDetector() {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the context that is currently in effect.
|
|
|
|
*
|
2016-11-16 01:59:39 +01:00
|
|
|
* @return {!promise.Thenable<Context>} Current context.
|
2016-11-03 01:33:53 +01:00
|
|
|
*/
|
|
|
|
getContext() {
|
|
|
|
return this.schedule(
|
|
|
|
new command.Command(ExtensionCommand.GET_CONTEXT),
|
|
|
|
'get WebDriver.context');
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes target context for commands between chrome- and content.
|
|
|
|
*
|
|
|
|
* Changing the current context has a stateful impact on all subsequent
|
|
|
|
* commands. The {@link Context.CONTENT} context has normal web
|
|
|
|
* platform document permissions, as if you would evaluate arbitrary
|
|
|
|
* JavaScript. The {@link Context.CHROME} context gets elevated
|
|
|
|
* permissions that lets you manipulate the browser chrome itself,
|
|
|
|
* with full access to the XUL toolkit.
|
|
|
|
*
|
|
|
|
* Use your powers wisely.
|
|
|
|
*
|
2016-11-16 01:59:39 +01:00
|
|
|
* @param {!promise.Thenable<void>} ctx The context to switch to.
|
2016-11-03 01:33:53 +01:00
|
|
|
*/
|
|
|
|
setContext(ctx) {
|
|
|
|
return this.schedule(
|
|
|
|
new command.Command(ExtensionCommand.SET_CONTEXT)
|
|
|
|
.setParameter("context", ctx),
|
|
|
|
'set WebDriver.context');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// PUBLIC API
|
|
|
|
|
|
|
|
|
|
|
|
exports.Binary = Binary;
|
2017-04-20 03:09:25 +02:00
|
|
|
exports.Channel = Channel;
|
2016-11-03 01:33:53 +01:00
|
|
|
exports.Context = Context;
|
|
|
|
exports.Driver = Driver;
|
|
|
|
exports.Options = Options;
|
|
|
|
exports.Profile = Profile;
|
|
|
|
exports.ServiceBuilder = ServiceBuilder;
|