vendor/php-webdriver/webdriver/lib/Remote/RemoteWebDriver.php line 111

Open in your IDE?
  1. <?php
  2. namespace Facebook\WebDriver\Remote;
  3. use Facebook\WebDriver\Interactions\WebDriverActions;
  4. use Facebook\WebDriver\JavaScriptExecutor;
  5. use Facebook\WebDriver\WebDriver;
  6. use Facebook\WebDriver\WebDriverBy;
  7. use Facebook\WebDriver\WebDriverCapabilities;
  8. use Facebook\WebDriver\WebDriverCommandExecutor;
  9. use Facebook\WebDriver\WebDriverElement;
  10. use Facebook\WebDriver\WebDriverHasInputDevices;
  11. use Facebook\WebDriver\WebDriverNavigation;
  12. use Facebook\WebDriver\WebDriverOptions;
  13. use Facebook\WebDriver\WebDriverWait;
  14. class RemoteWebDriver implements WebDriverJavaScriptExecutorWebDriverHasInputDevices
  15. {
  16.     /**
  17.      * @var HttpCommandExecutor|null
  18.      */
  19.     protected $executor;
  20.     /**
  21.      * @var WebDriverCapabilities
  22.      */
  23.     protected $capabilities;
  24.     /**
  25.      * @var string
  26.      */
  27.     protected $sessionID;
  28.     /**
  29.      * @var RemoteMouse
  30.      */
  31.     protected $mouse;
  32.     /**
  33.      * @var RemoteKeyboard
  34.      */
  35.     protected $keyboard;
  36.     /**
  37.      * @var RemoteTouchScreen
  38.      */
  39.     protected $touch;
  40.     /**
  41.      * @var RemoteExecuteMethod
  42.      */
  43.     protected $executeMethod;
  44.     /**
  45.      * @var bool
  46.      */
  47.     protected $isW3cCompliant;
  48.     /**
  49.      * @param HttpCommandExecutor $commandExecutor
  50.      * @param string $sessionId
  51.      * @param WebDriverCapabilities|null $capabilities
  52.      * @param bool $isW3cCompliant false to use the legacy JsonWire protocol, true for the W3C WebDriver spec
  53.      */
  54.     protected function __construct(
  55.         HttpCommandExecutor $commandExecutor,
  56.         $sessionId,
  57.         WebDriverCapabilities $capabilities null,
  58.         $isW3cCompliant false
  59.     ) {
  60.         $this->executor $commandExecutor;
  61.         $this->sessionID $sessionId;
  62.         $this->isW3cCompliant $isW3cCompliant;
  63.         if ($capabilities !== null) {
  64.             $this->capabilities $capabilities;
  65.         }
  66.     }
  67.     /**
  68.      * Construct the RemoteWebDriver by a desired capabilities.
  69.      *
  70.      * @param string $selenium_server_url The url of the remote Selenium WebDriver server
  71.      * @param DesiredCapabilities|array $desired_capabilities The desired capabilities
  72.      * @param int|null $connection_timeout_in_ms Set timeout for the connect phase to remote Selenium WebDriver server
  73.      * @param int|null $request_timeout_in_ms Set the maximum time of a request to remote Selenium WebDriver server
  74.      * @param string|null $http_proxy The proxy to tunnel requests to the remote Selenium WebDriver through
  75.      * @param int|null $http_proxy_port The proxy port to tunnel requests to the remote Selenium WebDriver through
  76.      * @param DesiredCapabilities $required_capabilities The required capabilities
  77.      *
  78.      * @return static
  79.      */
  80.     public static function create(
  81.         $selenium_server_url 'http://localhost:4444/wd/hub',
  82.         $desired_capabilities null,
  83.         $connection_timeout_in_ms null,
  84.         $request_timeout_in_ms null,
  85.         $http_proxy null,
  86.         $http_proxy_port null,
  87.         DesiredCapabilities $required_capabilities null
  88.     ) {
  89.         $selenium_server_url preg_replace('#/+$#'''$selenium_server_url);
  90.         $desired_capabilities self::castToDesiredCapabilitiesObject($desired_capabilities);
  91.         $executor = new HttpCommandExecutor($selenium_server_url$http_proxy$http_proxy_port);
  92.         if ($connection_timeout_in_ms !== null) {
  93.             $executor->setConnectionTimeout($connection_timeout_in_ms);
  94.         }
  95.         if ($request_timeout_in_ms !== null) {
  96.             $executor->setRequestTimeout($request_timeout_in_ms);
  97.         }
  98.         // W3C
  99.         $parameters = [
  100.             'capabilities' => [
  101.                 'firstMatch' => [(object) $desired_capabilities->toW3cCompatibleArray()],
  102.             ],
  103.         ];
  104.         if ($required_capabilities !== null && !empty($required_capabilities->toArray())) {
  105.             $parameters['capabilities']['alwaysMatch'] = (object) $required_capabilities->toW3cCompatibleArray();
  106.         }
  107.         // Legacy protocol
  108.         if ($required_capabilities !== null) {
  109.             // TODO: Selenium (as of v3.0.1) does accept requiredCapabilities only as a property of desiredCapabilities.
  110.             // This has changed with the W3C WebDriver spec, but is the only way how to pass these
  111.             // values with the legacy protocol.
  112.             $desired_capabilities->setCapability('requiredCapabilities', (object) $required_capabilities->toArray());
  113.         }
  114.         $parameters['desiredCapabilities'] = (object) $desired_capabilities->toArray();
  115.         $command WebDriverCommand::newSession($parameters);
  116.         $response $executor->execute($command);
  117.         return static::createFromResponse($response$executor);
  118.     }
  119.     /**
  120.      * [Experimental] Construct the RemoteWebDriver by an existing session.
  121.      *
  122.      * This constructor can boost the performance a lot by reusing the same browser for the whole test suite.
  123.      * You cannot pass the desired capabilities because the session was created before.
  124.      *
  125.      * @param string $selenium_server_url The url of the remote Selenium WebDriver server
  126.      * @param string $session_id The existing session id
  127.      * @param int|null $connection_timeout_in_ms Set timeout for the connect phase to remote Selenium WebDriver server
  128.      * @param int|null $request_timeout_in_ms Set the maximum time of a request to remote Selenium WebDriver server
  129.      * @param bool $isW3cCompliant True to use W3C WebDriver (default), false to use the legacy JsonWire protocol
  130.      * @return static
  131.      */
  132.     public static function createBySessionID(
  133.         $session_id,
  134.         $selenium_server_url 'http://localhost:4444/wd/hub',
  135.         $connection_timeout_in_ms null,
  136.         $request_timeout_in_ms null
  137.     ) {
  138.         // BC layer to not break the method signature
  139.         $isW3cCompliant func_num_args() > func_get_arg(4) : true;
  140.         $executor = new HttpCommandExecutor($selenium_server_urlnullnull);
  141.         if ($connection_timeout_in_ms !== null) {
  142.             $executor->setConnectionTimeout($connection_timeout_in_ms);
  143.         }
  144.         if ($request_timeout_in_ms !== null) {
  145.             $executor->setRequestTimeout($request_timeout_in_ms);
  146.         }
  147.         if (!$isW3cCompliant) {
  148.             $executor->disableW3cCompliance();
  149.         }
  150.         return new static($executor$session_idnull$isW3cCompliant);
  151.     }
  152.     /**
  153.      * Close the current window.
  154.      *
  155.      * @return RemoteWebDriver The current instance.
  156.      */
  157.     public function close()
  158.     {
  159.         $this->execute(DriverCommand::CLOSE, []);
  160.         return $this;
  161.     }
  162.     /**
  163.      * Create a new top-level browsing context.
  164.      *
  165.      * @codeCoverageIgnore
  166.      * @deprecated Use $driver->switchTo()->newWindow()
  167.      * @return WebDriver The current instance.
  168.      */
  169.     public function newWindow()
  170.     {
  171.         return $this->switchTo()->newWindow();
  172.     }
  173.     /**
  174.      * Find the first WebDriverElement using the given mechanism.
  175.      *
  176.      * @param WebDriverBy $by
  177.      * @return RemoteWebElement NoSuchElementException is thrown in HttpCommandExecutor if no element is found.
  178.      * @see WebDriverBy
  179.      */
  180.     public function findElement(WebDriverBy $by)
  181.     {
  182.         $raw_element $this->execute(
  183.             DriverCommand::FIND_ELEMENT,
  184.             JsonWireCompat::getUsing($by$this->isW3cCompliant)
  185.         );
  186.         return $this->newElement(JsonWireCompat::getElement($raw_element));
  187.     }
  188.     /**
  189.      * Find all WebDriverElements within the current page using the given mechanism.
  190.      *
  191.      * @param WebDriverBy $by
  192.      * @return RemoteWebElement[] A list of all WebDriverElements, or an empty array if nothing matches
  193.      * @see WebDriverBy
  194.      */
  195.     public function findElements(WebDriverBy $by)
  196.     {
  197.         $raw_elements $this->execute(
  198.             DriverCommand::FIND_ELEMENTS,
  199.             JsonWireCompat::getUsing($by$this->isW3cCompliant)
  200.         );
  201.         $elements = [];
  202.         foreach ($raw_elements as $raw_element) {
  203.             $elements[] = $this->newElement(JsonWireCompat::getElement($raw_element));
  204.         }
  205.         return $elements;
  206.     }
  207.     /**
  208.      * Load a new web page in the current browser window.
  209.      *
  210.      * @param string $url
  211.      *
  212.      * @return RemoteWebDriver The current instance.
  213.      */
  214.     public function get($url)
  215.     {
  216.         $params = ['url' => (string) $url];
  217.         $this->execute(DriverCommand::GET$params);
  218.         return $this;
  219.     }
  220.     /**
  221.      * Get a string representing the current URL that the browser is looking at.
  222.      *
  223.      * @return string The current URL.
  224.      */
  225.     public function getCurrentURL()
  226.     {
  227.         return $this->execute(DriverCommand::GET_CURRENT_URL);
  228.     }
  229.     /**
  230.      * Get the source of the last loaded page.
  231.      *
  232.      * @return string The current page source.
  233.      */
  234.     public function getPageSource()
  235.     {
  236.         return $this->execute(DriverCommand::GET_PAGE_SOURCE);
  237.     }
  238.     /**
  239.      * Get the title of the current page.
  240.      *
  241.      * @return string The title of the current page.
  242.      */
  243.     public function getTitle()
  244.     {
  245.         return $this->execute(DriverCommand::GET_TITLE);
  246.     }
  247.     /**
  248.      * Return an opaque handle to this window that uniquely identifies it within this driver instance.
  249.      *
  250.      * @return string The current window handle.
  251.      */
  252.     public function getWindowHandle()
  253.     {
  254.         return $this->execute(
  255.             DriverCommand::GET_CURRENT_WINDOW_HANDLE,
  256.             []
  257.         );
  258.     }
  259.     /**
  260.      * Get all window handles available to the current session.
  261.      *
  262.      * Note: Do not use `end($driver->getWindowHandles())` to find the last open window, for proper solution see:
  263.      * https://github.com/php-webdriver/php-webdriver/wiki/Alert,-tabs,-frames,-iframes#switch-to-the-new-window
  264.      *
  265.      * @return array An array of string containing all available window handles.
  266.      */
  267.     public function getWindowHandles()
  268.     {
  269.         return $this->execute(DriverCommand::GET_WINDOW_HANDLES, []);
  270.     }
  271.     /**
  272.      * Quits this driver, closing every associated window.
  273.      */
  274.     public function quit()
  275.     {
  276.         $this->execute(DriverCommand::QUIT);
  277.         $this->executor null;
  278.     }
  279.     /**
  280.      * Inject a snippet of JavaScript into the page for execution in the context of the currently selected frame.
  281.      * The executed script is assumed to be synchronous and the result of evaluating the script will be returned.
  282.      *
  283.      * @param string $script The script to inject.
  284.      * @param array $arguments The arguments of the script.
  285.      * @return mixed The return value of the script.
  286.      */
  287.     public function executeScript($script, array $arguments = [])
  288.     {
  289.         $params = [
  290.             'script' => $script,
  291.             'args' => $this->prepareScriptArguments($arguments),
  292.         ];
  293.         return $this->execute(DriverCommand::EXECUTE_SCRIPT$params);
  294.     }
  295.     /**
  296.      * Inject a snippet of JavaScript into the page for asynchronous execution in the context of the currently selected
  297.      * frame.
  298.      *
  299.      * The driver will pass a callback as the last argument to the snippet, and block until the callback is invoked.
  300.      *
  301.      * You may need to define script timeout using `setScriptTimeout()` method of `WebDriverTimeouts` first.
  302.      *
  303.      * @param string $script The script to inject.
  304.      * @param array $arguments The arguments of the script.
  305.      * @return mixed The value passed by the script to the callback.
  306.      */
  307.     public function executeAsyncScript($script, array $arguments = [])
  308.     {
  309.         $params = [
  310.             'script' => $script,
  311.             'args' => $this->prepareScriptArguments($arguments),
  312.         ];
  313.         return $this->execute(
  314.             DriverCommand::EXECUTE_ASYNC_SCRIPT,
  315.             $params
  316.         );
  317.     }
  318.     /**
  319.      * Take a screenshot of the current page.
  320.      *
  321.      * @param string $save_as The path of the screenshot to be saved.
  322.      * @return string The screenshot in PNG format.
  323.      */
  324.     public function takeScreenshot($save_as null)
  325.     {
  326.         $screenshot base64_decode($this->execute(DriverCommand::SCREENSHOT), true);
  327.         if ($save_as !== null) {
  328.             $directoryPath dirname($save_as);
  329.             if (!file_exists($directoryPath)) {
  330.                 mkdir($directoryPath0777true);
  331.             }
  332.             file_put_contents($save_as$screenshot);
  333.         }
  334.         return $screenshot;
  335.     }
  336.     /**
  337.      * Status returns information about whether a remote end is in a state in which it can create new sessions.
  338.      */
  339.     public function getStatus()
  340.     {
  341.         $response $this->execute(DriverCommand::STATUS);
  342.         return RemoteStatus::createFromResponse($response);
  343.     }
  344.     /**
  345.      * Construct a new WebDriverWait by the current WebDriver instance.
  346.      * Sample usage:
  347.      *
  348.      * ```
  349.      *   $driver->wait(20, 1000)->until(
  350.      *     WebDriverExpectedCondition::titleIs('WebDriver Page')
  351.      *   );
  352.      * ```
  353.      * @param int $timeout_in_second
  354.      * @param int $interval_in_millisecond
  355.      *
  356.      * @return WebDriverWait
  357.      */
  358.     public function wait($timeout_in_second 30$interval_in_millisecond 250)
  359.     {
  360.         return new WebDriverWait(
  361.             $this,
  362.             $timeout_in_second,
  363.             $interval_in_millisecond
  364.         );
  365.     }
  366.     /**
  367.      * An abstraction for managing stuff you would do in a browser menu. For example, adding and deleting cookies.
  368.      *
  369.      * @return WebDriverOptions
  370.      */
  371.     public function manage()
  372.     {
  373.         return new WebDriverOptions($this->getExecuteMethod(), $this->isW3cCompliant);
  374.     }
  375.     /**
  376.      * An abstraction allowing the driver to access the browser's history and to navigate to a given URL.
  377.      *
  378.      * @return WebDriverNavigation
  379.      * @see WebDriverNavigation
  380.      */
  381.     public function navigate()
  382.     {
  383.         return new WebDriverNavigation($this->getExecuteMethod());
  384.     }
  385.     /**
  386.      * Switch to a different window or frame.
  387.      *
  388.      * @return RemoteTargetLocator
  389.      * @see RemoteTargetLocator
  390.      */
  391.     public function switchTo()
  392.     {
  393.         return new RemoteTargetLocator($this->getExecuteMethod(), $this$this->isW3cCompliant);
  394.     }
  395.     /**
  396.      * @return RemoteMouse
  397.      */
  398.     public function getMouse()
  399.     {
  400.         if (!$this->mouse) {
  401.             $this->mouse = new RemoteMouse($this->getExecuteMethod(), $this->isW3cCompliant);
  402.         }
  403.         return $this->mouse;
  404.     }
  405.     /**
  406.      * @return RemoteKeyboard
  407.      */
  408.     public function getKeyboard()
  409.     {
  410.         if (!$this->keyboard) {
  411.             $this->keyboard = new RemoteKeyboard($this->getExecuteMethod(), $this$this->isW3cCompliant);
  412.         }
  413.         return $this->keyboard;
  414.     }
  415.     /**
  416.      * @return RemoteTouchScreen
  417.      */
  418.     public function getTouch()
  419.     {
  420.         if (!$this->touch) {
  421.             $this->touch = new RemoteTouchScreen($this->getExecuteMethod());
  422.         }
  423.         return $this->touch;
  424.     }
  425.     /**
  426.      * Construct a new action builder.
  427.      *
  428.      * @return WebDriverActions
  429.      */
  430.     public function action()
  431.     {
  432.         return new WebDriverActions($this);
  433.     }
  434.     /**
  435.      * Set the command executor of this RemoteWebdriver
  436.      *
  437.      * @deprecated To be removed in the future. Executor should be passed in the constructor.
  438.      * @internal
  439.      * @codeCoverageIgnore
  440.      * @param WebDriverCommandExecutor $executor Despite the typehint, it have be an instance of HttpCommandExecutor.
  441.      * @return RemoteWebDriver
  442.      */
  443.     public function setCommandExecutor(WebDriverCommandExecutor $executor)
  444.     {
  445.         $this->executor $executor;
  446.         return $this;
  447.     }
  448.     /**
  449.      * Get the command executor of this RemoteWebdriver
  450.      *
  451.      * @return HttpCommandExecutor
  452.      */
  453.     public function getCommandExecutor()
  454.     {
  455.         return $this->executor;
  456.     }
  457.     /**
  458.      * Set the session id of the RemoteWebDriver.
  459.      *
  460.      * @deprecated To be removed in the future. Session ID should be passed in the constructor.
  461.      * @internal
  462.      * @codeCoverageIgnore
  463.      * @param string $session_id
  464.      * @return RemoteWebDriver
  465.      */
  466.     public function setSessionID($session_id)
  467.     {
  468.         $this->sessionID $session_id;
  469.         return $this;
  470.     }
  471.     /**
  472.      * Get current selenium sessionID
  473.      *
  474.      * @return string
  475.      */
  476.     public function getSessionID()
  477.     {
  478.         return $this->sessionID;
  479.     }
  480.     /**
  481.      * Get capabilities of the RemoteWebDriver.
  482.      *
  483.      * @return WebDriverCapabilities
  484.      */
  485.     public function getCapabilities()
  486.     {
  487.         return $this->capabilities;
  488.     }
  489.     /**
  490.      * Returns a list of the currently active sessions.
  491.      *
  492.      * @param string $selenium_server_url The url of the remote Selenium WebDriver server
  493.      * @param int $timeout_in_ms
  494.      * @return array
  495.      */
  496.     public static function getAllSessions($selenium_server_url 'http://localhost:4444/wd/hub'$timeout_in_ms 30000)
  497.     {
  498.         $executor = new HttpCommandExecutor($selenium_server_urlnullnull);
  499.         $executor->setConnectionTimeout($timeout_in_ms);
  500.         $command = new WebDriverCommand(
  501.             null,
  502.             DriverCommand::GET_ALL_SESSIONS,
  503.             []
  504.         );
  505.         return $executor->execute($command)->getValue();
  506.     }
  507.     public function execute($command_name$params = [])
  508.     {
  509.         $command = new WebDriverCommand(
  510.             $this->sessionID,
  511.             $command_name,
  512.             $params
  513.         );
  514.         if ($this->executor) {
  515.             $response $this->executor->execute($command);
  516.             return $response->getValue();
  517.         }
  518.         return null;
  519.     }
  520.     /**
  521.      * Execute custom commands on remote end.
  522.      * For example vendor-specific commands or other commands not implemented by php-webdriver.
  523.      *
  524.      * @see https://github.com/php-webdriver/php-webdriver/wiki/Custom-commands
  525.      * @param string $endpointUrl
  526.      * @param string $method
  527.      * @param array $params
  528.      * @return mixed|null
  529.      */
  530.     public function executeCustomCommand($endpointUrl$method 'GET'$params = [])
  531.     {
  532.         $command = new CustomWebDriverCommand(
  533.             $this->sessionID,
  534.             $endpointUrl,
  535.             $method,
  536.             $params
  537.         );
  538.         if ($this->executor) {
  539.             $response $this->executor->execute($command);
  540.             return $response->getValue();
  541.         }
  542.         return null;
  543.     }
  544.     /**
  545.      * @internal
  546.      * @return bool
  547.      */
  548.     public function isW3cCompliant()
  549.     {
  550.         return $this->isW3cCompliant;
  551.     }
  552.     /**
  553.      * Create instance based on response to NEW_SESSION command.
  554.      * Also detect W3C/OSS dialect and setup the driver/executor accordingly.
  555.      *
  556.      * @internal
  557.      * @return static
  558.      */
  559.     protected static function createFromResponse(WebDriverResponse $responseHttpCommandExecutor $commandExecutor)
  560.     {
  561.         $responseValue $response->getValue();
  562.         if (!$isW3cCompliant = isset($responseValue['capabilities'])) {
  563.             $commandExecutor->disableW3cCompliance();
  564.         }
  565.         if ($isW3cCompliant) {
  566.             $returnedCapabilities DesiredCapabilities::createFromW3cCapabilities($responseValue['capabilities']);
  567.         } else {
  568.             $returnedCapabilities = new DesiredCapabilities($responseValue);
  569.         }
  570.         return new static($commandExecutor$response->getSessionID(), $returnedCapabilities$isW3cCompliant);
  571.     }
  572.     /**
  573.      * Prepare arguments for JavaScript injection
  574.      *
  575.      * @param array $arguments
  576.      * @return array
  577.      */
  578.     protected function prepareScriptArguments(array $arguments)
  579.     {
  580.         $args = [];
  581.         foreach ($arguments as $key => $value) {
  582.             if ($value instanceof WebDriverElement) {
  583.                 $args[$key] = [
  584.                     $this->isW3cCompliant ?
  585.                         JsonWireCompat::WEB_DRIVER_ELEMENT_IDENTIFIER
  586.                         'ELEMENT' => $value->getID(),
  587.                 ];
  588.             } else {
  589.                 if (is_array($value)) {
  590.                     $value $this->prepareScriptArguments($value);
  591.                 }
  592.                 $args[$key] = $value;
  593.             }
  594.         }
  595.         return $args;
  596.     }
  597.     /**
  598.      * @return RemoteExecuteMethod
  599.      */
  600.     protected function getExecuteMethod()
  601.     {
  602.         if (!$this->executeMethod) {
  603.             $this->executeMethod = new RemoteExecuteMethod($this);
  604.         }
  605.         return $this->executeMethod;
  606.     }
  607.     /**
  608.      * Return the WebDriverElement with the given id.
  609.      *
  610.      * @param string $id The id of the element to be created.
  611.      * @return RemoteWebElement
  612.      */
  613.     protected function newElement($id)
  614.     {
  615.         return new RemoteWebElement($this->getExecuteMethod(), $id$this->isW3cCompliant);
  616.     }
  617.     /**
  618.      * Cast legacy types (array or null) to DesiredCapabilities object. To be removed in future when instance of
  619.      * DesiredCapabilities will be required.
  620.      *
  621.      * @param array|DesiredCapabilities|null $desired_capabilities
  622.      * @return DesiredCapabilities
  623.      */
  624.     protected static function castToDesiredCapabilitiesObject($desired_capabilities null)
  625.     {
  626.         if ($desired_capabilities === null) {
  627.             return new DesiredCapabilities();
  628.         }
  629.         if (is_array($desired_capabilities)) {
  630.             return new DesiredCapabilities($desired_capabilities);
  631.         }
  632.         return $desired_capabilities;
  633.     }
  634. }