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

nl.hsac.fitnesse.sample.DevToolsBrowserTest Maven / Gradle / Ivy

There is a newer version: 5.3.17
Show newest version
package nl.hsac.fitnesse.sample;

import fitnesse.util.Base64;
import nl.hsac.fitnesse.fixture.slim.SlimFixtureException;
import nl.hsac.fitnesse.fixture.slim.StopTestException;
import nl.hsac.fitnesse.fixture.slim.web.BrowserTest;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.devtools.DevTools;
import org.openqa.selenium.devtools.HasDevTools;
import org.openqa.selenium.devtools.v129.emulation.Emulation;
import org.openqa.selenium.devtools.v129.fetch.Fetch;
import org.openqa.selenium.devtools.v129.fetch.model.RequestPattern;
import org.openqa.selenium.devtools.v129.fetch.model.RequestStage;
import org.openqa.selenium.devtools.v129.log.Log;
import org.openqa.selenium.devtools.v129.network.Network;
import org.openqa.selenium.devtools.v129.network.model.Cookie;
import org.openqa.selenium.devtools.v129.network.model.CookiePriority;
import org.openqa.selenium.devtools.v129.network.model.CookieSameSite;
import org.openqa.selenium.devtools.v129.network.model.Headers;
import org.openqa.selenium.devtools.v129.network.model.RequestId;
import org.openqa.selenium.devtools.v129.network.model.RequestWillBeSent;
import org.openqa.selenium.devtools.v129.network.model.ResourceType;
import org.openqa.selenium.devtools.v129.network.model.ResponseReceived;
import org.openqa.selenium.devtools.v129.network.model.TimeSinceEpoch;
import org.openqa.selenium.devtools.v129.page.Page;
import org.openqa.selenium.devtools.v129.performance.Performance;
import org.openqa.selenium.devtools.v129.performance.model.Metric;
import org.openqa.selenium.devtools.v129.runtime.Runtime;
import org.openqa.selenium.devtools.v129.security.Security;
import org.openqa.selenium.remote.Augmenter;

import java.math.BigInteger;
import java.sql.Timestamp;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;

import static java.util.Collections.singletonList;
import static java.util.Optional.empty;

/**
 * Examples
 *
 * @param 
 */
public class DevToolsBrowserTest extends BrowserTest {
    private DevTools devTools;
    private final List logEntries = new ArrayList<>();
    private final List requests = new ArrayList<>();
    private final List responses = new ArrayList<>();
    private final List requestPatternsToOverride = new ArrayList<>();

    /**
     * Experimental class leveraging selenium 4's devTools api's. Use with devTools enabled browser
     * Current api: v106
     */

    public DevToolsBrowserTest() {
        super();
        ensureDevToolsEnabledDriver();
        enableConsoleLogCapturing();
    }

    public DevToolsBrowserTest(int secondsBeforeTimeout) {
        super(secondsBeforeTimeout);
        ensureDevToolsEnabledDriver();
        enableConsoleLogCapturing();
    }

    public DevToolsBrowserTest(int secondsBeforeTimeout, boolean confirmAlertIfAvailable) {
        super(secondsBeforeTimeout, confirmAlertIfAvailable);
        ensureDevToolsEnabledDriver();
        enableConsoleLogCapturing();
    }

    private void ensureDevToolsEnabledDriver() {
        WebDriver driver = new Augmenter().augment(getSeleniumHelper().driver());
        if (!(driver instanceof HasDevTools)) {
            throw new StopTestException(false, "DevTools enabled Browser Test can only be used with a chromium based browser (Chrome/Edge)");
        }
        devTools = ((HasDevTools) driver).getDevTools();
        devTools.createSessionIfThereIsNotOne();
        devTools.send(Network.enable(Optional.of(100000), Optional.of(100000), Optional.of(100000)));
    }

    /**
     * Inject custom request headers in each request the browser sends
     *
     * @param headers A map conaining headers as key:value
     */
    public void injectHttpHeaders(Map headers) {
        devTools.send(Network.setExtraHTTPHeaders(new Headers(headers)));
    }

    /**
     * Clear the browser's cache
     */
    public void clearBrowserCache() {
        devTools.send(Network.clearBrowserCache());
    }

    /**
     * Delete ALL cookies (not limited to current domain)
     */
    public void clearBrowserCookies() {
        devTools.send(Network.clearBrowserCookies());
    }

    /**
     * Get a map containing all cookies for the given URL
     *
     * @param url The url to list cookies for
     * @return a map containing the cookies as name:value pairs
     */
    public Map getCookiesForUrl(String url) {
        Map result = new HashMap<>();
        List cookies = devTools.send(Network.getCookies(Optional.of(singletonList(url))));
        for (Cookie c : cookies) {
            result.put(c.getName(), c.getValue());
        }
        return result.size() > 0 ? result : null;
    }

    /**
     * Set a cookie for a specific domain
     *
     * @param name   The name of the cookie
     * @param value  The cookie value
     * @param domain The domain the cookie is valid for
     */
    public void setCookieWithValueForDomain(String name, String value, String domain) {
        devTools.send(Network.setCookie(name, value, empty(), Optional.of(domain),
                empty(), empty(), empty(), empty(), empty(), empty(),
                empty(), empty(), empty(), empty()));
    }

    /**
     * Set a cookie with the possibility to set all properties
     *
     * @param cookieData a map containing key/value for name, value, url, domain, path, secure, httpOnly, sameSite, expires, priority
     *                   Only name and value are mandatory, the others are optional.
     */
    public void setCookieFromMap(Map cookieData) {
        devTools.send(Network.setCookie(
                String.valueOf(cookieData.get("name")),
                String.valueOf(cookieData.get("value")),
                Optional.of(String.valueOf(cookieData.get("url"))),
                Optional.of(String.valueOf(cookieData.get("domain"))),
                Optional.of(String.valueOf(cookieData.get("path"))),
                Optional.of(Boolean.valueOf(cookieData.get("secure").toString())),
                Optional.of(Boolean.valueOf(cookieData.get("httpOnly").toString())),
                Optional.of(CookieSameSite.fromString(String.valueOf(cookieData.get("sameSite")))),
                Optional.of(new TimeSinceEpoch(new BigInteger(String.valueOf(cookieData.get("expires"))))),
                Optional.of(CookiePriority.fromString(String.valueOf(cookieData.get("priority")))),
                empty(),
                empty(),
                empty(),
                empty()
        ));
    }

    /**
     * Toggle browser caching
     *
     * @param disable true to disable caching, false to re-enable it
     */
    public void disableBrowserCache(boolean disable) {
        devTools.send(Network.setCacheDisabled(disable));
    }

    /**
     * Mock geo location and accuracy
     *
     * @param lat      Latitude
     * @param lon      Longitude
     * @param accuracy Accuracy
     */
    public void setGeoLocationLatLonAccuracy(String lat, String lon, String accuracy) {
        devTools.send(Emulation.setGeolocationOverride(Optional.of(Double.valueOf(lat)), Optional.of(Double.valueOf(lon)), Optional.of(Integer.valueOf(accuracy))));
    }

    /**
     * Mock geo location with a default accuracy of 1.
     *
     * @param lat Latitude
     * @param lon Longitude
     */
    public void setGeoLocationLatLon(String lat, String lon) {
        setGeoLocationLatLonAccuracy(lat, lon, "1");
    }

    /**
     * Override the user agent string
     *
     * @param userAgent The useragent String to identify with
     */
    public void setUserAgent(String userAgent) {
        devTools.send(Network.setUserAgentOverride(userAgent, empty(), empty(), empty()));
    }

    /**
     * Take a screenshot of the full web page, by mocking the viewport size to the current content's size
     *
     * @param baseName The base name of the screensho file
     * @return A link to the screenshot file
     */
    public String takeFullPageScreenshot(String baseName) {
        Page.GetLayoutMetricsResponse layoutMetrics = devTools.send(Page.getLayoutMetrics());

        devTools.send(Emulation.setDeviceMetricsOverride(
                layoutMetrics.getContentSize().getWidth().intValue(),
                layoutMetrics.getContentSize().getHeight().intValue(), 1, false,
                empty(), empty(), empty(), empty(), empty(), empty(), empty(), empty(), empty(), empty()));

        String result = takeScreenshot(baseName);

        devTools.send(Emulation.clearDeviceMetricsOverride());

        return result;
    }

    /**
     * Ignore SSL errors
     *
     * @param ignore true to ignore, false for default behaviour
     */
    public void ignoreCertificateErrors(boolean ignore) {
        devTools.send(Security.setIgnoreCertificateErrors(ignore));
    }

    /**
     * Retrieve the console output
     *
     * @return Log entries, one on each line
     */
    public String consoleLog() {
        return formattedConsoleLog(logEntries);
    }

    private String formattedConsoleLog(List logEntryList) {
        devTools.send(Log.enable());
        String logView = logEntryList.stream().map(entry -> entry + "\r\n").collect(Collectors.joining());
        return logView;
    }

    /**
     * Clear the list of logEntries
     */
    public void clearConsoleLog() {
        logEntries.clear();
    }

    /**
     * Start collecting performance metrics
     */
    public void startPerformanceMetrics() {
        devTools.send(Performance.enable(Optional.of(Performance.EnableTimeDomain.TIMETICKS)));
    }

    /**
     * Stop collecting performance metrics
     */
    public void stopPerformanceMetrics() {
        devTools.send(Performance.disable());
    }

    /**
     * Get the collected performance metrics as a key:value map
     *
     * @return All collected metrics
     */
    public Map performanceMetrics() {
        return devTools.send(Performance.getMetrics()).stream()
                .collect(Collectors.toMap(Metric::getName, Metric::getValue, (a, b) -> b));
    }

    /**
     * Get a specific metric by name
     *
     * @param metric the metric to retrieve
     * @return a number representing the metric's measurement
     */
    public Number performanceMetric(String metric) {
        return devTools.send(Performance.getMetrics()).stream().filter(m ->
                m.getName().equalsIgnoreCase(metric)).findFirst().map(Metric::getValue).orElse(null);
    }

    /**
     * Keep track of requests and responses during the session
     */
    public void startRequestLogging() {
        devTools.addListener(Network.requestWillBeSent(), r -> requests.add(r));
        devTools.addListener(Network.responseReceived(), r -> responses.add(r));
    }

    /**
     * Keep track of requets and responses for a specific ResourceType during the session.
     *
     * @param resourceType The resource type to log. Valid options: Document, Stylesheet, Image, Media, Font, Script,
     *                     TextTrack, XHR, Fetch, EventSource, WebSocket, Manifest, SignedExchange, Ping,
     *                     CSPViolationReport, Other
     */
    public void startRequestLoggingFor(String resourceType) {
        devTools.addListener(Network.requestWillBeSent(), requestWillBeSent -> {
            if (requestWillBeSent.getType().orElse(ResourceType.OTHER).equals(ResourceType.fromString(resourceType))) {
                requests.add(requestWillBeSent);
            }
        });
        devTools.addListener(Network.responseReceived(), responseReceived -> {
            if (responseReceived.getType().equals(ResourceType.fromString(resourceType))) {
                responses.add(responseReceived);
            }
        });
    }

    /**
     * Currently only shows browser's own messages (i.e. XSS warnings)
     * Cannot use Runtime.ConsoleApiCalled , as the fromJson of RemoteObject is broken in alpha 6.
     * Await new version that uses the idealized version of the CDP impl.
     */
    private void enableConsoleLogCapturing() {
        //TODO: Add runtime Console API call logging when new version is released
        devTools.send(Log.enable());
        devTools.addListener(Runtime.consoleAPICalled(), call -> {
            String entry = call.getType().toString() + "; " +
                    new Timestamp(System.currentTimeMillis()).getTime() + "; " +
                    call.getArgs().stream().map(Object::toString).collect(Collectors.joining());

            logEntries.add(entry);
        });
        devTools.addListener(Log.entryAdded(), logEntry -> {
            String entry = logEntry.getLevel().name() + "; " +
                    logEntry.getTimestamp().toString() + "; " +
                    logEntry.getText();
            logEntries.add(entry);
        });

    }

    /**
     * Override the response for a given URL pattern.
     * This method can not be used in combination with override authentication
     *
     * @param urlPattern   The url pattern to match. Wildcards ('*' -> zero or more, '?' -> exactly one) are allowed.
     *                     Escape character is backslash.
     * @param type         The resource type. Allowed Values: Document, Stylesheet, Image, Media, Font, Script, TextTrack, XHR,
     *                     Fetch, EventSource, WebSocket, Manifest, SignedExchange, Ping, CSPViolationReport, Other
     * @param responseBody The response to inject
     */
    public void overrideResponseForUrlOfTypeWith(String urlPattern, String type, String responseBody) {
        requestPatternsToOverride.add(new RequestPattern(
                Optional.of(urlPattern),
                Optional.of(ResourceType.fromString(type)),
                Optional.of(RequestStage.RESPONSE)));

        devTools.addListener(Fetch.requestPaused(), requestPaused -> {
            if (requestPaused.getRequest().getUrl().matches(wildcardAsRegex(urlPattern))) {
                devTools.send(Fetch.fulfillRequest(
                        requestPaused.getRequestId(),
                        requestPaused.getResponseStatusCode().orElse(200),
                        requestPaused.getResponseHeaders(),
                        empty(),
                        Optional.of(Base64.encode(responseBody)),
                        empty()));
            } else {
                devTools.send(Fetch.continueRequest(requestPaused.getRequestId(), empty(), empty(), empty(), empty(), empty()));
            }
        });
    }

    /**
     * Enable response- or authentication overrides
     */
    public void enableNetworkOverrides() {
        Optional> patterns = requestPatternsToOverride.isEmpty() ? empty() : Optional.of(requestPatternsToOverride);
        boolean overrideAuth = false;
        devTools.send(Fetch.enable(patterns, Optional.of(overrideAuth)));
    }

    /**
     * Clear any response- or authentication overrides
     */
    public void clearNetworkOverrides() {
        devTools.clearListeners();
        requestPatternsToOverride.clear();
        devTools.send(Fetch.disable());
    }

    /**
     * Get the response body of a request to a url that matches the pattern
     *
     * @param requestUrlPattern The url pattern to match. Wildcards ('*' -> zero or more, '?' -> exactly one) are allowed.
     *                          Escape character is backslash.
     * @return the response body as a String
     */
    public String responseBodyOfRequestTo(String requestUrlPattern) {
        RequestId id = getRequestIdForUrl(requestUrlPattern);
        Network.GetResponseBodyResponse body = devTools.send(Network.getResponseBody(id));
        return body.getBody();
    }

    /**
     * Get the response status of a request to a url that matches the pattern
     *
     * @param requestUrlPattern The url pattern to match. Wildcards ('*' -> zero or more, '?' -> exactly one) are allowed.
     *                          Escape character is backslash.
     * @return the response status code (i.e. 200 for OK)
     */
    public int responseStatusOfRequestTo(String requestUrlPattern) {
        RequestId id = getRequestIdForUrl(requestUrlPattern);
        return responses.stream().filter(response -> response.getRequestId().toString().equals(id.toString())).findFirst()
                .orElseThrow(() -> new SlimFixtureException("No response found for: " + requestUrlPattern)).getResponse().getStatus();
    }

    /**
     * Get a list of logged urls that were requested
     *
     * @return list of request URL's
     */
    public List requestLog() {
        return requests.stream().map(request -> request.getRequest().getUrl()).collect(Collectors.toList());
    }

    private RequestId getRequestIdForUrl(String requestUrlPattern) {
        for (RequestWillBeSent r : requests) {
            if (r.getRequest().getUrl().matches(wildcardAsRegex(requestUrlPattern))) {
                return r.getRequestId();
            }
        }
        throw new SlimFixtureException(false, "No request found for: " + requestUrlPattern);
    }

    /**
     * Convert a wildcard pattern to a regex pattern so we can match consistently
     *
     * @param wildcardPattern The pattern to convert
     * @return a regex String
     */
    private String wildcardAsRegex(String wildcardPattern) {
        return "\\Q" + wildcardPattern
                .replace("?", "\\E.\\Q")
                .replace("*", "\\E.*\\Q") +
                "\\E";
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy