Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x
  2. AbstractWebDriver.php

class AbstractWebDriver

Abstract WebDriver\AbstractWebDriver class

@package WebDriver

Hierarchy

  • class \WebDriver\AbstractWebDriver

Expanded class hierarchy of AbstractWebDriver

1 file declares its use of AbstractWebDriver
AbstractStorage.php in vendor/lullabot/php-webdriver/lib/WebDriver/Storage/AbstractStorage.php

File

vendor/lullabot/php-webdriver/lib/WebDriver/AbstractWebDriver.php, line 21

Namespace

WebDriver
View source
abstract class AbstractWebDriver {
    
    /**
     * URL
     *
     * @var string
     */
    protected $url;
    
    /**
     * Curl service
     *
     * @var \WebDriver\Service\CurlServiceInterface
     */
    private $curlService;
    
    /**
     * Transient options
     *
     * @var array
     */
    private $transientOptions;
    
    /**
     * Return array of supported method names and corresponding HTTP request methods
     *
     * @return array
     */
    protected abstract function methods();
    
    /**
     * Return array of obsolete method names and corresponding HTTP request methods
     *
     * @return array
     */
    protected function obsoleteMethods() {
        return array();
    }
    
    /**
     * Constructor
     *
     * @param string $url URL to Selenium server
     */
    public function __construct($url = 'http://localhost:4444/') {
        $this->url = $url;
        $this->transientOptions = array();
        $this->curlService = ServiceFactory::getInstance()->getService('service.curl');
    }
    
    /**
     * Magic method which returns URL to Selenium server
     *
     * @return string
     */
    public function __toString() {
        return $this->url;
    }
    
    /**
     * Returns URL to Selenium server
     *
     * @return string
     */
    public function getURL() {
        return $this->url;
    }
    
    /**
     * Set curl service
     *
     * @param \WebDriver\Service\CurlServiceInterface $curlService
     */
    public function setCurlService($curlService) {
        $this->curlService = $curlService;
    }
    
    /**
     * Get curl service
     *
     * @return \WebDriver\Service\CurlServiceInterface
     */
    public function getCurlService() {
        return $this->curlService;
    }
    
    /**
     * Set transient options
     *
     * @param mixed $transientOptions
     */
    public function setTransientOptions($transientOptions) {
        $this->transientOptions = is_array($transientOptions) ? $transientOptions : array();
    }
    
    /**
     * @return array
     */
    public function getTransientOptions() {
        return $this->transientOptions;
    }
    
    /**
     * Curl request to webdriver server.
     *
     * @param string               $requestMethod HTTP request method, e.g., 'GET', 'POST', or 'DELETE'
     * @param string               $command       If not defined in methods() this function will throw.
     * @param array|integer|string $parameters    If an array(), they will be posted as JSON parameters
     *                                            If a number or string, "/$params" is appended to url
     * @param array                $extraOptions  key=>value pairs of curl options to pass to curl_setopt()
     *
     * @return array array('value' => ..., 'info' => ...)
     *
     * @throws \WebDriver\Exception if error
     */
    protected function curl($requestMethod, $command, $parameters = null, $extraOptions = array()) {
        if ($parameters && is_array($parameters) && $requestMethod !== 'POST') {
            throw WebDriverException::factory(WebDriverException::NO_PARAMETERS_EXPECTED, sprintf('The http request method called for %s is %s but it has to be POST if you want to pass the JSON parameters %s', $command, $requestMethod, json_encode($parameters)));
        }
        $url = sprintf('%s%s', $this->url, $command);
        if ($parameters && (is_int($parameters) || is_string($parameters))) {
            $url .= '/' . $parameters;
        }
        $this->assertSerializable($parameters);
        list($rawResult, $info) = $this->curlService
            ->execute($requestMethod, $url, $parameters, array_replace($extraOptions, $this->transientOptions));
        $this->transientOptions = array();
        $httpCode = $info['http_code'];
        if ($httpCode === 0) {
            throw WebDriverException::factory(WebDriverException::CURL_EXEC, $info['error']);
        }
        $result = json_decode($rawResult, true);
        if (!empty($rawResult) && $result === null && json_last_error() != JSON_ERROR_NONE) {
            // Legacy webdriver 4xx responses are to be considered a plaintext error
            if ($httpCode >= 400 && $httpCode <= 499) {
                throw WebDriverException::factory(WebDriverException::CURL_EXEC, 'Webdriver http error: ' . $httpCode . ', payload :' . substr($rawResult, 0, 1000));
            }
            throw WebDriverException::factory(WebDriverException::CURL_EXEC, 'Payload received from webdriver is not valid json: ' . substr($rawResult, 0, 1000));
        }
        if (is_array($result) && !array_key_exists('status', $result) && !array_key_exists('value', $result)) {
            throw WebDriverException::factory(WebDriverException::CURL_EXEC, 'Payload received from webdriver is valid but unexpected json: ' . substr($rawResult, 0, 1000));
        }
        $value = $this->offsetGet('value', $result);
        if (($message = $this->offsetGet('message', $result)) === null) {
            $message = $this->offsetGet('message', $value);
        }
        // if not success, throw exception
        if (isset($result['status']) && (int) $result['status'] !== 0) {
            throw WebDriverException::factory($result['status'], $message);
        }
        if (($error = $this->offsetGet('error', $result)) === null) {
            $error = $this->offsetGet('error', $value);
        }
        if (isset($error)) {
            throw WebDriverException::factory($error, $message);
        }
        $sessionId = ($this->offsetGet('sessionId', $result) ?: $this->offsetGet('sessionId', $value)) ?: $this->offsetGet('webdriver.remote.sessionid', $value);
        return array(
            'value' => $value,
            'info' => $info,
            'sessionId' => $sessionId,
            'sessionUrl' => $sessionId ? $this->url . '/session/' . $sessionId : $info['url'],
        );
    }
    
    /**
     * Magic method that maps calls to class methods to execute WebDriver commands
     *
     * @param string $name      Method name
     * @param array  $arguments Arguments
     *
     * @return mixed
     *
     * @throws \WebDriver\Exception if invalid WebDriver command
     */
    public function __call($name, $arguments) {
        if (count($arguments) > 1) {
            throw WebDriverException::factory(WebDriverException::JSON_PARAMETERS_EXPECTED, 'Commands should have at most only one parameter, which should be the JSON Parameter object');
        }
        if (preg_match('/^(get|post|delete)/', $name, $matches)) {
            $requestMethod = strtoupper($matches[0]);
            $webdriverCommand = strtolower(substr($name, strlen($requestMethod)));
            $this->getRequestMethod($webdriverCommand);
            // validation
        }
        else {
            $webdriverCommand = $name;
            $requestMethod = $this->getRequestMethod($webdriverCommand);
        }
        $methods = $this->methods();
        if (!in_array($requestMethod, (array) $methods[$webdriverCommand])) {
            throw WebDriverException::factory(WebDriverException::INVALID_REQUEST, sprintf('%s is not an available http request method for the command %s.', $requestMethod, $webdriverCommand));
        }
        $result = $this->curl($requestMethod, '/' . $webdriverCommand, array_shift($arguments));
        return $result['value'];
    }
    
    /**
     * Sanity check
     *
     * @param mixed $parameters
     */
    private function assertSerializable($parameters) {
        if ($parameters === null || is_scalar($parameters)) {
            return;
        }
        if (is_array($parameters)) {
            foreach ($parameters as $value) {
                $this->assertSerializable($value);
            }
            return;
        }
        throw WebDriverException::factory(WebDriverException::UNEXPECTED_PARAMETERS, sprintf("Unable to serialize non-scalar type %s", is_object($parameters) ? get_class($parameters) : gettype($parameters)));
    }
    
    /**
     * Extract value from result
     *
     * @param string $key
     * @param mixed  $result
     *
     * @return string|null
     */
    private function offsetGet($key, $result) {
        return is_array($result) && array_key_exists($key, $result) ? $result[$key] : null;
    }
    
    /**
     * Get default HTTP request method for a given WebDriver command
     *
     * @param string $webdriverCommand
     *
     * @return string
     *
     * @throws \WebDriver\Exception if invalid WebDriver command
     */
    private function getRequestMethod($webdriverCommand) {
        if (!array_key_exists($webdriverCommand, $this->methods())) {
            throw WebDriverException::factory(array_key_exists($webdriverCommand, $this->obsoleteMethods()) ? WebDriverException::OBSOLETE_COMMAND : WebDriverException::UNKNOWN_COMMAND, sprintf('%s is not a valid WebDriver command.', $webdriverCommand));
        }
        $methods = $this->methods();
        $requestMethods = (array) $methods[$webdriverCommand];
        return array_shift($requestMethods);
    }

}

Members

Title Sort descending Modifiers Object type Summary Overrides
AbstractWebDriver::$curlService private property Curl service
AbstractWebDriver::$transientOptions private property Transient options
AbstractWebDriver::$url protected property URL
AbstractWebDriver::assertSerializable private function Sanity check
AbstractWebDriver::curl protected function Curl request to webdriver server.
AbstractWebDriver::getCurlService public function Get curl service
AbstractWebDriver::getRequestMethod private function Get default HTTP request method for a given WebDriver command
AbstractWebDriver::getTransientOptions public function
AbstractWebDriver::getURL public function Returns URL to Selenium server
AbstractWebDriver::methods abstract protected function Return array of supported method names and corresponding HTTP request methods 15
AbstractWebDriver::obsoleteMethods protected function Return array of obsolete method names and corresponding HTTP request methods 4
AbstractWebDriver::offsetGet private function Extract value from result
AbstractWebDriver::setCurlService public function Set curl service
AbstractWebDriver::setTransientOptions public function Set transient options
AbstractWebDriver::__call public function Magic method that maps calls to class methods to execute WebDriver commands 1
AbstractWebDriver::__construct public function Constructor 3
AbstractWebDriver::__toString public function Magic method which returns URL to Selenium server

API Navigation

  • Drupal Core 11.1.x
  • Topics
  • Classes
  • Functions
  • Constants
  • Globals
  • Files
  • Namespaces
  • Deprecated
  • Services
RSS feed
Powered by Drupal