All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.openqa.selenium.firefox.FirefoxDriver Maven / Gradle / Ivy

There is a newer version: 4.26.0
Show newest version
// 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.

package org.openqa.selenium.firefox;

import static org.openqa.selenium.remote.CapabilityType.PROXY;

import java.net.URI;
import java.net.URISyntaxException;
import java.nio.file.Path;
import java.util.Map;
import java.util.Optional;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.openqa.selenium.Beta;
import org.openqa.selenium.Capabilities;
import org.openqa.selenium.ImmutableCapabilities;
import org.openqa.selenium.MutableCapabilities;
import org.openqa.selenium.OutputType;
import org.openqa.selenium.PersistentCapabilities;
import org.openqa.selenium.Proxy;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.bidi.BiDi;
import org.openqa.selenium.bidi.BiDiException;
import org.openqa.selenium.bidi.HasBiDi;
import org.openqa.selenium.devtools.CdpEndpointFinder;
import org.openqa.selenium.devtools.CdpInfo;
import org.openqa.selenium.devtools.CdpVersionFinder;
import org.openqa.selenium.devtools.Connection;
import org.openqa.selenium.devtools.DevTools;
import org.openqa.selenium.devtools.DevToolsException;
import org.openqa.selenium.devtools.HasDevTools;
import org.openqa.selenium.devtools.noop.NoOpCdpInfo;
import org.openqa.selenium.html5.LocalStorage;
import org.openqa.selenium.html5.SessionStorage;
import org.openqa.selenium.html5.WebStorage;
import org.openqa.selenium.internal.Require;
import org.openqa.selenium.manager.SeleniumManagerOutput.Result;
import org.openqa.selenium.remote.CommandInfo;
import org.openqa.selenium.remote.FileDetector;
import org.openqa.selenium.remote.RemoteWebDriver;
import org.openqa.selenium.remote.RemoteWebDriverBuilder;
import org.openqa.selenium.remote.html5.RemoteWebStorage;
import org.openqa.selenium.remote.http.ClientConfig;
import org.openqa.selenium.remote.http.HttpClient;
import org.openqa.selenium.remote.service.DriverCommandExecutor;
import org.openqa.selenium.remote.service.DriverFinder;
import org.openqa.selenium.remote.service.DriverService;

/**
 * An implementation of the {#link WebDriver} interface that drives Firefox.
 *
 * 

The best way to construct a {@code FirefoxDriver} with various options is to make use of the * {@link FirefoxOptions}, like so: * *

 * FirefoxOptions options = new FirefoxOptions()
 *     .addPreference("browser.startup.page", 1)
 *     .addPreference("browser.startup.homepage", "https://www.google.co.uk")
 *     .setAcceptInsecureCerts(true)
 *     .setHeadless(true);
 * WebDriver driver = new FirefoxDriver(options);
 * 
*/ public class FirefoxDriver extends RemoteWebDriver implements WebStorage, HasExtensions, HasFullPageScreenshot, HasContext, HasDevTools, HasBiDi { private static final Logger LOG = Logger.getLogger(FirefoxDriver.class.getName()); private final Capabilities capabilities; private final RemoteWebStorage webStorage; private final HasExtensions extensions; private final HasFullPageScreenshot fullPageScreenshot; private final HasContext context; private final Optional cdpUri; private final Optional biDiUri; private Connection connection; private DevTools devTools; private final Optional biDi; /** * Creates a new FirefoxDriver using the {@link GeckoDriverService#createDefaultService)} server * configuration. * * @see #FirefoxDriver(FirefoxDriverService, FirefoxOptions) */ public FirefoxDriver() { this(new FirefoxOptions()); } /** * Creates a new FirefoxDriver instance with the specified options. * * @param options The options to use. * @see #FirefoxDriver(FirefoxDriverService, FirefoxOptions) */ public FirefoxDriver(FirefoxOptions options) { this(GeckoDriverService.createDefaultService(), options); } /** * Creates a new FirefoxDriver instance. The {@code service} will be started along with the * driver, and shutdown upon calling {@link #quit()}. * * @param service The service to use. * @see RemoteWebDriver#RemoteWebDriver(org.openqa.selenium.remote.CommandExecutor, Capabilities) */ public FirefoxDriver(FirefoxDriverService service) { this(service, new FirefoxOptions()); } public FirefoxDriver(FirefoxDriverService service, FirefoxOptions options) { this(service, options, ClientConfig.defaultConfig()); } public FirefoxDriver( FirefoxDriverService service, FirefoxOptions options, ClientConfig clientConfig) { this(generateExecutor(service, options, clientConfig), options); } private static FirefoxDriverCommandExecutor generateExecutor( FirefoxDriverService service, FirefoxOptions options, ClientConfig clientConfig) { Require.nonNull("Driver service", service); Require.nonNull("Driver options", options); Require.nonNull("Driver clientConfig", clientConfig); Result result = DriverFinder.getPath(service, options); service.setExecutable(result.getDriverPath()); if (result.getBrowserPath() != null && !result.getBrowserPath().isEmpty()) { options.setBinary(result.getBrowserPath()); } return new FirefoxDriverCommandExecutor(service, clientConfig); } private FirefoxDriver(FirefoxDriverCommandExecutor executor, FirefoxOptions options) { this(executor, options, ClientConfig.defaultConfig()); } private FirefoxDriver( FirefoxDriverCommandExecutor executor, FirefoxOptions options, ClientConfig clientConfig) { super(executor, checkCapabilitiesAndProxy(options)); webStorage = new RemoteWebStorage(getExecuteMethod()); extensions = new AddHasExtensions().getImplementation(getCapabilities(), getExecuteMethod()); fullPageScreenshot = new AddHasFullPageScreenshot().getImplementation(getCapabilities(), getExecuteMethod()); context = new AddHasContext().getImplementation(getCapabilities(), getExecuteMethod()); Capabilities capabilities = super.getCapabilities(); HttpClient.Factory factory = HttpClient.Factory.createDefault(); Optional reportedUri = CdpEndpointFinder.getReportedUri("moz:debuggerAddress", capabilities); Optional client = reportedUri.map(uri -> CdpEndpointFinder.getHttpClient(factory, uri)); Optional cdpUri; try { cdpUri = client.flatMap(httpClient -> CdpEndpointFinder.getCdpEndPoint(httpClient)); } catch (Exception e) { try { client.ifPresent(HttpClient::close); } catch (Exception ex) { e.addSuppressed(ex); } throw e; } try { client.ifPresent(HttpClient::close); } catch (Exception e) { LOG.log( Level.FINE, "failed to close the http client used to check the reported CDP endpoint: " + reportedUri.get(), e); } Optional webSocketUrl = Optional.ofNullable((String) capabilities.getCapability("webSocketUrl")); this.biDiUri = webSocketUrl.map( uri -> { try { return new URI(uri); } catch (URISyntaxException e) { LOG.warning(e.getMessage()); } return null; }); this.biDi = createBiDi(biDiUri); this.cdpUri = cdpUri; this.capabilities = cdpUri .map( uri -> new ImmutableCapabilities( new PersistentCapabilities(capabilities) .setCapability("se:cdp", uri.toString()) .setCapability("se:cdpVersion", "85.0"))) .orElse(new ImmutableCapabilities(capabilities)); } @Beta public static RemoteWebDriverBuilder builder() { return RemoteWebDriver.builder().oneOf(new FirefoxOptions()); } /** Check capabilities and proxy if it is set */ private static Capabilities checkCapabilitiesAndProxy(Capabilities capabilities) { if (capabilities == null) { return new ImmutableCapabilities(); } MutableCapabilities caps = new MutableCapabilities(capabilities); // Ensure that the proxy is in a state fit to be sent to the extension Proxy proxy = Proxy.extractFrom(capabilities); if (proxy != null) { caps.setCapability(PROXY, proxy); } return caps; } @Override public Capabilities getCapabilities() { return capabilities; } @Override public void setFileDetector(FileDetector detector) { throw new WebDriverException( "Setting the file detector only works on remote webdriver instances obtained " + "via RemoteWebDriver"); } @Override @Deprecated public LocalStorage getLocalStorage() { return webStorage.getLocalStorage(); } @Override @Deprecated public SessionStorage getSessionStorage() { return webStorage.getSessionStorage(); } @Override public String installExtension(Path path) { Require.nonNull("Path", path); return extensions.installExtension(path); } @Override public String installExtension(Path path, Boolean temporary) { Require.nonNull("Path", path); Require.nonNull("Temporary", temporary); return extensions.installExtension(path, temporary); } @Override public void uninstallExtension(String extensionId) { Require.nonNull("Extension ID", extensionId); extensions.uninstallExtension(extensionId); } /** * Capture the full page screenshot and store it in the specified location. * * @param Return type for getFullPageScreenshotAs. * @param outputType target type, @see OutputType * @return Object in which is stored information about the screenshot. * @throws WebDriverException on failure. */ @Override public X getFullPageScreenshotAs(OutputType outputType) throws WebDriverException { Require.nonNull("OutputType", outputType); return fullPageScreenshot.getFullPageScreenshotAs(outputType); } @Override public FirefoxCommandContext getContext() { return context.getContext(); } @Override public void setContext(FirefoxCommandContext commandContext) { Require.nonNull("Firefox Command Context", commandContext); context.setContext(commandContext); } /** * @deprecated Use W3C-compliant BiDi protocol. Use {{@link #maybeGetBiDi()}} */ @Deprecated @Override public Optional maybeGetDevTools() { if (devTools != null) { return Optional.of(devTools); } if (!cdpUri.isPresent()) { return Optional.empty(); } URI wsUri = cdpUri.orElseThrow( () -> new DevToolsException( "This version of Firefox or geckodriver does not support CDP")); HttpClient.Factory clientFactory = HttpClient.Factory.createDefault(); ClientConfig wsConfig = ClientConfig.defaultConfig().baseUri(wsUri); HttpClient wsClient = clientFactory.createClient(wsConfig); connection = new Connection(wsClient, wsUri.toString()); CdpInfo cdpInfo = new CdpVersionFinder().match("85.0").orElseGet(NoOpCdpInfo::new); devTools = new DevTools(cdpInfo::getDomains, connection); return Optional.of(devTools); } /** * @deprecated Use W3C-compliant BiDi protocol. Use {{@link #getBiDi()}} */ @Deprecated @Override public DevTools getDevTools() { if (!cdpUri.isPresent()) { throw new DevToolsException("This version of Firefox or geckodriver does not support CDP"); } return maybeGetDevTools() .orElseThrow(() -> new DevToolsException("Unable to initialize CDP connection")); } private Optional createBiDi(Optional biDiUri) { if (!biDiUri.isPresent()) { return Optional.empty(); } URI wsUri = biDiUri.orElseThrow( () -> new BiDiException("This version of Firefox or geckodriver does not support BiDi")); HttpClient.Factory clientFactory = HttpClient.Factory.createDefault(); ClientConfig wsConfig = ClientConfig.defaultConfig().baseUri(wsUri); HttpClient wsClient = clientFactory.createClient(wsConfig); org.openqa.selenium.bidi.Connection biDiConnection = new org.openqa.selenium.bidi.Connection(wsClient, wsUri.toString()); return Optional.of(new BiDi(biDiConnection)); } @Override public Optional maybeGetBiDi() { return biDi; } @Override public BiDi getBiDi() { if (!biDiUri.isPresent()) { throw new BiDiException("This version of Firefox or geckodriver does not support Bidi"); } return maybeGetBiDi() .orElseThrow(() -> new BiDiException("Unable to initialize Bidi connection")); } @Override public void quit() { super.quit(); } public static final class SystemProperty { /** System property that defines the location of the Firefox executable file. */ public static final String BROWSER_BINARY = "webdriver.firefox.bin"; /** * System property that defines the profile that should be used as a template. When the driver * starts, it will make a copy of the profile it is using, rather than using that profile * directly. */ public static final String BROWSER_PROFILE = "webdriver.firefox.profile"; } private static class FirefoxDriverCommandExecutor extends DriverCommandExecutor { public FirefoxDriverCommandExecutor(DriverService service) { this(service, ClientConfig.defaultConfig()); } public FirefoxDriverCommandExecutor(DriverService service, ClientConfig clientConfig) { super(service, getExtraCommands(), clientConfig); } private static Map getExtraCommands() { return Stream.of( new AddHasContext().getAdditionalCommands(), new AddHasExtensions().getAdditionalCommands(), new AddHasFullPageScreenshot<>().getAdditionalCommands()) .flatMap((m) -> m.entrySet().stream()) .collect(Collectors.toUnmodifiableMap(Map.Entry::getKey, Map.Entry::getValue)); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy