vendor/symfony/panther/src/Client.php line 264

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of the Panther project.
  4.  *
  5.  * (c) Kévin Dunglas <dunglas@gmail.com>
  6.  *
  7.  * For the full copyright and license information, please view the LICENSE
  8.  * file that was distributed with this source code.
  9.  */
  10. declare(strict_types=1);
  11. namespace Symfony\Component\Panther;
  12. use Facebook\WebDriver\Exception\NoSuchElementException;
  13. use Facebook\WebDriver\Exception\TimeoutException;
  14. use Facebook\WebDriver\JavaScriptExecutor;
  15. use Facebook\WebDriver\Remote\RemoteWebDriver;
  16. use Facebook\WebDriver\WebDriver;
  17. use Facebook\WebDriver\WebDriverBy;
  18. use Facebook\WebDriver\WebDriverCapabilities;
  19. use Facebook\WebDriver\WebDriverExpectedCondition;
  20. use Facebook\WebDriver\WebDriverHasInputDevices;
  21. use Symfony\Component\BrowserKit\AbstractBrowser;
  22. use Symfony\Component\BrowserKit\Request;
  23. use Symfony\Component\BrowserKit\Response;
  24. use Symfony\Component\DomCrawler\Form;
  25. use Symfony\Component\DomCrawler\Link;
  26. use Symfony\Component\Panther\Cookie\CookieJar;
  27. use Symfony\Component\Panther\DomCrawler\Crawler;
  28. use Symfony\Component\Panther\DomCrawler\Form as PantherForm;
  29. use Symfony\Component\Panther\DomCrawler\Link as PantherLink;
  30. use Symfony\Component\Panther\ProcessManager\BrowserManagerInterface;
  31. use Symfony\Component\Panther\ProcessManager\ChromeManager;
  32. use Symfony\Component\Panther\ProcessManager\FirefoxManager;
  33. use Symfony\Component\Panther\ProcessManager\SeleniumManager;
  34. use Symfony\Component\Panther\WebDriver\WebDriverMouse;
  35. /**
  36.  * @author Kévin Dunglas <dunglas@gmail.com>
  37.  * @author Dany Maillard <danymaillard93b@gmail.com>
  38.  *
  39.  * @method Crawler getCrawler()
  40.  */
  41. final class Client extends AbstractBrowser implements WebDriverJavaScriptExecutorWebDriverHasInputDevices
  42. {
  43.     use ExceptionThrower;
  44.     /**
  45.      * @var WebDriver|null
  46.      */
  47.     private $webDriver;
  48.     private $browserManager;
  49.     private $baseUri;
  50.     private $isFirefox false;
  51.     /**
  52.      * @param string[]|null $arguments
  53.      */
  54.     public static function createChromeClient(?string $chromeDriverBinary null, ?array $arguments null, array $options = [], ?string $baseUri null): self
  55.     {
  56.         return new self(new ChromeManager($chromeDriverBinary$arguments$options), $baseUri);
  57.     }
  58.     public static function createFirefoxClient(?string $geckodriverBinary null, ?array $arguments null, array $options = [], ?string $baseUri null): self
  59.     {
  60.         return new self(new FirefoxManager($geckodriverBinary$arguments$options), $baseUri);
  61.     }
  62.     public static function createSeleniumClient(?string $host null, ?WebDriverCapabilities $capabilities null, ?string $baseUri null, array $options = []): self
  63.     {
  64.         return new self(new SeleniumManager($host$capabilities$options), $baseUri);
  65.     }
  66.     public function __construct(BrowserManagerInterface $browserManager, ?string $baseUri null)
  67.     {
  68.         $this->browserManager $browserManager;
  69.         $this->baseUri $baseUri;
  70.     }
  71.     public function getBrowserManager(): BrowserManagerInterface
  72.     {
  73.         return $this->browserManager;
  74.     }
  75.     public function __destruct()
  76.     {
  77.         $this->quit();
  78.     }
  79.     public function start()
  80.     {
  81.         if (null !== $this->webDriver) {
  82.             return;
  83.         }
  84.         $this->webDriver $this->browserManager->start();
  85.         if ($this->browserManager instanceof FirefoxManager) {
  86.             $this->isFirefox true;
  87.             return;
  88.         }
  89.         if ($this->browserManager instanceof ChromeManager) {
  90.             $this->isFirefox false;
  91.             return;
  92.         }
  93.         if (method_exists($this->webDriver'getCapabilities')) {
  94.             $this->isFirefox 'firefox' === $this->webDriver->getCapabilities()->getBrowserName();
  95.             return;
  96.         }
  97.         $this->isFirefox false;
  98.     }
  99.     public function getRequest()
  100.     {
  101.         throw new \LogicException('HttpFoundation Request object is not available when using WebDriver.');
  102.     }
  103.     public function getResponse()
  104.     {
  105.         throw new \LogicException('HttpFoundation Response object is not available when using WebDriver.');
  106.     }
  107.     public function followRedirects($followRedirect true): void
  108.     {
  109.         if (!$followRedirect) {
  110.             throw new \InvalidArgumentException('Redirects are always followed when using WebDriver.');
  111.         }
  112.     }
  113.     public function isFollowingRedirects(): bool
  114.     {
  115.         return true;
  116.     }
  117.     public function setMaxRedirects($maxRedirects): void
  118.     {
  119.         if (-!== $maxRedirects) {
  120.             throw new \InvalidArgumentException('There are no max redirects when using WebDriver.');
  121.         }
  122.     }
  123.     public function getMaxRedirects(): int
  124.     {
  125.         return -1;
  126.     }
  127.     public function insulate($insulated true): void
  128.     {
  129.         if (!$insulated) {
  130.             throw new \InvalidArgumentException('Requests are always insulated when using WebDriver.');
  131.         }
  132.     }
  133.     public function setServerParameters(array $server): void
  134.     {
  135.         throw new \InvalidArgumentException('Server parameters cannot be set when using WebDriver.');
  136.     }
  137.     public function setServerParameter($key$value): void
  138.     {
  139.         throw new \InvalidArgumentException('Server parameters cannot be set when using WebDriver.');
  140.     }
  141.     public function getServerParameter($key$default '')
  142.     {
  143.         throw new \InvalidArgumentException('Server parameters cannot be retrieved when using WebDriver.');
  144.     }
  145.     public function getHistory()
  146.     {
  147.         throw new \LogicException('History is not available when using WebDriver.');
  148.     }
  149.     public function click(Link $link)
  150.     {
  151.         if ($link instanceof PantherLink) {
  152.             $link->getElement()->click();
  153.             return $this->crawler $this->createCrawler();
  154.         }
  155.         return parent::click($link);
  156.     }
  157.     public function submit(Form $form, array $values = [], array $serverParameters = [])
  158.     {
  159.         if ($form instanceof PantherForm) {
  160.             foreach ($values as $field => $value) {
  161.                 $form->get($field)->setValue($value);
  162.             }
  163.             $button $form->getButton();
  164.             if ($this->isFirefox) {
  165.                 // For Firefox, we have to wait for the page to reload
  166.                 // https://github.com/SeleniumHQ/selenium/issues/4570#issuecomment-327473270
  167.                 $selector WebDriverBy::cssSelector('html');
  168.                 $previousId $this->webDriver->findElement($selector)->getID();
  169.                 null === $button $form->getElement()->submit() : $button->click();
  170.                 try {
  171.                     $this->webDriver->wait(5)->until(static function (WebDriver $driver) use ($previousId$selector) {
  172.                         try {
  173.                             return $previousId !== $driver->findElement($selector)->getID();
  174.                         } catch (NoSuchElementException $e) {
  175.                             // The html element isn't already available
  176.                             return false;
  177.                         }
  178.                     });
  179.                 } catch (TimeoutException $e) {
  180.                     // Probably a form using AJAX, do nothing
  181.                 }
  182.             } else {
  183.                 null === $button $form->getElement()->submit() : $button->click();
  184.             }
  185.             return $this->crawler $this->createCrawler();
  186.         }
  187.         return parent::submit($form$values$serverParameters);
  188.     }
  189.     public function refreshCrawler(): Crawler
  190.     {
  191.         return $this->crawler $this->createCrawler();
  192.     }
  193.     public function request(string $methodstring $uri, array $parameters = [], array $files = [], array $server = [], string $content nullbool $changeHistory true): Crawler
  194.     {
  195.         if ('GET' !== $method) {
  196.             throw new \InvalidArgumentException('Only the GET method is supported when using WebDriver.');
  197.         }
  198.         if (null !== $content) {
  199.             throw new \InvalidArgumentException('Setting a content is not supported when using WebDriver.');
  200.         }
  201.         if (!$changeHistory) {
  202.             throw new \InvalidArgumentException('The history always change when using WebDriver.');
  203.         }
  204.         foreach (['parameters''files''server'] as $arg) {
  205.             if ([] !== $$arg) {
  206.                 throw new \InvalidArgumentException(\sprintf('The parameter "$%s" is not supported when using WebDriver.'$arg));
  207.             }
  208.         }
  209.         $this->get($uri);
  210.         return $this->crawler;
  211.     }
  212.     protected function createCrawler(): Crawler
  213.     {
  214.         $this->start();
  215.         $elements $this->webDriver->findElements(WebDriverBy::cssSelector('html'));
  216.         return new Crawler($elements$this->webDriver$this->webDriver->getCurrentURL());
  217.     }
  218.     protected function doRequest($request)
  219.     {
  220.         throw new \LogicException('Not useful in WebDriver mode.');
  221.     }
  222.     public function back()
  223.     {
  224.         $this->start();
  225.         $this->webDriver->navigate()->back();
  226.         return $this->crawler $this->createCrawler();
  227.     }
  228.     public function forward()
  229.     {
  230.         $this->start();
  231.         $this->webDriver->navigate()->forward();
  232.         return $this->crawler $this->createCrawler();
  233.     }
  234.     public function reload()
  235.     {
  236.         $this->start();
  237.         $this->webDriver->navigate()->refresh();
  238.         return $this->crawler $this->createCrawler();
  239.     }
  240.     public function followRedirect()
  241.     {
  242.         throw new \LogicException('Redirects are always followed when using WebDriver.');
  243.     }
  244.     public function restart()
  245.     {
  246.         if (null !== $this->webDriver) {
  247.             $this->webDriver->manage()->deleteAllCookies();
  248.         }
  249.         $this->quit(false);
  250.         $this->start();
  251.     }
  252.     public function getCookieJar()
  253.     {
  254.         $this->start();
  255.         return new CookieJar($this->webDriver);
  256.     }
  257.     /**
  258.      * @param string $locator The path to an element to be waited for. Can be a CSS selector or Xpath expression.
  259.      *
  260.      * @throws NoSuchElementException
  261.      * @throws TimeoutException
  262.      *
  263.      * @return Crawler
  264.      */
  265.     public function waitFor(string $locatorint $timeoutInSecond 30int $intervalInMillisecond 250)
  266.     {
  267.         $by $this->createWebDriverByFromLocator($locator);
  268.         $this->wait($timeoutInSecond$intervalInMillisecond)->until(
  269.             WebDriverExpectedCondition::presenceOfElementLocated($by)
  270.         );
  271.         return $this->crawler $this->createCrawler();
  272.     }
  273.     /**
  274.      * @param string $locator The path to an element to be waited for. Can be a CSS selector or Xpath expression.
  275.      *
  276.      * @throws NoSuchElementException
  277.      * @throws TimeoutException
  278.      *
  279.      * @return Crawler
  280.      */
  281.     public function waitForVisibility(string $locatorint $timeoutInSecond 30int $intervalInMillisecond 250)
  282.     {
  283.         $by $this->createWebDriverByFromLocator($locator);
  284.         $this->wait($timeoutInSecond$intervalInMillisecond)->until(
  285.             WebDriverExpectedCondition::visibilityOfElementLocated($by)
  286.         );
  287.         return $this->crawler $this->createCrawler();
  288.     }
  289.     public function getWebDriver(): WebDriver
  290.     {
  291.         $this->start();
  292.         return $this->webDriver;
  293.     }
  294.     public function get($uri)
  295.     {
  296.         $this->start();
  297.         // Prepend the base URI to URIs without a host
  298.         if (null !== $this->baseUri && (false !== $components = \parse_url($uri)) && !isset($components['host'])) {
  299.             $uri $this->baseUri.$uri;
  300.         }
  301.         $this->internalRequest = new Request($uri'GET');
  302.         $this->webDriver->get($uri);
  303.         $this->internalResponse = new Response($this->webDriver->getPageSource());
  304.         $this->crawler $this->createCrawler();
  305.         return $this;
  306.     }
  307.     public function close()
  308.     {
  309.         $this->start();
  310.         return $this->webDriver->close();
  311.     }
  312.     public function getCurrentURL()
  313.     {
  314.         $this->start();
  315.         return $this->webDriver->getCurrentURL();
  316.     }
  317.     public function getPageSource()
  318.     {
  319.         $this->start();
  320.         return $this->webDriver->getPageSource();
  321.     }
  322.     public function getTitle()
  323.     {
  324.         $this->start();
  325.         return $this->webDriver->getTitle();
  326.     }
  327.     public function getWindowHandle()
  328.     {
  329.         $this->start();
  330.         return $this->webDriver->getWindowHandle();
  331.     }
  332.     public function getWindowHandles()
  333.     {
  334.         $this->start();
  335.         return $this->webDriver->getWindowHandles();
  336.     }
  337.     public function quit(bool $quitBrowserManager true)
  338.     {
  339.         if (null !== $this->webDriver) {
  340.             $this->webDriver->quit();
  341.             $this->webDriver null;
  342.         }
  343.         if ($quitBrowserManager) {
  344.             $this->browserManager->quit();
  345.         }
  346.     }
  347.     public function takeScreenshot($saveAs null)
  348.     {
  349.         $this->start();
  350.         return $this->webDriver->takeScreenshot($saveAs);
  351.     }
  352.     public function wait($timeoutInSecond 30$intervalInMillisecond 250)
  353.     {
  354.         $this->start();
  355.         return $this->webDriver->wait($timeoutInSecond$intervalInMillisecond);
  356.     }
  357.     public function manage()
  358.     {
  359.         $this->start();
  360.         return $this->webDriver->manage();
  361.     }
  362.     public function navigate()
  363.     {
  364.         $this->start();
  365.         return $this->webDriver->navigate();
  366.     }
  367.     public function switchTo()
  368.     {
  369.         $this->start();
  370.         return $this->webDriver->switchTo();
  371.     }
  372.     public function execute($name$params)
  373.     {
  374.         $this->start();
  375.         return $this->webDriver->execute($name$params);
  376.     }
  377.     public function findElement(WebDriverBy $locator)
  378.     {
  379.         $this->start();
  380.         return $this->webDriver->findElement($locator);
  381.     }
  382.     public function findElements(WebDriverBy $locator)
  383.     {
  384.         $this->start();
  385.         return $this->webDriver->findElements($locator);
  386.     }
  387.     public function executeScript($script, array $arguments = [])
  388.     {
  389.         if (!$this->webDriver instanceof JavaScriptExecutor) {
  390.             throw $this->createException(JavaScriptExecutor::class);
  391.         }
  392.         return $this->webDriver->executeScript($script$arguments);
  393.     }
  394.     public function executeAsyncScript($script, array $arguments = [])
  395.     {
  396.         if (!$this->webDriver instanceof JavaScriptExecutor) {
  397.             throw $this->createException(JavaScriptExecutor::class);
  398.         }
  399.         return $this->webDriver->executeAsyncScript($script$arguments);
  400.     }
  401.     public function getKeyboard()
  402.     {
  403.         if (!$this->webDriver instanceof WebDriverHasInputDevices) {
  404.             throw $this->createException(WebDriverHasInputDevices::class);
  405.         }
  406.         return $this->webDriver->getKeyboard();
  407.     }
  408.     public function getMouse(): WebDriverMouse
  409.     {
  410.         if (!$this->webDriver instanceof WebDriverHasInputDevices) {
  411.             throw $this->createException(WebDriverHasInputDevices::class);
  412.         }
  413.         return new WebDriverMouse($this->webDriver->getMouse(), $this);
  414.     }
  415.     private function createWebDriverByFromLocator(string $locator): WebDriverBy
  416.     {
  417.         $locator trim($locator);
  418.         return '' === $locator || '/' !== $locator[0]
  419.             ? WebDriverBy::cssSelector($locator)
  420.             : WebDriverBy::xpath($locator);
  421.     }
  422.     /**
  423.      * Checks the web driver connection (and logs "pong" into the DevTools console).
  424.      *
  425.      * @param int $timeout sets the connection/request timeout in ms
  426.      *
  427.      * @return bool true if connected, false otherwise
  428.      */
  429.     public function ping(int $timeout 1000): bool
  430.     {
  431.         if (null === $this->webDriver) {
  432.             return false;
  433.         }
  434.         if ($this->webDriver instanceof RemoteWebDriver) {
  435.             $this
  436.                 ->webDriver
  437.                 ->getCommandExecutor()
  438.                 ->setConnectionTimeout($timeout)
  439.                 ->setRequestTimeout($timeout);
  440.         }
  441.         try {
  442.             if ($this->webDriver instanceof JavaScriptExecutor) {
  443.                 $this->webDriver->executeScript('console.log("pong");');
  444.             } else {
  445.                 $this->webDriver->findElement(WebDriverBy::tagName('html'));
  446.             }
  447.         } catch (\Exception $e) {
  448.             return false;
  449.         } finally {
  450.             if ($this->webDriver instanceof RemoteWebDriver) {
  451.                 $this
  452.                     ->webDriver
  453.                     ->getCommandExecutor()
  454.                     ->setConnectionTimeout(0)
  455.                     ->setRequestTimeout(0);
  456.             }
  457.         }
  458.         return true;
  459.     }
  460.     /**
  461.      * @return \LogicException|\RuntimeException
  462.      */
  463.     private function createException(string $implementableClass): \Exception
  464.     {
  465.         if (null === $this->webDriver) {
  466.             return new \LogicException(sprintf('WebDriver not started yet. Call method `start()` first before calling any `%s` method.'$implementableClass));
  467.         }
  468.         return new \RuntimeException(sprintf('"%s" does not implement "%s".', \get_class($this->webDriver), $implementableClass));
  469.     }
  470. }