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

geb.Browser.groovy Maven / Gradle / Ivy

Go to download

Geb (pronounced "jeb") is a browser automation solution. It brings together the power of WebDriver, the elegance of jQuery content selection, the robustness of Page Object modelling and the expressiveness of the Groovy language.

The newest version!
/* Copyright 2009 the original author or authors.
 *
 * Licensed 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 geb

import com.google.common.net.UrlEscapers
import geb.driver.RemoteDriverOperations
import geb.error.IncorrectDriverTypeException
import geb.error.NoNewWindowException
import geb.error.UnexpectedPageException
import geb.error.WebStorageNotSupportedException
import geb.js.JavascriptInterface
import geb.navigator.factory.NavigatorFactory
import geb.report.ReportState
import geb.url.UrlFragment
import geb.waiting.PotentiallyWaitingExecutor
import geb.webstorage.LocalStorage
import geb.webstorage.SessionStorage
import geb.webstorage.WebStorage
import org.openqa.selenium.NoSuchWindowException
import org.openqa.selenium.WebDriver
import org.openqa.selenium.WebDriverException
import org.openqa.selenium.html5.WebStorage as SeleniumWebStorage

import java.time.Duration

import static com.google.common.net.UrlEscapers.urlFragmentEscaper
import static groovy.lang.Closure.DELEGATE_FIRST
import static java.lang.Integer.MAX_VALUE

/**
 * The browser is the centre of Geb. It encapsulates a {@link org.openqa.selenium.WebDriver} implementation and references
 * a {@link geb.Page} object that provides access to the content.
 * 

* Browser objects dynamically delegate all method calls and property read/writes that it doesn't implement to the current * page instance via {@code propertyMissing ( )} and {@code methodMissing ( )}. */ class Browser { protected final static String HAS_NETWORK_CONDITIONS_CLASS_NAME = "org.openqa.selenium.chromium.HasNetworkConditions" public static final String UTF8 = "UTF-8" public static final String QUERY_STRING_SEPARATOR = "&" public static final String CLOSE_OPTION = "close" private Page page private final Configuration config private final RemoteDriverOperations remoteDriverOperations = new RemoteDriverOperations(this.class.classLoader) private String reportGroup = null private NavigatorFactory navigatorFactory = null /** * If the driver is remote, this object allows access to its capabilities (users of Geb should not access this object, it is used internally). */ @Lazy WebDriver augmentedDriver = remoteDriverOperations.getAugmentedDriver(driver) /** * Create a new browser with a default configuration loader, loading the default configuration file. * * @see geb.ConfigurationLoader */ Browser() { this(new ConfigurationLoader().conf) } /** * Create a new browser backed by the given configuration. * * @see geb.Configuration */ Browser(Configuration config) { this.config = config } /** * Creates a new browser instance backed by the given configuration, then applies {@code props} as property overrides on the browser. * * @see geb.Configuration */ Browser(Map props, Configuration config) { this(config) this.metaClass.setProperties(this, props) } /** * Creates a new browser object via the default constructor and executes the closure * with the browser instance as the closure's delegate. * * @return the created browser */ static Browser drive(Closure script) { drive(new Browser(), script) } /** * Creates a new browser with the configuration and executes the closure * with the browser instance as the closure's delegate. * * @return the created browser */ static Browser drive(Configuration conf, Closure script) { drive(new Browser(conf), script) } /** * Creates a new browser with the properties and executes the closure * with the browser instance as the closure's delegate. * * @return the created browser */ static Browser drive(Map browserProperties, Closure script) { drive(new Browser(browserProperties), script) } /** * Executes the closure with browser as its delegate. * * @return browser */ static Browser drive(Browser browser, Closure script) { script.delegate = browser script() browser } /** * Provides access to the current page object. *

* All browser objects are created with a page type of {@link geb.Page} initially. */ Page getPage() { if (page == null) { page = createPage(Page) } this.page } /** * Provides access to the configuration object assoicated with this browser. */ Configuration getConfig() { this.config } /** * The driver implementation used to automate the actual browser. *

* The driver implementation to use is determined by the configuration. * * @see geb.Configuration#getDriver() */ WebDriver getDriver() { config.driver } /** * Returns the factory that creates navigator instances for this browser. * * @return The navigator factory */ NavigatorFactory getNavigatorFactory() { if (navigatorFactory == null) { navigatorFactory = createNavigatorFactory() } navigatorFactory } /** * Set (or change) the webdriver underneath this browser. *

* This should only be called before making any requests as a means to override the driver instance * that would be created from the configuration. Where possible, prefer using the configuration mechanism * to specify the driver implementation to use. *

* This method delegates to {@link geb.Configuration#setDriver(org.openqa.selenium.WebDriver)}. */ void setDriver(WebDriver driver) { config.driver = driver } /** * The url to resolve all relative urls against. Typically the root of the application or system * Geb is interacting with. *

* The base url is determined by the configuration. * * @see geb.Configuration#getBaseUrl() */ String getBaseUrl() { config.baseUrl } /** * Changes the base url used for resolving relative urls. *

* This method delegates to {@link geb.Configuration#setBaseUrl(java.lang.Object)}. */ void setBaseUrl(String baseUrl) { config.baseUrl = baseUrl } /** * Retrieves the current url * * @see org.openqa.selenium.WebDriver#getCurrentUrl() */ String getCurrentUrl() { driver.currentUrl } /** * Delegates the method call directly to the current page object. */ def methodMissing(String name, args) { getPage()."$name"(*args) } /** * Delegates the property access directly to the current page object. */ def propertyMissing(String name) { getPage()."$name" } /** * Delegates the property assignment directly to the current page object. */ def propertyMissing(String name, value) { getPage()."$name" = value } /** * Changes the browser's page to be an instance of the given class. *

* This method performs the following: *

    *
  • Create a new instance of the given class (which must be {@link geb.Page} or a subclass thereof) and connect it to this browser object *
  • Inform any registered page change listeners *
  • Set the browser's page property to the created instance (if it is not already of this type) *
*

* Note that it does not verify that the page matches the current content by running its at checker * * @return an initialized page instance set as the current page */ public T page(Class pageClass) { makeCurrentPage(createPage(pageClass)) } /** * Changes the browser's page to be an instance of the first given type whose at checker returns a true value. *

* This method performs the following: *

    *
  • For each given page type: *
      *
    • Create a new instance of the class (which must be {@link geb.Page} or a subclass thereof) and connect it to the browser object *
    • Test if the page represents the new instance by running its at checker *
    • If the page's at checker is successful: *
        *
      • Inform any registered page change listeners *
      • Set the browser's page property to the instance *
      • Discard the rest of the potentials *
      *
    • If the page's at checker is not successful: *
        *
      • Try the next potential *
      *
    *
  • If none of the pages match: *
      *
    • Check if not at an unexpected page *
    *
* * * @return an initialized page instance set as the current page */ Page page(Class[] potentialPageClasses) { try { verifyPages(potentialPageClasses.collect { createPage(it) }) } catch (UnexpectedPageException e) { checkIfAtAnUnexpectedPage(potentialPageClasses) throw e } } /** * Sets this browser's page to be the given page after initializing it. * *

* This method performs the following: *

    *
  • Connect the instance passed in to this browser object *
  • Inform any registered page change listeners *
  • Set the browser's page property to the instance passed in *

    * Note that it does not verify that the page matches the current content by running its at checker * * @return a page instance passed as the first argument after initializing */ public T page(T page) { makeCurrentPage(initialisePage(page)) page } /** * Changes the browser's page to be an instance of the first given instance whose at checker returns a true value. *

    * This method performs the following: *

      *
    • For each given page instance: *
        *
      • Connects the page instance to the browser object *
      • Test if the page represents the instance by running its at checker *
      • If the page's at checker is successful: *
          *
        • Inform any registered page change listeners *
        • Set the browser's page property to the instance (if it is not already of this type) *
        • Discard the rest of the potentials *
        *
      *
    • If the page's at checker is not successful: *
        *
      • Try the next potential *
      *
    • If none of the pages match: *
        *
      • Check if not at an unexpected page *
      *
    * * @return an initialized page instance set as the current page */ Page page(Page[] potentialPageInstances) { try { verifyPages(potentialPageInstances.collect { initialisePage(it) }) } catch (UnexpectedPageException e) { checkIfAtAnUnexpectedPage(potentialPageInstances) throw e } } /** * Checks if the browser is at the current page by running the at checker for this page type * * A new instance of the page is created for the at check. If the at checker is successful, * this browser object's page instance is updated to the new instance of the given page type and the new instance is returned. * * If the given pageType does not define an at checker, UndefinedAtCheckerException is thrown. * *

    * If implicit assertions * are enabled (which they are by default). This method will only ever return a page instance or throw an {@link AssertionError} * * @return a page instance of the given page type when the at checker succeeded or null otherwise (never null if implicit assertions are enabled) */ public T at(Class pageType) { validatePage(pageType) doAt(createPage(pageType)) } /** * Checks if the browser is at the current page by running the at checker using {@code at(pageType)}. * * If the at checker is successful, the {@code assertions} closure passed as the last argument is executed with the delegate set to an instance of the {@code pageType} class. * If implicit assertions are enabled (which they are by default), each statement in the {@code assertions} closure is implicitly asserted. * * @return the value returned from {@code assertions} closure */ public R at(@DelegatesTo.Target Class pageType, @DelegatesTo(strategy = DELEGATE_FIRST, genericTypeIndex = 0) Closure assertions) { runAssertionsWithPageDelegate(at(pageType), assertions) } /** * Checks if the browser is at the current page by running the at checker for this page instance * * If the at checker is successful, this browser object's page instance is updated to the said instance of the page. * * If the given page object does not define an at checker, UndefinedAtCheckerException is thrown. * *

    * If implicit assertions * are enabled (which they are by default), this method will only ever return a page instance or throw an {@link AssertionError} * * @return a page instance of the passed page after initializing when the at checker succeeded or null otherwise (never null if implicit assertions are enabled) */ public T at(T page) { doAt(page) } /** * Checks if the browser is at the current page by running the at checker using {@code at(page)}. * * If the at checker is successful, the {@code assertions} closure passed as the last argument is executed with the delegate set to the said instance of the page. * If implicit assertions are enabled (which they are by default), each statement in the {@code assertions} closure is implicitly asserted. * * @return the value returned from {@code assertions} closure */ public R at(@DelegatesTo.Target T page, @DelegatesTo(strategy = DELEGATE_FIRST) Closure assertions) { runAssertionsWithPageDelegate(at(page), assertions) } /** * Checks if the browser is at the given page by running the at checker for this page type, suppressing assertion errors. * * If the at checker is successful, this browser object's page instance is updated the one the method is called with. * * If the given pageType does not define an at checker, UndefinedAtCheckerException is thrown. * * If the at check throws an {@link AssertionError} * (as it will when implicit assertions * are enabled) this method will suppress the exception and return false. * * @return true if browser is at the given page otherwise false */ boolean isAt(Class pageType, boolean honourGlobalAtCheckWaiting = true) { isAt(createPage(pageType), honourGlobalAtCheckWaiting) } /** * Checks if the browser is at the given page by running the at checker for this page instance, suppressing assertion errors. * * If the at checker is successful, this browser object's page instance is updated the one the method is called with. * * If the given page instance does not define an at checker, UndefinedAtCheckerException is thrown. * * If the at check throws an {@link AssertionError} * (as it will when implicit assertions * are enabled) this method will suppress the exception and return false. * * @return true if browser is at the given page otherwise false */ boolean isAt(Page page, boolean honourGlobalAtCheckWaiting = true) { initialisePage(page) pageEventListener.beforeAtCheck(this, page) def isAt = page.verifyAtSafely(honourGlobalAtCheckWaiting) if (isAt) { makeCurrentPage(page) pageEventListener.onAtCheckSuccess(this, page) } else { pageEventListener.onAtCheckFailure(this, page) } isAt } /** * Check if at one of the pages configured to be unexpected. * * @param expectedPages allows to specify which of the unexpected pages to ignore for the check * @throws UnexpectedPageException when at an unexpected page */ void checkIfAtAnUnexpectedPage(Class[] expectedPages) { doCheckIfAtAnUnexpectedPage(expectedPages) } /** * Check if at one of the pages configured to be unexpected. * * @param expectedPages allows to specify which of the unexpected pages to ignore for the check * @throws UnexpectedPageException when at an unexpected page */ void checkIfAtAnUnexpectedPage(Page[] expectedPages) { doCheckIfAtAnUnexpectedPage(expectedPages) } /** * Sends the browser to the given url. If it is relative it is resolved against the {@link #getBaseUrl() base url}. */ void go(String url) { go([:], url, null) } /** * Sends the browser to the given url and fragment. If the url is relative it is resolved against the {@link #getBaseUrl() base url}. */ void go(String url, UrlFragment fragment) { go([:], url, fragment) } /** * Sends the browser to the {@link #getBaseUrl() base url} with the given query params and fragment. */ void go(Map params = [:], UrlFragment fragment) { go(params, null, fragment) } /** * Sends the browser to the given url with the given query params and fragment. If the url is relative it is resolved against the {@link #getBaseUrl() base url}. */ void go(Map params = [:], String url = null, UrlFragment fragment = null) { def newUri = calculateUri(url, params, fragment) def currentUri = retrieveCurrentUri() driver.get(newUri.toString()) if (sameUrlWithDifferentFragment(currentUri, newUri)) { driver.navigate().refresh() } if (!page) { page(Page) } } /** * Sends the browser to the given page type's url, sets the page to a new instance of the given type and verifies the at checker of that page. * * @return a page instance of the passed page type when the at checker succeeded * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T to(Map params = [:], Class pageType, Object[] args) { to(params, pageType, null, args) } /** * Sends the browser to the given page type's url, sets the page to a new instance of the given type and verifies the at checker of that page. * * @return a page instance of the passed page type when the at checker succeeded * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T to(Map params = [:], Class pageType, UrlFragment fragment, Object[] args) { to(params, createPage(pageType), fragment, args) } /** * Sends the browser to the given page instance url, sets the page to the given page instance and verifies the at checker of that page. * * @return a page instance of the passed page after initializing when the at checker succeeded * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T to(Map params = [:], T page, Object[] args) { to(params, page, null, args) } /** * Sends the browser to the given page instance url, sets the page to a new instance of the given type and verifies the at checker of that page. * * @return a page instance of the passed page type when the at checker succeeded * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T to(Map params = [:], T page, UrlFragment fragment, Object[] args) { via(params, page, fragment, args) page.shouldVerifyAtImplicitly ? at(page) : page } /** * Sends the browser to the given page type's url and sets the page to a new instance of the given type. * * @return a page instance of the passed page type * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T via(Map params = [:], Class pageType, Object[] args) { via(params, pageType, null, args) } /** * Sends the browser to the given page type's url and sets the page to a new instance of the given type. * * @return a page instance of the passed page type * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T via(Map params = [:], Class pageType, UrlFragment fragment, Object[] args) { via(params, createPage(pageType), fragment, args) } /** * Sends the browser to the given page instance url and sets the page the given instance. * * @return a page instance that was passed after initializing it. * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T via(Map params = [:], T page, Object[] args) { via(params, page, null, args) } /** * Sends the browser to the given page instance url and sets the page the given instance. * * @return a page instance that was passed after initializing it. * @see #page(geb.Page) * @see geb.Page#to(java.util.Map, geb.url.UrlFragment, java.lang.Object) */ public T via(Map params = [:], T page, UrlFragment fragment, Object[] args) { initialisePage(page) page.to(params, fragment, args) page } /** * Clears cookies for the current domain and a number of additional domains by navigating the browser to each of to the urls passed as the argument to this method and clearing cookies for them. * * @param additionalUrls urls for which the cookies should be cleared in addition to the current domain */ void clearCookies(String... additionalUrls) { clearCookies() additionalUrls.each { go(it) clearCookies() } } /** * Clears all cookies for the current domain that the browser has. *

    * If the driven browser has accumulated cookies for additional domains that are to be cleared then {@link #clearCookies(java.lang.String[])} method should be used. * * @see org.openqa.selenium.WebDriver.Options#deleteAllCookies() */ void clearCookies() { driver?.manage()?.deleteAllCookies() } /** * Clears all cookies that the browser currently has, suppressing any webdriver exceptions. */ void clearCookiesQuietly() { try { clearCookies() } catch (WebDriverException e) { // ignore } } /** * Clears web storage, that is both local and session storage for the current domain. */ void clearWebStorage() { localStorage.clear() sessionStorage.clear() } /** * Clears web storage, that is both local and session storage for the current domain, suppressing any * webdriver exceptions. */ void clearWebStorageQuietly() { try { clearWebStorage() } catch (WebDriverException ignore) { } } /** * Quits the driver. * * @see org.openqa.selenium.WebDriver#quit() */ void quit() { driver.quit() } /** * Closes the current driver window. * * @see org.openqa.selenium.WebDriver#close() */ void close() { driver.close() } /** * Retrieves current window * * @see org.openqa.selenium.WebDriver#getWindowHandle() */ String getCurrentWindow() { driver.windowHandle } /** * Retrieves all available windows * * @see org.openqa.selenium.WebDriver#getWindowHandles() */ Set getAvailableWindows() { driver.windowHandles } /** * Executes a closure within the context of a window specified by a name * * @param window name of the window to use as context * @param block closure to be executed in the window context * @return The return value of {@code block} */ public T withWindow(String window, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { withWindow([:], window, block) } /** * Executes a closure within the context of all windows for which the specification * closure returns groovy truth. * * @param specification closure executed once in context of each window, if it returns groovy truth for a given * window then also the block closure is executed in the context of that window * @param block closure to be executed in the window context * @return List containing values returned from {@code block} for each window for which {@code specification} returns groovy truth */ public List withWindow(Closure specification, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { withWindow([:], specification, block) } /** * Executes a closure within the context of all windows for which the specification * closure returns groovy truth. * * @param options a map that can be used to pass additional options * @param specification closure executed once in context of each window, if it returns groovy truth for a given * window then also the block closure is executed in the context of that window * @param block closure to be executed in the window context * @return List containing values returned from {@code block} for each window for which {@code specification} returns groovy truth */ public List withWindow(Map options, Closure specification, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { def anyMatching = false def original = currentWindow def originalPage = getPage() def blockResults = [] try { availableWindows.each { switchToWindow(it) if (specification.call()) { try { blockResults << doWithWindow(options, block) } finally { anyMatching = true } } } } finally { switchToWindow(original) page originalPage } if (!anyMatching) { throw new NoSuchWindowException('Could not find a window that would match the specification') } blockResults } /** * Executes a closure within the context of a window specified by a name * * @param options a map that can be used to pass additional options * @param window name of the window to use as context * @param block closure to be executed in the window context * @return The return value of {@code block} */ public T withWindow(Map options, String window, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { def original = currentWindow def originalPage = getPage() switchToWindow(window) try { doWithWindow(options, block) } finally { switchToWindow(original) page originalPage } } /** * Expects the first closure argument to open a new window and calls the second closure argument in the context * of the newly opened window. A map of options can also be specified that allows to close the new window, switch to a * different page for closure executed in the context of the new window and also to wait for the window opening if the * window opening is asynchronous. * * @param options a map that can be used to pass additional options * @param windowOpeningBlock a closure that should open a new window * @param block closure to be executed in the new window context * @return The return value of {@code block} * @throws geb.error.NoNewWindowException if the window opening closure doesn't open one or opens more * than one new window */ public T withNewWindow(Map options, Closure windowOpeningBlock, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { def originalWindow = currentWindow def originalPage = getPage() def wait = options.containsKey("wait") ? options.wait : config.withNewWindowConfig.wait def newWindow = executeNewWindowOpening(windowOpeningBlock, wait) try { switchToWindow(newWindow) if (options.page) { verifyAtImplicitly(options.page) } Closure cloned = block.clone() cloned.delegate = browser cloned.resolveStrategy = DELEGATE_FIRST cloned.call() } finally { if ((!options.containsKey(CLOSE_OPTION) && config.withNewWindowConfig.close.orElse(true)) || options.close) { driver.close() } switchToWindow(originalWindow) page originalPage } } /** * Expects the first closure argument to open a new window and calls the second closure argument in the context * of the newly opened window. * * @param windowOpeningBlock a closure that should open a new window * @param block closure to be executed in the new window context * @return The return value of {@code block} * @throws geb.error.NoNewWindowException if the window opening closure doesn't open one or opens more * than one new window */ public T withNewWindow(Closure windowOpeningBlock, @DelegatesTo(value = Browser, strategy = DELEGATE_FIRST) Closure block) { withNewWindow([:], windowOpeningBlock, block) } /** * Creates a new instance of the given page type and initialises it. * * @return The newly created page instance */ public T createPage(Class pageType) { validatePage(pageType) initialisePage(pageType.newInstance()) } /** * Returns a newly created javascript interface connected to this browser. */ JavascriptInterface getJs() { new JavascriptInterface(this) } /** * The directory that will be used for the {@link #report(java.lang.String) method}. *

    * Uses the {@link geb.Configuration#getReportsDir()} for the base location for reports (throwing an exception if this is not set), and * appending the current report group. The returned file object is guaranteed to exist on the filesystem. *

    * If the current report group is {@code null}, this method returns the same as {@code config.reportsDir}. * * @see #reportGroup(java.lang.String) * @see #report(java.lang.String) */ File getReportGroupDir() { def reportsDir = config.reportsDir if (reportsDir == null) { throw new IllegalStateException("No reports dir has been configured, you need to set in the config file or via the build adapter.") } def reportGroupDir = reportGroup ? new File(reportsDir, reportGroup) : reportsDir if (!(reportGroupDir.mkdirs() || reportGroupDir.exists())) { throw new IllegalStateException("Could not create report group dir '${reportGroupDir}'") } reportGroupDir } /** * Sets the "group" for all subsequent reports, which is the relative path inside the reports dir that reports will be written to. * * @param path a relative path, or {@code null} to have reports written to the base reports dir */ void reportGroup(String path) { reportGroup = path } /** * Sets the report group to be the full name of the class, replacing "." with "/". * * @see #reportGroup(String) */ void reportGroup(Class clazz) { reportGroup(clazz.name.replace('.', '/')) } /** * Removes the directory returned by {@link #getReportGroupDir()} from the filesystem if it exists. */ void cleanReportGroupDir() { def dir = getReportGroupDir() if (dir != null) { if (dir.exists() && !dir.deleteDir()) { throw new IllegalStateException("Could not clean report dir '${dir}'") } } } /** * Writes a snapshot of the browser's state to the current {@link #getReportGroupDir()} using * the {@link geb.Configuration#getReporter() config's reporter}. * * @param label The name for the report file (should not include a file extension) */ void report(String label) { config.reporter.writeReport(new ReportState(this, label, getReportGroupDir())) } /** * Indefinitely waits for {@code geb.unpause} javascript variable to be set to {@code true} in the driven browser. *

    * Can be used as an alternative to setting a breakpoint and running the JVM in debug mode for debugging purposes. */ void pause() { js.exec ''' if (!window.geb) { window.geb = {}; } window.geb.unpause = false; ''' waitFor(MAX_VALUE) { js.'geb.unpause' } } /** * Returns an object that allows access to and manipulation of local storage. * * @see geb.webstorage.WebStorage */ WebStorage getLocalStorage() { new LocalStorage(seleniumWebStorage) } /** * Returns an object that allows access to and manipulation of session storage. * * @see geb.webstorage.WebStorage */ WebStorage getSessionStorage() { new SessionStorage(seleniumWebStorage) } void verifyAtImplicitly(Class targetPage) { verifyAtImplicitly(createPage(targetPage)) } void verifyAtImplicitly(Page targetPage) { initialisePage(targetPage) if (targetPage.shouldVerifyAtImplicitly) { if (!at(targetPage)) { throw new UnexpectedPageException(targetPage) } } else { page(targetPage) } } /** * Allows to introduce network latency to the browser being driven. *

    * Helpful for exposing network related timing issues in automation code under development. *

    * Only supported when using a driver implementing or augmentable to org.openqa.selenium.chromium.HasNetworkConditions. * * @throws geb.error.IncorrectDriverTypeException When called while using a driver which does not implement and is not augmentable to org.openqa.selenium.chromium.HasNetworkConditions */ void setNetworkLatency(Duration duration) { def params = [ network_conditions: [ latency : duration.toMillis(), throughput: MAX_VALUE >> 2 ] ] remoteDriverOperations.executeCommand(driverWithNetworkConditions, "setNetworkConditions", params) } /** * Removes any previously configured network latency. * * @throws geb.error.IncorrectDriverTypeException When called while using a driver which does not implement and is not augmentable to org.openqa.selenium.chromium.HasNetworkConditions */ void resetNetworkLatency() { remoteDriverOperations.executeCommand(driverWithNetworkConditions, "deleteNetworkConditions") } public Optional driverAs(Class castType) { if (castType.isInstance(driver)) { Optional.of(driver as T) } else if (castType.isInstance(augmentedDriver)) { Optional.of(augmentedDriver as T) } else { Optional.empty() } } protected switchToWindow(String window) { driver.switchTo().window(window) } protected T doWithWindow(Map options, Closure block) { try { if (options.page) { verifyAtImplicitly(options.page) } Closure cloned = block.clone() cloned.delegate = browser cloned.resolveStrategy = DELEGATE_FIRST cloned.call() } finally { if (options.close || (!options.containsKey(CLOSE_OPTION) && config.withWindowConfig.close)) { driver.close() } } } /** * Called to create the navigator factory, the first time it is requested. * * @return The navigator factory */ protected NavigatorFactory createNavigatorFactory() { config.createNavigatorFactory(this) } private WebDriver getDriverWithNetworkConditions() { optionalHasNetworkConditionsClass.map { hasNetworkConditionsClass -> driverAs(hasNetworkConditionsClass).orElse(null) }.orElseThrow { def message = "This operation is only possible on driver instances implementing or augmentable to " + "${HAS_NETWORK_CONDITIONS_CLASS_NAME}" new IncorrectDriverTypeException(message) } } private Optional> getOptionalHasNetworkConditionsClass() { try { Optional.of(getClass().classLoader.loadClass(HAS_NETWORK_CONDITIONS_CLASS_NAME)) } catch (ClassNotFoundException e) { Optional.empty() } } private String executeNewWindowOpening(Closure windowOpeningBlock, wait) { def originalWindows = availableWindows windowOpeningBlock.call() if (wait) { config.getWaitForParam(wait).waitFor { (availableWindows - originalWindows).size() == 1 } } def newWindows = (availableWindows - originalWindows) as List if (newWindows.size() != 1) { def message = newWindows ? 'There has been more than one window opened' : 'No new window has been opened' throw new NoNewWindowException(message) } newWindows.first() } private SeleniumWebStorage getSeleniumWebStorage() { driverAs(SeleniumWebStorage).orElseThrow { new WebStorageNotSupportedException() } } private String toQueryString(Map params) { def escaper = UrlEscapers.urlFormParameterEscaper() if (params) { params.collectMany { name, value -> def values = value instanceof Collection ? value : [value] values.collect { v -> "${escaper.escape(name.toString())}=${escaper.escape(v.toString())}" } }.join(QUERY_STRING_SEPARATOR) } else { null } } private String getBaseUrlRequired() { def baseUrl = getBaseUrl() if (baseUrl == null) { throw new geb.error.NoBaseUrlDefinedException() } baseUrl } private URI calculateUri(String path, Map params, UrlFragment fragment) { def absolute = calculateAbsoluteUri(path) def effectiveFragment = urlFragmentEscaper().escape(fragment?.toString() ?: "") ?: absolute.rawFragment if (absolute.opaque) { new URI(absolute.scheme, absolute.schemeSpecificPart, effectiveFragment) } else { def uriStringBuilder = new StringBuilder() << new URI( absolute.scheme, absolute.userInfo, absolute.host, absolute.port, absolute.path, null, null ) def queryString = [absolute.rawQuery, toQueryString(params)].findAll().join(QUERY_STRING_SEPARATOR) ?: null if (queryString) { uriStringBuilder << "?" << queryString } if (effectiveFragment) { uriStringBuilder << "#" << effectiveFragment } new URI(uriStringBuilder.toString()) } } private URI calculateAbsoluteUri(String path) { def absolute if (path) { absolute = new URI(path) if (!absolute.absolute) { absolute = new URI(getBaseUrlRequired()).resolve(absolute) } } else { absolute = new URI(getBaseUrlRequired()) } absolute } private Page verifyPages(List pages) { new PotentiallyWaitingExecutor(config.atCheckWaiting).execute { Map pageVerificationResults = [:] def match = pages.find { AtVerificationResult atVerificationResult = it.getAtVerificationResult(false) if (!atVerificationResult) { pageVerificationResults.put(it, atVerificationResult) } atVerificationResult } if (match) { makeCurrentPage(match) } else { throw new UnexpectedPageException(pageVerificationResults) } } } private doCheckIfAtAnUnexpectedPage(def expectedPages) { config.unexpectedPages.each { def pageInstance = createPage(it) if (isAt(pageInstance, false)) { pageEventListener.unexpectedPageEncountered(this, pageInstance) throw new UnexpectedPageException(pageInstance, *expectedPages) } } } /** * Sets this browser's page to be the given page, which has already been initialised with this browser instance. *

    * Any page change listeners are informed and {@link geb.Page#onUnload(geb.Page)} is called * on the previous page and {@link geb.Page#onLoad(geb.Page)} is called on the incoming page. *

    */ private Page makeCurrentPage(Page newPage) { if (newPage != page) { pageEventListener.pageWillChange(this, page, newPage) page?.onUnload(newPage) def previousPage = page page = newPage page.onLoad(previousPage) } page } private T initialisePage(T page) { if (!this.is(page.browser)) { page.init(this) } page } /** * Runs a page's at checker, expecting the page to be initialised with this browser instance. */ private T doAt(T page) { initialisePage(page) pageEventListener.beforeAtCheck(this, page) def atResult try { atResult = page.verifyAt() } catch (AssertionError e) { pageEventListener.onAtCheckFailure(this, page) throw e } if (atResult) { makeCurrentPage(page) pageEventListener.onAtCheckSuccess(this, page) page } else { pageEventListener.onAtCheckFailure(this, page) null } } private void validatePage(Class pageType) { if (!Page.isAssignableFrom(pageType)) { throw new IllegalArgumentException("$pageType is not a subclass of ${Page}") } } private R runAssertionsWithPageDelegate(Page page, Closure assertions) { Closure clone = assertions.clone() clone.delegate = page clone.call() } private URI retrieveCurrentUri() { def currentUri = null try { def currentUrl = driver.currentUrl currentUri = currentUrl ? new URI(currentUrl) : null } catch (NullPointerException npe) { } catch (URISyntaxException use) { } catch (WebDriverException webDriverException) { if (!webDriverException.message.contains("Remote browser did not respond to getCurrentUrl")) { throw webDriverException } } currentUri } private boolean sameUrlWithDifferentFragment(URI current, URI next) { current && next.fragment && ignoreFragment(current) == ignoreFragment(next) } private URI ignoreFragment(URI uri) { new URI(uri.scheme, uri.schemeSpecificPart, null) } private PageEventListener getPageEventListener() { config.pageEventListener } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy