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

com.seleniumtests.driver.CustomEventFiringWebDriver Maven / Gradle / Ivy

The newest version!
/**
 * Orignal work: Copyright 2015 www.seleniumtests.com
 * Modified work: Copyright 2016 www.infotel.com
 * 				Copyright 2017-2019 B.Hecquet
 *
 * 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 com.seleniumtests.driver;

import com.neotys.selenium.proxies.NLWebDriver;
import com.seleniumtests.browserfactory.BrowserInfo;
import com.seleniumtests.browserfactory.SeleniumRobotCapabilityType;
import com.seleniumtests.connectors.selenium.SeleniumGridConnector;
import com.seleniumtests.core.StatisticsStorage;
import com.seleniumtests.core.StatisticsStorage.DriverUsage;
import com.seleniumtests.customexception.RetryableDriverException;
import com.seleniumtests.customexception.ScenarioException;
import com.seleniumtests.customexception.WebSessionEndedException;
import com.seleniumtests.util.helper.WaitHelper;
import com.seleniumtests.util.logging.SeleniumRobotLogger;
import com.seleniumtests.util.osutility.OSUtilityFactory;
import com.seleniumtests.util.video.VideoRecorder;
//import net.lightbody.bmp.BrowserMobProxy;
import io.appium.java_client.AppiumDriver;
import io.appium.java_client.ExecutesMethod;
import io.appium.java_client.HidesKeyboard;
import io.appium.java_client.remote.SupportsContextSwitching;
import org.apache.commons.codec.binary.Base64;
import org.apache.commons.codec.binary.Base64OutputStream;
import org.apache.commons.io.FileUtils;
import org.apache.logging.log4j.Logger;
import org.mockito.Mockito;
import org.openqa.selenium.*;
import org.openqa.selenium.interactions.Interactive;
import org.openqa.selenium.interactions.Sequence;
import org.openqa.selenium.remote.*;
import org.openqa.selenium.safari.SafariDriver;
import org.openqa.selenium.support.events.EventFiringDecorator;
import org.openqa.selenium.support.events.WebDriverListener;

import javax.annotation.Nullable;
import javax.imageio.ImageIO;
import java.awt.AWTError;
import java.awt.AWTException;
import java.awt.GraphicsDevice;
import java.awt.GraphicsEnvironment;
import java.awt.HeadlessException;
import java.awt.MouseInfo;
import java.awt.PointerInfo;
import java.awt.Rectangle;
import java.awt.Robot;
import java.awt.Toolkit;
import java.awt.datatransfer.StringSelection;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.file.Paths;
import java.util.*;

/**
 * This class acts as a proxy for everything related to selenium driver actions (mostly a bypass) or with the machine holding
 * the browser
 * - send keys via keyboard
 * - move mouse
 * - upload file to browser
 * - capture video
 * 
 * When action do not need a real driver, static methods are provided
 * It also handles the grid mode, masking it to requester.
 */
public class CustomEventFiringWebDriver implements HasCapabilities, WebDriver, JavascriptExecutor, TakesScreenshot, Interactive, SupportsContextSwitching, HidesKeyboard {
	
	private static final String OTHER_BROWSER = "other";
	private static final String SAFARI_BROWSER = "safari";
	private static final Logger logger = SeleniumRobotLogger.getLogger(CustomEventFiringWebDriver.class);
	private static Set videoRecorders = Collections.synchronizedSet(new HashSet<>());
    private FileDetector fileDetector = new UselessFileDetector();
    private static final int MAX_DIMENSION = 100000;
    private Set currentHandles;
    private final List driverPids;
	private WebDriver driver;
	private final WebDriver originalDriver;
	private final NLWebDriver neoloadDriver;
	private TestType testType;
	private boolean driverExited = false;
	private final DriverMode driverMode;
	private final BrowserInfo browserInfo;
//	private final BrowserMobProxy mobProxy;
	private SeleniumGridConnector gridConnector;
	private final Integer attachExistingDriverPort;
	private MutableCapabilities internalCapabilities = new MutableCapabilities();
	
	/*
	 * returns pixel aspect ratio of web site, it depends on OS and browser zoom
	 * We can force to use PAR=1 by calling driver.executeScript("Some JS", false)
	 * Cases which need the real PAR:
	 * - Snapshot capture
	 * - operations on ScreenZon
	 * - more generally, every operation where we interact with screen
	 * Cases which need PAR=1
	 * - operations on PictureElement
	 * - more generally, every operation where we interact with selenium elements which do not take PAR into account
	 */
	private static final String JS_PIXEL_RATIO = "var pixelRatio;"
			+ "try{"
			+ "    if (arguments.length==0 || arguments[0]) {"				
			+ "        pixelRatio = devicePixelRatio"
			+ "    } else {"
			+ "        pixelRatio = 1"
			+ "    }"
			+ "} catch(err){"
			+ "    pixelRatio=1"
			+ "}";
    
    private static final String JS_GET_VIEWPORT_SIZE_WIDTH = String.format(
    		JS_PIXEL_RATIO
    				+ "var width = %d;"
    				+ "if (window.innerWidth) {"
    				+ "   width = Math.min(window.innerWidth, width);"
    				+ "} "
    				+ "if (document.documentElement && document.documentElement.clientWidth) {"
    				+ "   width = Math.min(document.documentElement.clientWidth, width);"
    				+ "} "
    				+ "var b = document.getElementsByTagName('html')[0]; "
    				+ "if (b.clientWidth) {"
    				+ "   width = Math.min(b.clientWidth, width);"
    				+ "}"
    				+ "return width * pixelRatio;", MAX_DIMENSION);
    
    private static final String JS_GET_VIEWPORT_SIZE_HEIGHT = String.format(
    				JS_PIXEL_RATIO
    				+ "var height = %d;"
    				+ "if (window.innerHeight) {"
    				+ "    height = Math.min(window.innerHeight, height);"
    				+ "}"
    				+ "if (document.documentElement && document.documentElement.clientHeight) {"
    				+ "    height = Math.min(document.documentElement.clientHeight, height);"
    				+ "}"
    				+ "var b = document.getElementsByTagName('html')[0]; "
    				+ "if (b.clientHeight) {"
    				+ "    height = Math.min(b.clientHeight, height);"
    				+ "}"
    				+ "return height * pixelRatio;", MAX_DIMENSION);

    private static final String JS_GET_CURRENT_SCROLL_POSITION =
            "var doc = document.documentElement; "
            + "var x = window.scrollX || ((window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0));"
            + "var y = window.scrollY || ((window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0));"
            + "return [x, y];";


    // IMPORTANT: Notice there's a major difference between scrollWidth
    // and scrollHeight. While scrollWidth is the maximum between an
    // element's width and its content width, scrollHeight might be
    // smaller (!) than the clientHeight, which is why we take the
    // maximum between them.
    private static final String JS_GET_CONTENT_ENTIRE_SIZE =
    		JS_PIXEL_RATIO +
    		"var scrollWidth = document.documentElement.scrollWidth; " +
            "var bodyScrollWidth = document.body.scrollWidth; " +
            "var totalWidth = Math.max(scrollWidth, bodyScrollWidth); " +
            "var clientHeight = document.documentElement.clientHeight; " +
            "var bodyClientHeight = document.body.clientHeight; " +
            "var scrollHeight = document.documentElement.scrollHeight; " +
            "var bodyScrollHeight = document.body.scrollHeight; " +
            "var maxDocElementHeight = Math.max(clientHeight, scrollHeight); " +
            "var maxBodyHeight = Math.max(bodyClientHeight, bodyScrollHeight); " +
            "var totalHeight = Math.max(maxDocElementHeight, maxBodyHeight); " +
            "return [totalWidth * pixelRatio, totalHeight * pixelRatio];";
    
    /*
    Get the first parent element which is scrollable
    
    TESTED WITH
	- no scrollable element in parents
	- scrollable element in parents
     */
    // according to https://www.w3schools.com/jsref/prop_element_scrolltop.asp
    // root scrollable (overflow) element is 'document.body' for safari and 'document.documentElement' for other browsers
    private static final String JS_SCROLL_PARENT = "function getScrollParent(element, includeHidden, browserName) {" +
			"    var rootElement = browserName === \"safari\" ? document.body: document.documentElement;" +
			"    var style = getComputedStyle(element);" +
			"    var excludeStaticParent = style.position === \"absolute\";" +
			"    var overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/;" +
			"" +
			"    if (style.position === \"fixed\") return null;" +
			"    for (var parent = element; (parent = getParentNode(parent));) {" +
			"        style = getComputedStyle(parent);" +
			"        if (excludeStaticParent && style.position === \"static\") {" +
			"            continue;" +
			"        }" +
			"        if (overflowRegex.test(style.overflow + style.overflowY + style.overflowX)) return parent;" +
			"    }" +
			"" +
			"    return null;" +
			"}" +
			"" +
			"function getParentNode(node) {" +
			// le parent est le document
			"    if (node.parentNode.nodeType === Node.DOCUMENT_NODE) {" +
			"        return null;" +
			"    " +
			// cas standard
			"    } else if (node.parentNode.nodeType === Node.ELEMENT_NODE) {" +
			"        return node.parentNode;" +
			"        " +
			// cas d'un shadow DOM / documentFragment
			"    } else if (node.parentNode.nodeType === Node.DOCUMENT_FRAGMENT_NODE) {" +
			"        return node.parentNode.host;" +
			"    }" +
			"}" +
    		"return getScrollParent(arguments[0], false, arguments[1]);";
    

    /*
	an other methode for finding scolling parent, that computes full xpath and then search a matching parent
    this method may be very slow for big DOM
	@param element			the element for which we want to get a scrollable parent
	@param includeHidden	whether to include hidden scrollable parents in search
	@param browserName		name of the browser

	TESTED WITH
	- element does not exist
	- no scrollable element in parents
	- scrollable element in parents
	*/
    private static final String JS_SCROLL_PARENT2 = "getScrollableParent = function(element, includeHidden, browserName) {"
    
    		/**
    			function for getting the full XPath of an element, or null if element cannot be found
    			Based on example found here: https://stackoverflow.com/questions/2661818/javascript-get-xpath-of-a-node/43688599#43688599
    			@param root		root element to search in
    			@param element	the element for which we want to get xpath
    			@param xpath	the base xpath (should be empty on first call, used for recursion)
    			*/
    		
    		+ "path = function(root, element, xpath) {"
    		+ "    _xPathIndex = function (node) {"
    		// Returns -1 in case of error, 0 if no siblings matching the same expression,"
    		//  otherwise."
    		+ "         function areNodesSimilar(left, right) {"
    		+ "             if (left === right) {"
    		+ "                 return true;"
    		+ "             }"
    		+ ""
    		+ "             if (left.nodeType === Node.ELEMENT_NODE && right.nodeType === Node.ELEMENT_NODE) {"
    		+ "                 return left.localName === right.localName;"
    		+ "             }"
    		+ ""
    		+ "             if (left.nodeType === right.nodeType) {"
    		+ "                 return true;"
    		+ "             }"
    		+ ""
    		// XPath treats CDATA as text nodes.
    		+ "             const leftType = left.nodeType === Node.CDATA_SECTION_NODE ? Node.TEXT_NODE : left.nodeType;"
    		+ "             const rightType = right.nodeType === Node.CDATA_SECTION_NODE ? Node.TEXT_NODE : right.nodeType;"
    		+ "             return leftType === rightType;"
    		+ "         }"
    		+ ""
    		+ "         const siblings = node.parentNode ? node.parentNode.children : null;"
    		// Root node - no siblings.
    		+ "         if (!siblings) {"
    		+ "             return 0;"
    		+ "         } "
    		+ "         let hasSameNamedElements;"
    		+ "         for (let i = 0; i < siblings.length; ++i) {"
    		+ "             if (areNodesSimilar(node, siblings[i]) && siblings[i] !== node) {"
    		+ "                 hasSameNamedElements = true;"
    		+ "                 break;"
    		+ "             }"
    		+ "         }"
    		+ "         if (!hasSameNamedElements) {"
    		+ "             return 0;"
    		+ "         }"
    		// XPath indices start with 1.
    		+ "         let ownIndex = 1;  "
    		+ "         for (let i = 0; i < siblings.length; ++i) {"
    		+ "             if (areNodesSimilar(node, siblings[i])) {"
    		+ "                 if (siblings[i] === node) {"
    		+ "                      return ownIndex;"
    		+ "                 }"
    		+ "                 ++ownIndex;"
    		+ "                   }"
    		+ "         }"
    		// An error occurred: |node| not found in parent's children.
    		+ "         return -1;"
    		+ "     };"
    		+ "     "
    		+ "     var elements = Array.from(root.children);"
    		+ "     "
    		// handle shadowRoot
    		+ "     if (root.shadowRoot != null) {"
    		+ "         elements = Array.from(root.shadowRoot.children).concat(elements);"
    		+ "     } "
    		// handle slots which carry some element inside even if children or shadowRoot is null
    		+ "     if (root.localName === 'slot' && root.assignedNodes() != null) {"
    		+ "         elements = Array.from(root.assignedElements()).concat(elements);"
    		+ "     } "
    		+ "     "
    		+ "     for (var i=0; i 0) {"
    		+ "             p += '[' + ownIndex + ']';"
    		+ "         }"
    		+ "         "
    		+ "         if (element === elements[i]) {"
//    		+ "             console.log(\"trouve\");"
    		+ "             return p;"
    		+ "             break;"
    		+ "         }"
    		+ ""
    		+ "         var ret = path(elements[i], element, p);"
    		+ "         if (ret != null) {"
    		+ "             return ret;"
    		+ "         }"
    		+ "     }"
    		+ ""
    		+ "     return null;"
    		+ "   "
    		+ " };"
    		+ " "
    		+ " "
    		+ " let rootElement = browserName === \"safari\" ? document.body: document.documentElement;"
    		+ " if (element == null) {"
    		+ "     return rootElement;"
    		+ " }"
    		+ " "
    		// get full xpath of the element
    		+ " let elementFullXpath = path(document, element, '');"
    		+ " let style = getComputedStyle(element);"
    		+ " let excludeStaticParent = style.position === \"absolute\";"
    		+ " let overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/;"
    		+ " if (style.position === \"fixed\") return rootElement;"
    		+ " "
    		// parse xpath from end to begin to find a scrollable element
    		+ " if (elementFullXpath != null) {"
    		+ "     while ((elementFullXpath.match(new RegExp(\"/\", \"g\")) || []).length > 1) {"
    		+ "         elementFullXpath = elementFullXpath.substring(0, elementFullXpath.lastIndexOf('/'));"
    		+ "         let elementToEvaluate = document.evaluate(elementFullXpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;"
    		+ ""
    		+ "         if (elementToEvaluate === null) {"
    		+ "             return rootElement;"
    		+ "         } else {"
    		+ "             style = getComputedStyle(elementToEvaluate);"
    		+ "             if (excludeStaticParent && style.position === \"static\") {"
    		+ "                 continue;"
    		+ "             } "
    		+ ""
    		+ "             if (overflowRegex.test(style.overflow + style.overflowY + style.overflowX)) return elementToEvaluate;"
    		+ "         }"
    		+ "     }"
    		+ " }"
    		+ " return rootElement;"
    		+ "};" +
    		"return getScrollableParent(arguments[0], false, arguments[1]);";
    
    
    private static final String JS_GET_ELEMENT_STYLE = "function getStyle(dom) {" + 
    		"  var style;" + 
    		"  var returns = {};" + 
    		"  /* FireFox and Chrome way */" + 
    		"  if (window.getComputedStyle) {" + 
    		"    style = window.getComputedStyle(dom, null);" + 
    		"    for (var i = 0, l = style.length; i < l; i++) {" + 
    		"      var prop = style[i];" + 
    		"      var val = style.getPropertyValue(prop);" + 
    		"      returns[prop] = val;" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"" + 
    		"  /* IE and Opera way */" + 
    		"  if (dom.currentStyle) {" + 
    		"    style = dom.currentStyle;" + 
    		"    for (var prop in style) {" + 
    		"      returns[prop] = style[prop];" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"  /* Style from style attribute */" + 
    		"  if ((style = dom.style)) {" + 
    		"    for (var prop in style) {" + 
    		"      if (typeof style[prop] != \"function\") {" + 
    		"        returns[prop] = style[prop];" + 
    		"      }" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"  return returns;" + 
    		"}";

    private static final String JS_IS_ELEMENT_VISIBLE = "function isVisible(elem) {" + 
    		"  var style = getStyle(elem);" + 
    		"" + 
    		"  if (style[\"display\"] === \"none\") return false;" + 
    		"  if (style[\"visibility\"] !== \"visible\") return false;" + 
    		"  if (style[\"opacity\"] === 0) return false;" + 
    		"" + 
    		"  if (" + 
    		"    elem.offsetWidth +" + 
    		"      elem.offsetHeight +" + 
    		"      elem.getBoundingClientRect().height +" + 
    		"      elem.getBoundingClientRect().width ===" + 
    		"    0" + 
    		"  )" + 
    		"    return false;" + 
    		"" + 
    		"  var elementPoints = {" + 
    		"    center: {" + 
    		"      x: elem.getBoundingClientRect().left + elem.offsetWidth / 2," + 
    		"      y: elem.getBoundingClientRect().top + elem.offsetHeight / 2" + 
    		"    }," + 
    		"    topLeft: {" + 
    		"      x: elem.getBoundingClientRect().left," + 
    		"      y: elem.getBoundingClientRect().top" + 
    		"    }," + 
    		"    topRight: {" + 
    		"      x: elem.getBoundingClientRect().right," + 
    		"      y: elem.getBoundingClientRect().top" + 
    		"    }," + 
    		"    bottomLeft: {" + 
    		"      x: elem.getBoundingClientRect().left," + 
    		"      y: elem.getBoundingClientRect().bottom" + 
    		"    }," + 
    		"    bottomRight: {" + 
    		"      x: elem.getBoundingClientRect().right," + 
    		"      y: elem.getBoundingClientRect().bottom" + 
    		"    }" + 
    		"  };" + 
    		"" + 
    		"  var docWidth = document.documentElement.clientWidth || window.innerWidth;" + 
    		"  var docHeight = document.documentElement.clientHeight || window.innerHeight;" + 
    		"" + 
    		"  if (elementPoints.topLeft.x > docWidth) return false;" + 
    		"  if (elementPoints.topLeft.y > docHeight) return false;" + 
    		"  if (elementPoints.bottomRight.x < 0) return false;" + 
    		"  if (elementPoints.bottomRight.y < 0) return false;" + 
    		"" + 
    		"  for (var index in elementPoints) {" + 
    		"    var point = elementPoints[index];" + 
    		"    var pointContainer = document.elementFromPoint(point.x, point.y);" + 
    		"    if (pointContainer !== null) {" + 
    		"      do {" + 
    		"        if (pointContainer === elem) return true;" + 
    		"      } while ((pointContainer = pointContainer.parentNode));" + 
    		"    }" + 
    		"  }" + 
    		"  return false;" + 
    		"}";
    
    // return true if a modal is present: a 'visible' div whose size is the same as the viewport
    private static final String JS_GET_MODAL = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +
    		"" +
    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +
    		"" +
    		JS_GET_ELEMENT_STYLE +
    		"" + 
    		JS_IS_ELEMENT_VISIBLE + 

    		"var modals;" + 
    		"var nodes = document.querySelectorAll(\"div\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  modals = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    modals.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 

    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +

    		"if (modals && modals.length > 0) {" + 
    		// only keep modals which are wide enough and visible
    		"  modals = modals.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth >= viewportWidth && e.clientHeight >= viewportHeight;" + 
    		"  });" + 
    		"  if (modals.length > 0) {" + 
    		"    return true;" + 
    		"  }" + 
    		"}" +

    		"return false;";
    
    // returns the top pixel from which fixed positioned headers are not present
    private static final String JS_GET_TOP_HEADER = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +

    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +

    		JS_GET_ELEMENT_STYLE +

    		JS_IS_ELEMENT_VISIBLE + 

    		JS_PIXEL_RATIO +
    		"var headers;" + 
    		"var nodes = document.querySelectorAll(\"div,header,ul\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  headers = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    headers.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 

    		"var topPixel = 0;" + 
    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +

    		"if (headers && headers.length > 0) {" + 
    		// only keep headers which are wide enough, not too high and visible
    		"  headers = headers.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth > viewportWidth / 2 && e.clientHeight < viewportHeight / 2;" + 
    		"  });" + 
    		"  headers = headers.sort(function(a, b) {" + 
    		"    if (a == null || a.offsetTop == null) return 1;" + 
    		"    if (b == null || b.offsetTop == null) return -1;" + 
    		"    return a.offsetTop - b.offsetTop;" + 
    		"  });" + 

    		"  for (var i = 0; i < headers.length; i++) {" + 
    		"    var header = headers[i];" + 
    		"    if (header.offsetTop <= topPixel + 10) {" + 
    		"      topPixel = header.offsetTop + header.scrollHeight;" + 
    		"    } else {" + 
    		"      break;" + 
    		"    }" + 
    		"  }" + 
    		"}" + 

    		"return Math.round(topPixel * pixelRatio);";
    
    private static final String JS_GET_BOTTOM_FOOTER = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +

    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +

    		JS_GET_ELEMENT_STYLE +

    		JS_IS_ELEMENT_VISIBLE +

    		JS_PIXEL_RATIO +

    		"var footers;" + 
    		"var nodes = document.querySelectorAll(\"div,footer,ul\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  footers = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    footers.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 

    		"var bottomPixel = document.documentElement.clientHeight;" + // count from bottom
    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +

    		"if (footers && footers.length > 0) {" + 
    		// only keep footers which are wide enough, not too high and visible
    		"  footers = footers.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth > viewportWidth / 2 && e.clientHeight < viewportHeight / 2;" + 
    		"  });" + 
    		"  footers = footers.sort(function(a, b) {" + 
    		"    if (a == null || a.offsetBottom == null) return -1;" + 
    		"    if (b == null || b.offsetBottom == null) return 1;" + 
    		"    return b.offsetBottom - a.offsetBottom;" + 
    		"  });" + 

    		"  for (var i = 0; i < footers.length; i++) {" + 
    		"    var footer = footers[i];" + 
    		"    if (footer.offsetTop + footer.scrollHeight >= bottomPixel - 10) {" + 
    		"      bottomPixel = footer.offsetTop;" + 
    		"    } else {" + 
    		"      break;" + 
    		"    }" + 
    		"  }" + 
    		"}" + 

    		"return Math.round(document.documentElement.clientHeight * pixelRatio - bottomPixel * pixelRatio);";
    		
    
    public static final String NON_JS_UPLOAD_FILE_THROUGH_POPUP = 
    		"var action = 'upload_file_through_popup';";
    public static final String NON_JS_CAPTURE_DESKTOP = 
    		"var action = 'capture_desktop_snapshot_to_base64_string';return '';";
    
    public CustomEventFiringWebDriver(final WebDriver driver) {
    	this(driver, null, null, TestType.WEB, DriverMode.LOCAL, null);
    }

    public CustomEventFiringWebDriver(final WebDriver driver, List driverPids, BrowserInfo browserInfo, TestType testType, DriverMode localDriver, SeleniumGridConnector gridConnector) {
    	this(driver, driverPids, browserInfo, testType, localDriver, gridConnector, null, new ArrayList<>());
    }
	public CustomEventFiringWebDriver(final WebDriver driver, List driverPids, BrowserInfo browserInfo, TestType testType, DriverMode localDriver, SeleniumGridConnector gridConnector, Integer attachExistingDriverPort, List wdListeners) {

		this.driverPids = driverPids == null ? new ArrayList<>(): driverPids;
		this.browserInfo = browserInfo;
		this.testType = testType;
		this.driverMode = localDriver;
//		this.mobProxy = mobProxy;
		this.gridConnector = gridConnector;
		this.attachExistingDriverPort = attachExistingDriverPort;
		this.originalDriver = driver; // store the original driver in case decorated one cannot be used (getSessionId)
		this.driver = driver;

		// Augmenter is only supported for web test because augmenting driver namely tries to create a CDP connection with a browser
		// do not augment mocked drivers (used in unit tests), as the mocked driver is masked by augmentation. Spyed driver are OK
		// as they behave as real drivers
		if (isWebTest() && !(Mockito.mockingDetails(driver).isMock() && !Mockito.mockingDetails(driver).isSpy())) {
			try {
				this.driver = new Augmenter().augment(this.driver);
			} catch (Exception e) {
				throw new RetryableDriverException("Error augmenting driver", e);
			}
		}

		this.driver = new EventFiringDecorator(new DriverExceptionListener(this)).decorate(this.driver);

		for (WebDriverListener wdListener: wdListeners) {
			this.driver = new EventFiringDecorator(wdListener).decorate(this.driver);
		}

		// NEOLOAD //
		if (driver instanceof NLWebDriver) {
			neoloadDriver = (NLWebDriver)driver;
		} else {
			neoloadDriver = null;
		}
    }

    public void setFileDetector(final FileDetector detector) {
        if (detector == null) {
            throw new WebDriverException("file detector is null");
        }

        fileDetector = detector;
    }
    
    /**
     * Method for updating window handles when an operation may create a new window (a click action)
     * This is called for Composite actions, native actions (from DriverListener) and JS actions
     */
    public void updateWindowsHandles() {
    	if (isWebTest()) {
    		currentHandles = getWindowHandles();
    			
    	} else {
    		currentHandles = new TreeSet<>();
    	}
    }
    
    public void setCurrentHandles(Set currentHandles) {
		this.currentHandles = currentHandles;
	}

	@Override
    public Set getWindowHandles() {
    	
    	if (!isWebTest()) {
    		return new TreeSet<>();
    	}
    	
    	// issue #169: workaround for ios / IE tests where getWindowHandles sometimes fails with: class org.openqa.selenium.WebDriverException: Returned value cannot be converted to List: true
    	for (int i = 0; i < 10; i++) {
			try  {
				return driver.getWindowHandles();
			} catch (UnhandledAlertException e) {
	    		logger.info("getWindowHandles: Handling alert");
	    		handleAlert();
	    		return driver.getWindowHandles();
			} catch (WebSessionEndedException e) {
				logger.warn("session already terminated");
				return new TreeSet<>();
			} catch (Exception e) {
				logger.info("error getting window handles: " + e.getMessage());
				WaitHelper.waitForSeconds(2);
			}
		}
    	return driver.getWindowHandles();
    	
    }
    
    private void handleAlert() {
    	try {
	    	Alert alert = driver.switchTo().alert();
			alert.dismiss();
    	} catch (Exception e) {
    		// do nothing in case of error
    	}
    }
    
    @Override
    public String getWindowHandle() {
    	
    	if (!isWebTest()) {
    		return "";
    	}
    	
    	try {
    		return driver.getWindowHandle();
    	} catch (UnhandledAlertException e) {
    		logger.info("getWindowHandle: Handling alert");
    		handleAlert();
    		return driver.getWindowHandle();
    	}
    }
    
    @Override
    public void close() {
    	try {
			driver.close();
		} catch (NullPointerException e) {
			// NPE raised with IE when closing session
			// nothing to do
    	} catch (UnhandledAlertException e) {
    		logger.info("close: Handling alert");
    		handleAlert();
    		driver.close();
    	}
    }
    
    @Override
    public String getCurrentUrl() {
    	try {
	    	return driver.getCurrentUrl();
	    } catch (UnhandledAlertException e) {
    		logger.info("getCurrentUrl: Handling alert");
			handleAlert();
			return driver.getCurrentUrl();
		} catch (NoSuchWindowException e) {
			return "";
		}
    }
    
    @Override
    public String getTitle() {
    	try {
    		return driver.getTitle();
    	} catch (UnhandledAlertException e) {
    		logger.info("getTitle: Handling alert");
			handleAlert();
			return driver.getTitle();
    	} catch (NoSuchWindowException e) {
			return "";
    	}
    }
    
    

    public FileDetector getFileDetector() {
        return fileDetector;
    }

    /**
     * Returns the decorated and augmented Selenium driver.
     * This means that if you use a Chrome browser, {@code getWebDriver() instanceof ChromeDriver} will return false.
     * You will have to use {@code getOriginalDriver() instanceof ChromeDriver}
     * However, {@code getWebDriver() instanceof HasCdp} will return true because driver is augmented at creation with all implemented interfaces
     * @return
     */
    public WebDriver getWebDriver() {
        return driver;
    }

	/**
	 * Method returning the real created driver, before it has been decorated and augmented
	 * This allows to write {@code getOriginalDriver() instanceof ChromeDriver} which returns true for a chrome browser
	 * @return
	 */
	public WebDriver getOriginalDriver() {
		return originalDriver;
	}
    
    public void setDriverExited() {
    	driverExited = true;
    }
    
    /**
     * Returns true if all browser windows are closed and so, no actions can be performed
	 * In case of mobile App, check the app is still there
     * @return
     */
    public boolean isBrowserOrAppClosed() {
    	try {
    		getSessionId();
    		getCapabilities();

			// check only for web browser tests, not browser inside mobile app, as in the later case, embedded browser may not
			// have any active window, if app has switched to a NATIVE page
			if (testType.family().equals(TestType.WEB)) {
				return driver.getWindowHandles().isEmpty();
			} else {
				// in case of mobile app, we assume the app is not closed. Closing the app should only be done at test end
				// closing the session. If we are here, the session is still alive
				return false;
			}
    	} catch (NoSuchSessionException | UnsupportedCommandException | NullPointerException /* for HtmlUnit */ e) {
    		return true;
    	}
    	
    }
    
    public String getSessionId() {
    	try {
    		return ((RemoteWebDriver)originalDriver).getSessionId().toString();
    	} catch (ClassCastException e) {
    		return UUID.randomUUID().toString();
    	} catch (NullPointerException e) {
    		return null;
    	}
    }
    
    /**
     * Handle WebDriver exception when this method is not implemented
     */
    @Override
    public String getPageSource() {
    	try {
    		return driver.getPageSource();
    	} catch (UnhandledAlertException e) {
    		logger.info("getPageSource: Handling alert");
			handleAlert();
			return driver.getPageSource();
    	} catch (WebDriverException e) {
    		logger.info("page source not get: " + e.getMessage());
    		return null;
    	}
    }
    
    public Set getCurrentHandles() {
    	if (currentHandles == null) {
    		updateWindowsHandles();
    	}
		return currentHandles;
	}
    
    /**
     * Return the device aspect ratio
     * @return
     */
    public double getDeviceAspectRatio() {
    	if (isWebTest()) {
			try {
				Number devicePixelRatio = (Number) ((JavascriptExecutor) driver).executeScript(JS_PIXEL_RATIO + " return pixelRatio;", true);
				return devicePixelRatio.doubleValue();
			} catch (Exception e) {
				return 1.0;
			}
    	} else {
    		return 1.0;
    	}
    }
    
	public int getViewPortWidthWithoutScrollbar() {
    	return getViewPortWidthWithoutScrollbar(true);
    }
    /**
	 * Get viewport width, do not take scrollbar into account
	 * @param usePixelAspectRatio	if true, take zoom level into account to compute size
	 * @return
	 */
	public int getViewPortWidthWithoutScrollbar(boolean usePixelAspectRatio) {
    	if (isWebTest()) {
    		try {
    			
	    		Number width = (Number)executeScript(JS_GET_VIEWPORT_SIZE_WIDTH, usePixelAspectRatio);
	    		
	    		// issue #238: check we get a non max size
	    		if (width.intValue() == MAX_DIMENSION) {
	    			driver.switchTo().defaultContent();
	    			width = (Number)executeScript(JS_GET_VIEWPORT_SIZE_WIDTH, usePixelAspectRatio);
	    		}
	    		return width.intValue();
	    		
    		} catch (Exception e) {
    			return driver.manage().window().getSize().getWidth();
    		}
    	} else {
    		return driver.manage().window().getSize().getWidth();
    	}
    }
	
	/**
	 * Get viewport size, do not take scrollbar into account
	 * @return
	 */
	public int getViewPortHeighthWithoutScrollbar() {
		return getViewPortHeighthWithoutScrollbar(true);
	}
	
	/**
	 * Get viewport size, do not take scrollbar into account
	 * @param usePixelAspectRatio	if true, take zoom level into account to compute size
	 * @return
	 */
	public int getViewPortHeighthWithoutScrollbar(boolean usePixelAspectRatio) {
		if (isWebTest()) {
			try {
				
				Number height = (Number)executeScript(JS_GET_VIEWPORT_SIZE_HEIGHT, usePixelAspectRatio);
				
				// issue #238: check we get a non max size
				if (height.intValue() == MAX_DIMENSION) {
					driver.switchTo().defaultContent();
					height = (Number)executeScript(JS_GET_VIEWPORT_SIZE_HEIGHT, usePixelAspectRatio);
				}
				return height.intValue();
				
			} catch (Exception e) {
				return driver.manage().window().getSize().getHeight();
			}
		} else {
			return driver.manage().window().getSize().getHeight();
		}
	}
    
    /**
     * get dimensions of the visible part of the page
     * TODO: handle mobile app case
     * @return
     */
    @SuppressWarnings("unchecked")
	public Dimension getViewPortDimensionWithoutScrollbar() {
    	return getViewPortDimensionWithoutScrollbar(true);
    }
    
    /**
     * get dimensions of the visible part of the page
     * 
     * @return
     */
	public Dimension getViewPortDimensionWithoutScrollbar(boolean usePixelAspectRatio) {
    	Dimension foundDimension = new Dimension(getViewPortWidthWithoutScrollbar(usePixelAspectRatio), getViewPortHeighthWithoutScrollbar(usePixelAspectRatio));
		
		// issue #233: prevent too big size at it may be used to process images and Buffer
		if (foundDimension.width * foundDimension.height * 8L > Integer.MAX_VALUE) {
			return new Dimension(2000, 10000);
		} else {
			return foundDimension;
		}
    }
    
	
	public Dimension getContentDimension() {
		return getContentDimension(true);
	}
    /**
     * Get the whole webpage dimension
     * TODO: handle mobile app case
     * @return
     */

	@SuppressWarnings("unchecked")
    public Dimension getContentDimension(boolean usePixelAspectRatio) {
    	if (isWebTest()) {
    		try {
				List dims = (List)executeScript(JS_GET_CONTENT_ENTIRE_SIZE, usePixelAspectRatio);
		    	
		    	// issue #238: check we get a non zero size
		    	if (dims.get(0).intValue() == 0 || dims.get(1).intValue() == 0) {
					driver.switchTo().defaultContent();
		    		dims = (List)executeScript(JS_GET_CONTENT_ENTIRE_SIZE, usePixelAspectRatio);
		    	}
		    	
		    	return new Dimension(dims.get(0).intValue(), dims.get(1).intValue());
		    	
    		} catch (Exception e) {
    			return driver.manage().window().getSize();
    		}
    	} else {
    		return driver.manage().window().getSize();
    	}
    }
	
	/**
	 * TODO: handle mobile app case
	 */
	public void scrollTop() {
		if (isWebTest()) {
			((JavascriptExecutor) driver).executeScript("window.top.scroll(0, 0)");
		} 
	}
	
	public void scrollTo(int x, int y) {
		if (isWebTest()) {
			((JavascriptExecutor) driver).executeScript(String.format("window.top.scroll(%d, %d)", x, y));
			
			// wait for scrolling end
			Point previousScrollPosition = getScrollPosition();
			int i = 0;
			boolean fixed = false;
			do {
				Point scrollPosition = getScrollPosition();
				if (scrollPosition.x == x && scrollPosition.y == y) {
					break;
				} else if (scrollPosition.x == previousScrollPosition.x && scrollPosition.y == previousScrollPosition.y) {
					if (!fixed) {
						fixed = true;
					} else {
						break;
					}
				}
				previousScrollPosition = scrollPosition;
				WaitHelper.waitForMilliSeconds(100);
				i++;
			} while (i < 10);
		} 
	}
	
	/**
	 * Return the size of top header when it is 'fixed' positionned
	 * @return
	 */
	public Long getTopFixedHeaderSize() {
		return getTopFixedHeaderSize(true);
	}
	public Long getTopFixedHeaderSize(boolean usePixelAspectRatio) {
		if (isWebTest()) {
			return (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_TOP_HEADER, usePixelAspectRatio);
		} else { 
			return 0L;
		}
	}
	
	
	/**
	 * Return the size of bottom footer when it is 'fixed' positionned
	 * @return
	 */
	public Long getBottomFixedFooterSize() {
		return getBottomFixedFooterSize(true);
	}
	public Long getBottomFixedFooterSize(boolean usePixelAspectRatio) {
		if (isWebTest()) {
			return (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_BOTTOM_FOOTER, usePixelAspectRatio);
		} else { 
			return 0L;
		}
	}
	
	/**
	 * Check if a modal is displayed
	 * @return
	 */
	public boolean isModalDisplayed() {
		if (isWebTest()) {
			return (Boolean) ((JavascriptExecutor) driver).executeScript(JS_GET_MODAL);
		} else { 
			return false;
		}
	}
	
	/**
	 * scroll to the given element
	 * we scroll 200 px to the left of the element so that we see all of it
	 * @param element
	 */
	public void scrollToElement(WebElement element, int yOffset) {
		if (isWebTest()) {
			try {
				WebElement parentScrollableElement = (WebElement) ((JavascriptExecutor) driver).executeScript(JS_SCROLL_PARENT, element, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
				Long topHeaderSize = (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_TOP_HEADER);

				// try a second method (the first one is quicker but does not work when element is inside a document fragment, slot or shadow DOM
//				if ((parentScrollableElement == null || "html".equalsIgnoreCase(parentScrollableElement.getTagName())) && !(driver instanceof InternetExplorerDriver)) {
//					parentScrollableElement = (WebElement) ((JavascriptExecutor) driver).executeScript(JS_SCROLL_PARENT2, element, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
//				}
				
				if (parentScrollableElement != null) {
					scrollParent(element, yOffset, parentScrollableElement, topHeaderSize);
				} else {
					// go to default behavior
					throw new JavascriptException("No parent found");
				}
				
			} catch (Exception e) {
				// fall back to legacy behavior
				if (yOffset == Integer.MAX_VALUE) {
					yOffset = -200;
				}
				try {
					scrollWindowToElement(element, yOffset);
				} catch (Exception e1) {
					logger.info(String.format("Cannot scroll to element %s: %s", element.toString(), e1.getMessage()));
				}
			}
		}
	}

	/**
	 * Scroll parent of element, in case we have a scroll inside an other scroll
	 * @param element
	 * @param yOffset
	 * @param parentScrollableElement
	 * @param topHeaderSize
	 * @return
	 */
	private int scrollParent(WebElement element, int yOffset, WebElement parentScrollableElement, Long topHeaderSize) {
		String parentTagName = parentScrollableElement.getTagName();
		
		// When the scrollable element is the document itself (html or body), use the legacy behavior scrolling the window itself
		if ("html".equalsIgnoreCase(parentTagName) || "body".equalsIgnoreCase(parentTagName)) {
			if (yOffset == Integer.MAX_VALUE) {// equivalent to HtmlElement.OPTIMAL_SCROLLING but, for grid, we do not want dependency between the 2 classes
				yOffset = (int) Math.min(-200, -topHeaderSize);
			}
			
			scrollWindowToElement(element, yOffset);
		
		// When scrollable element is a "div" or anything else, use scrollIntoView because it's the easiest way to make the element visible
		} else {
			((JavascriptExecutor) driver).executeScript("arguments[0].scrollIntoView(true);", element);
			
			// check if scrollbar of parent is at bottom. In this case, going upside (yOffset < 0), could hide searched element.
			// scrollIntoView is configured to position scrollbar to top position of the searched element.
			// in case offset is > 0, this has no impact as scrollBottom will always be < 0
			WaitHelper.waitForMilliSeconds(500); // wait a bit so that scrolling is complete
			
			// if top header is present, scroll up so that our element is not hidden behind it (scrollIntoView scrolls so that element is at the top of the view even if header masks it)
			if (topHeaderSize > 0) {// equivalent to HtmlElement.OPTIMAL_SCROLLING but, for grid, we do not want dependency between the 2 classes
				Integer scrollOffset = (int) (yOffset == Integer.MAX_VALUE ? topHeaderSize: topHeaderSize - yOffset);
				((JavascriptExecutor) driver).executeScript(
						"if((arguments[0].scrollHeight - arguments[0].scrollTop - arguments[0].clientHeight) > 0) {" +
						"   var rootElement = arguments[1] === \"safari\" ? document.body: document.documentElement;" +
						"   rootElement.scrollTop -= " + scrollOffset + ";" +
						"}", parentScrollableElement, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
			}
			
			WaitHelper.waitForMilliSeconds(500); // wait a bit so that scrolling is complete
		}
		return yOffset;
	}
	
	private void scrollWindowToElement(WebElement element, int yOffset) {
		((JavascriptExecutor) driver).executeScript("window.top.scroll(" + Math.max(element.getLocation().x - 200, 0) + "," + Math.max(element.getLocation().y + yOffset, 0) + ")");
	}
	
	/**
	 * TODO: handle mobile app case
	 */
	@SuppressWarnings("unchecked")
	public Point getScrollPosition() {
		if (isWebTest()) {
			try {
				List dims = (List)((JavascriptExecutor) driver).executeScript(JS_GET_CURRENT_SCROLL_POSITION);
				return new Point(dims.get(0).intValue(), dims.get(1).intValue());
			} catch (Exception e) {
    			return new Point(0, 0);
    		}
		} else {
			throw new WebDriverException("scroll position can only be get for web");
		}
	}
	
	/**
	 * Returns the rectangle of all screens on the system
	 * @return
	 */
	public static Rectangle getScreensRectangle() {
		GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
		
		Rectangle screenRect = new Rectangle(0, 0, 0, 0);
		for (GraphicsDevice gd : ge.getScreenDevices()) {
		    screenRect = screenRect.union(gd.getDefaultConfiguration().getBounds());
		}
		return screenRect;
	}
	private static Rectangle getMainScreenRectangle() {
		GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
		
		GraphicsDevice gd = ge.getDefaultScreenDevice();
		return gd.getDefaultConfiguration().getBounds();

	}
	
	/**
	 * Take screenshot of the desktop and put it in a file
	 * Do not expose this method because we need to check that we have a graphical environment. 
	 */
	private static BufferedImage captureDesktopToBuffer() {
		
		
		try {
			Rectangle screenRect = getScreensRectangle();
			return new Robot().createScreenCapture(screenRect);
		} catch (HeadlessException | AWTError | AWTException e) {
			throw new ScenarioException("Cannot capture image", e);
		}
	}
	
	private static BufferedImage captureMainDesktopScreenToBuffer() {
		
		
		try {
			Rectangle screenRect = getMainScreenRectangle();
			return new Robot().createScreenCapture(screenRect);
		} catch (HeadlessException | AWTError | AWTException e) {
			throw new ScenarioException("Cannot capture image", e);
		}
	}

	/**
	 * After quitting driver, if it fails, some pids may remain. Kill them
	 */
	@Override
	public void quit() {
		
		// get list of pids we could have to kill. Sometimes, Chrome does not close all its processes
		// so we have to know which processes to kill when driver is still active
		List pidsToKill = new ArrayList<>();
		if (browserInfo != null && driverMode == DriverMode.LOCAL) {
			pidsToKill.addAll(browserInfo.getAllBrowserSubprocessPids(driverPids));
			
			// issue #401: also get the browser process in case of attached browser (chrome only)
			if (attachExistingDriverPort != null && attachExistingDriverPort > 1000) {
				Integer browserProcess = OSUtilityFactory.getInstance().getProcessIdByListeningPort(attachExistingDriverPort);
				if (browserProcess != null) {
					pidsToKill.add(browserProcess.longValue());
				}
			}
		}
		
		if (driverExited) {
			return;
		}
		
		Capabilities caps;
		try {
			caps = getCapabilities();
		} catch (WebDriverException | NullPointerException /* for HtmlUnitDriver */ e) {
			caps = new MutableCapabilities();
		}
		
		// close windows before quitting (this is the only way to close chrome attached browser when it's not started by selenium)
		try {
			List handles = new ArrayList<>(driver.getWindowHandles());
	        Collections.reverse(handles);

	        for (String handle: handles) {
				driver.switchTo().window(handle);
				driver.close();
			}
		} catch (Exception e) {
			// nothing to do
		}
		
		
		Long duration = 0L;
		try {
			duration = new Date().getTime() - (Long)internalCapabilities.getCapability(SeleniumRobotCapabilityType.START_TIME);
		} catch (Exception e) {
			// catch error
		}
		String gridHub = caps.getCapability(SeleniumRobotCapabilityType.GRID_HUB) != null ? caps.getCapability(SeleniumRobotCapabilityType.GRID_HUB).toString(): null;
		String sessionId = caps.getCapability(SeleniumRobotCapabilityType.SESSION_ID) != null ? caps.getCapability(SeleniumRobotCapabilityType.SESSION_ID).toString(): null;
		String driverSessionId = getSessionId();
		
		// store driver stats
		DriverUsage usage = new DriverUsage(gridHub, 
				(String) caps.getCapability(SeleniumRobotCapabilityType.GRID_NODE), 
				(Long) internalCapabilities.getCapability(SeleniumRobotCapabilityType.START_TIME), 
				duration, 
				sessionId, 
				caps.getBrowserName(), 
				(Long) internalCapabilities.getCapability(SeleniumRobotCapabilityType.STARTUP_DURATION), 
				(String) internalCapabilities.getCapability(SeleniumRobotCapabilityType.TEST_NAME));
		StatisticsStorage.addDriverUsage(usage);
		
		try {
			driver.quit();
		} catch (WebDriverException e) {
			
			logger.error("Error while quitting driver: " + e.getMessage());
			
			// in case stopping fails, kill the session on node directly
			// It happens on an InternetExplorer where an unexpected modal, associated to a mechanism that prevents IE to be stopped, driver.close() and driver.quit()
			// do not work as exepected
			sessionId = driverSessionId != null ? driverSessionId: sessionId;
			if (sessionId != null && gridConnector != null) {
				logger.info(String.format("Try to stop session %s calling node", sessionId));
				gridConnector.stopSession(sessionId);
			}
			caps = new MutableCapabilities();
		} finally {
			
			// wait for browser processes to stop
			WaitHelper.waitForSeconds(2);
			
			// only kill processes in local mode
			if (!pidsToKill.isEmpty()) {
		    	for (Long pid: pidsToKill) {
		    		OSUtilityFactory.getInstance().killProcess(pid.toString(), true);
		    	}
			}
			
		}
	}
	
	/**
	 * Use copy to clipboard and copy-paste keyboard shortcut to write something on upload window
	 */
	public static void uploadFileUsingClipboard(File tempFile) {

		// Copy to clipboard
		Toolkit.getDefaultToolkit().getSystemClipboard().setContents(new StringSelection(tempFile.getAbsolutePath()), null);
		Robot robot;
		try {
			robot = new Robot();
		
			WaitHelper.waitForSeconds(1);
	
//			// Press Enter
//			robot.keyPress(KeyEvent.VK_ENTER);
//	
//			// Release Enter
//			robot.keyRelease(KeyEvent.VK_ENTER);
	
			// Press CTRL+V
			robot.keyPress(KeyEvent.VK_CONTROL);
			robot.keyPress(KeyEvent.VK_V);
	
			// Release CTRL+V
			robot.keyRelease(KeyEvent.VK_CONTROL);
			robot.keyRelease(KeyEvent.VK_V);
			WaitHelper.waitForSeconds(1);
	
			// Press Enter
			robot.keyPress(KeyEvent.VK_ENTER);
			robot.keyRelease(KeyEvent.VK_ENTER);
		} catch (AWTException e) {
			throw new ScenarioException("could not initialize robot to upload file: " + e.getMessage());
		}
	}
	
	/**
	 * Upload file typing file path directly
	 * @param tempFile
	 */
	public static void uploadFileUsingKeyboardTyping(File tempFile) {
		try {
			Keyboard keyboard = new Keyboard();
			Robot robot = keyboard.getRobot();
			
			WaitHelper.waitForSeconds(1);
			
//			// Press Enter
//			robot.keyPress(KeyEvent.VK_ENTER);
//	
//			// Release Enter
//			robot.keyRelease(KeyEvent.VK_ENTER);
			
			keyboard.typeKeys(tempFile.getAbsolutePath());
			
			WaitHelper.waitForSeconds(1);
			
			// Press Enter
			robot.keyPress(KeyEvent.VK_ENTER);
			robot.keyRelease(KeyEvent.VK_ENTER);
			
		} catch (AWTException e) {
			throw new ScenarioException("could not initialize robot to upload file typing keys: " + e.getMessage());
		}
	}
	
	public static void uploadFile(String fileName, String base64Content, DriverMode driverMode, SeleniumGridConnector gridConnector) throws IOException {
		
		if (driverMode == DriverMode.LOCAL) {
			byte[] byteArray = base64Content.getBytes();
	        File tempFile = new File("tmp/" + fileName);
	        byte[] decodeBuffer = Base64.decodeBase64(byteArray);
	        FileUtils.writeByteArrayToFile(tempFile, decodeBuffer);
	
	        try {
	        	uploadFileUsingClipboard(tempFile);
	        } catch (IllegalStateException e) {
	        	uploadFileUsingKeyboardTyping(tempFile);
	        }
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.uploadFileToBrowser(fileName, base64Content);
		} else {
			throw new ScenarioException("driver supports uploadFile only in local and grid mode");
		}
	}
	
	/**
	 * move mouse taking screen placing into account. Sometimes, coordinates may be negative if first screen has an other screen on the left 
	 * @param robot
	 */
	private static void moveMouse(Robot robot, int x, int y) {
		Rectangle screenRectangle = getScreensRectangle();
		robot.mouseMove(x + screenRectangle.x, y + screenRectangle.y);
	}
	private static void moveMouseMainScreen(Robot robot, int x, int y) {
		Rectangle screenRectangle = getMainScreenRectangle();
		robot.mouseMove(x + screenRectangle.x, y + screenRectangle.y);
	}
	
	/**
	 * Left clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void leftClicOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		leftClicOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void leftClicOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				disableStepDisplay();
				
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("leftClicOnDesktopAt: problem using Robot: " + e.getMessage());
			} finally {
				enableStepDisplay();
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.leftClic(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports leftClicOnDesktopAt only in local and grid mode");
		}
	}
	
	/**
	 * Get mouse coordinates
	 */
	public static java.awt.Point getMouseCoordinates(DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				PointerInfo pointerInfo = MouseInfo.getPointerInfo();
				return pointerInfo.getLocation();
			} catch (HeadlessException e) {
				throw new ScenarioException("getMouseCoordinades: problem using Robot: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			return gridConnector.getMouseCoordinates();
		} else {
			throw new ScenarioException("driver supports getMouseCoordinades only in local and grid mode");
		}
	}
	
	/**
	 * Left clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void doubleClickOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		doubleClickOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void doubleClickOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				
				disableStepDisplay();
				
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
				robot.delay(10);
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("doubleClickOnDesktopAt: problem using Robot: " + e.getMessage());
			} finally {
				
				enableStepDisplay();
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.doubleClick(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports doubleClickOnDesktopAt only in local and grid mode");
		}
	}
	
	/**
	 * right clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void rightClicOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		rightClicOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void rightClicOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				
				disableStepDisplay();
				
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON3_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON3_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("rightClicOnDesktopAt: problem using Robot: " + e.getMessage());
			} finally {
				
				enableStepDisplay();
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.rightClic(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports sendKeysToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * write text to desktop.
	 * @param textToWrite	text to write
	 * @return
	 */
	public static void writeToDesktop(String textToWrite, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
	
			try {
				Keyboard keyboard = new Keyboard();
				keyboard.typeKeys(textToWrite);
			} catch (AWTException e) {
				throw new ScenarioException("writeToDesktop: could not initialize robot to type keys: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.writeText(textToWrite);
		} else {
			throw new ScenarioException("driver supports writeToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * send keys to desktop
	 * This is useful for typing special keys like ENTER
	 * @param keyCodes	List of KeyEvent
	 */
	public static void sendKeysToDesktop(List keyCodes, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
			try {
				Robot robot = new Robot();
				
				WaitHelper.waitForSeconds(1);
				
				for (Integer key: keyCodes) {
					robot.keyPress(key);
					robot.keyRelease(key);
				}
			} catch (AWTException e) {
				throw new ScenarioException("could not initialize robot to type keys: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.sendKeysWithKeyboard(keyCodes);
		} else {
			throw new ScenarioException("driver supports sendKeysToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * Returns a Base64 string of the desktop
	 * @param driverMode
	 * @param gridConnector
	 * @return
	 */
	public static String captureDesktopToBase64String(DriverMode driverMode, SeleniumGridConnector gridConnector) {
		return captureDesktopToBase64String(false, driverMode, gridConnector);
	}
	
	/**
	 * Returns a Base64 string of the desktop
	 * @param onlyMainScreen	if true, only capture the default (or main) screen
	 * @param driverMode
	 * @param gridConnector
	 * @return
	 */
	public static String captureDesktopToBase64String(boolean onlyMainScreen, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
			
			try {
				disableStepDisplay();
				
				BufferedImage bi;
				if (onlyMainScreen) {
					bi = captureMainDesktopScreenToBuffer();
				} else {
					bi = captureDesktopToBuffer();
				}
				
				ByteArrayOutputStream os = new ByteArrayOutputStream();
				OutputStream b64 = new Base64OutputStream(os);
				try {
					ImageIO.write(bi, "png", b64);
					return os.toString("UTF-8");
				} catch (IOException e) {
					return "";
				}
				
			} finally {
				enableStepDisplay();
			}

		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			return gridConnector.captureDesktopToBuffer();
		} else {
			throw new ScenarioException("driver supports captureDesktopToBase64String only in local and grid mode");
		}
	}
	
	/**
	 * Start video capture using VideoRecorder class
	 * @param driverMode
	 * @param gridConnector
	 * @param videoName		name of the video to record so that it's unique. Only used locally. In remote, grid sessionId is used
	 */
	public static VideoRecorder startVideoCapture(DriverMode driverMode, SeleniumGridConnector gridConnector, File videoFolder, String videoName) {
		if (driverMode == DriverMode.LOCAL) {
			try {
				VideoRecorder recorder = new VideoRecorder(videoFolder, videoName);
				recorder.start();
				videoRecorders.add(recorder);
				return recorder;
			} catch (HeadlessException e) {
				throw new ScenarioException("could not initialize video capture with headless robot: " + e.getMessage());
			}
			
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.startVideoCapture();
			return new VideoRecorder(videoFolder, videoName, false);
		} else {
			throw new ScenarioException("driver supports startVideoCapture only in local and grid mode");
		}
	}
	
	/**
	 * Stop video capture using VideoRecorder class
	 * @param driverMode
	 * @param gridConnector
	 * @throws IOException 
	 */
	public static File stopVideoCapture(DriverMode driverMode, SeleniumGridConnector gridConnector, VideoRecorder recorder) throws IOException {
		if (driverMode == DriverMode.LOCAL && recorder != null) {
			videoRecorders.remove(recorder);
			return recorder.stop();
		} else if (driverMode == DriverMode.GRID && gridConnector != null && recorder != null) {
			return gridConnector.stopVideoCapture(Paths.get(recorder.getFolderPath().getAbsolutePath(), recorder.getFileName()).toString());
			
		} else {
			throw new ScenarioException("driver supports stopVideoCapture only in local and grid mode");
		}
	}
	
	public static void displayStepOnScreen(String textToWrite, DriverMode driverMode, SeleniumGridConnector gridConnector, VideoRecorder recorder) {
		if (driverMode == DriverMode.LOCAL) {
			
			recorder.displayRunningStep(textToWrite);
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.displayRunningStep(textToWrite);
		} else {
			throw new ScenarioException("driver supports writeToDesktop only in local and grid mode");
		}
	}
	

    public List getDriverPids() {
		return driverPids;
	}

	public BrowserInfo getBrowserInfo() {
		return browserInfo;
	}

	@Override
    public Capabilities getCapabilities() {
		try {
			return ((HasCapabilities)driver).getCapabilities();
		} catch (ClassCastException e) {
			return new MutableCapabilities();
		}
    }

//	public BrowserMobProxy getMobProxy() {
//		return mobProxy;
//	}

	// NEOLOAD //
	public NLWebDriver getNeoloadDriver() {
		return neoloadDriver;
	}

	public MutableCapabilities getInternalCapabilities() {
		return internalCapabilities;
	}

	@Override
	public void get(String url) {
		driver.get(url);
	}

	@Override
	public List findElements(By by) {
		return driver.findElements(by);
	}

	@Override
	public WebElement findElement(By by) {
		return driver.findElement(by);
	}

	@Override
	public TargetLocator switchTo() {
		return driver.switchTo();
	}

	@Override
	public Navigation navigate() {
		return driver.navigate();
	}

	@Override
	public Options manage() {
		return driver.manage();
	}

	@Override
	public  X getScreenshotAs(OutputType target) throws WebDriverException {
		try {
			return ((TakesScreenshot)driver).getScreenshotAs(target);
		} catch (ClassCastException e) {
			// HTMLUNIT does not support taking screenshots, so casting is impossible
			return null;
		}
	}

	@Override
	public Object executeScript(String script, Object... args) {
		return ((JavascriptExecutor)driver).executeScript(script, args);
	}

	@Override
	public Object executeAsyncScript(String script, Object... args) {
		return ((JavascriptExecutor)driver).executeAsyncScript(script, args);
	}

	@Override
	public ScriptKey pin(String script) {
		return JavascriptExecutor.super.pin(script);
	}

	@Override
	public void unpin(ScriptKey key) {
		JavascriptExecutor.super.unpin(key);
	}

	@Override
	public Set getPinnedScripts() {
		return JavascriptExecutor.super.getPinnedScripts();
	}

	@Override
	public Object executeScript(ScriptKey key, Object... args) {
		return JavascriptExecutor.super.executeScript(key, args);
	}

	@Override
	public void perform(Collection actions) {
		((Interactive)driver).perform(actions);
		
	}

	@Override
	public void resetInputState() {
		((Interactive)driver).resetInputState();
		
	}

	/**
	 * Returns true if this is a web test
	 * a web test may be a desktop or mobile web test (e.g: using chrome browser), but also a mobile application test where webview context is in use
	 * @return
	 */
	public boolean isWebTest() {
		return (testType.family().equals(TestType.WEB) || testType.family().equals(TestType.APP) && getContext().toLowerCase().contains("web"));
	}

	public boolean isDriverExited() {
		return driverExited;
	}
	
	private static void disableStepDisplay() {
		synchronized (videoRecorders) {
			for (VideoRecorder videoRecorder: videoRecorders) {
				videoRecorder.disableStepDisplay();
			}
		}
	}
	private static void enableStepDisplay() {
		synchronized (videoRecorders) {
			for (VideoRecorder videoRecorder: videoRecorders) {
				videoRecorder.enableStepDisplay();
			}
		}
	}
	
	public static Set getVideoRecorders() {
		return videoRecorders;
	}
	
	// for tests
	public static void resetVideoRecorder() {
		videoRecorders.clear();
	}

	@Override
	public void hideKeyboard() {
		if (testType.isMobile()) {
			((HidesKeyboard)originalDriver).hideKeyboard();
		}
	}

	@Override
	public WebDriver context(String name) {
		if (testType.isMobile()) {
			((SupportsContextSwitching)originalDriver).context(name);
		}
		return this;
	}

	@Override
	public Set getContextHandles() {
		if (testType.isMobile()) {
			return ((SupportsContextSwitching)originalDriver).getContextHandles();
		} else {
			Set contexts = new HashSet<>();
			contexts.add("WEB");
			return contexts;
		}
	}

	@Nullable
	@Override
	public String getContext() {
		if (testType.isMobile()) {
			return ((SupportsContextSwitching) originalDriver).getContext();
		} else { // desktop case
			return testType.family().equals(TestType.APP) ? "APP": "WEB";
		}
	}

	@Override
	public Response execute(String driverCommand, Map parameters) {
		if (testType.isMobile()) {
			return ((AppiumDriver) originalDriver).execute(driverCommand, parameters);
		} else {
			throw new UnsupportedCommandException("execute is specific to mobile");
		}
	}

	@Override
	public Response execute(String driverCommand) {
		if (testType.isMobile()) {
			return ((AppiumDriver) originalDriver).execute(driverCommand);
		} else {
			throw new UnsupportedCommandException("execute is specific to mobile");
		}
	}

	/**
	 * For tests
	 * @param testType
	 */
	public void setTestType(TestType testType) {
		this.testType = testType;
	}

	@Override
	public ExecutesMethod assertExtensionExists(String extName) {
		if (testType.isMobile()) {
			return ((HidesKeyboard) originalDriver).assertExtensionExists(extName);
		}
		return null;
	}

	@Override
	public ExecutesMethod markExtensionAbsence(String extName) {
		if (testType.isMobile()) {
			return ((HidesKeyboard) originalDriver).markExtensionAbsence(extName);
		}
		return null;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy