5
11

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

AsiaQuestAdvent Calendar 2019

Day 21

PHP + phpunit + php-webdriver + docker-selenium でブラウザテスト ②テスト実行、エラー対策

Last updated at Posted at 2019-12-25

目次

PHP + phpunit + php-webdriver + docker-selenium でブラウザテスト ①環境構築
PHP + phpunit + php-webdriver + docker-selenium でブラウザテスト ②テスト実行、エラー対策

①の続きです。

サンプルテストを書く

本格的に書く前に、テスト実行環境が完成したかどうか確認します。

testsディレクトリ配下に SampleTest.php というファイルを作成します。(ファイル名はなんでもOK)

<?php

use Facebook\WebDriver\Chrome\ChromeOptions;
use Facebook\WebDriver\Remote\DesiredCapabilities;
use Facebook\WebDriver\Remote\RemoteWebDriver;
use PHPUnit\Framework\TestCase;

class SampleTest extends TestCase
{
    /**
     * @group google
     */
    public function testGoogle()
    {
    	// Seleniumサーバを指定
    	// Dockerコンテナ内からlocalhostにアクセスするためhost.docker.internalを使う
        $seleniumServerHost = 'http://host.docker.internal:4444/wd/hub';

		// Chromeドライバの設定項目を指定
        $options = new ChromeOptions();
        $options->addArguments([
            '--headless',     // ← ヘッドレスモードでChromeを起動
            '--disable-gpu', 
            '--no-sandbox',  // ← Windows環境でのみ必要らしい。消しても動いたのでDocker環境なら不要かも。
            ]);

		// Chromeドライバに先ほどの設定を反映する
        $desiredCapabilities = DesiredCapabilities::chrome()
            ->setCapability(ChromeOptions::CAPABILITY, $options);
        // Chromeドライバを生成
        $driver = RemoteWebDriver::create($seleniumServerHost, $desiredCapabilities);

		// 指定したURLにアクセス
        $driver->get('https://www.google.com');

		//getTitle()メソッドでそのサイトのタイトルを取得、getCurrentURL()メソッドでURLを取得
        echo PHP_EOL, $driver->getTitle(), PHP_EOL;
        echo $driver->getCurrentURL();
        $this->assertSame('Google', $driver->getTitle());

        // ブラウザのウィンドウを閉じ、セッションも終了する。
        // close()メソッドはウィンドウを閉じるだけでセッションが終了されないので注意。
        $driver->quit();
    }
}

Chromeドライバのオプション
ヘッドレスモードだと画面表示せずテストが実行されるためテストにかかる時間が減るらしい。が、このオプションをつけても後述するVNCサーバで動作を目視できるので効いていないかもしれない・・・。
古い情報しか見つけられなかったが、ヘッドレス Chrome ことはじめによると、ヘッドレスモードで起動する場合は --disable-gpu が必須。

ほか
そのほかテストの書き方については、
ドライバの公式 facebook/php-webdriver のWikiや
Qiitaの記事 PHPUnit + php-webdriver でWebUIのテストを書く などを参考にしてください。

参考

facebook/php-webdriver
seleniumを使ってPHPでChromeの自動操作をする
PHPUnit + php-webdriver でWebUIのテストを書く
規模別PHPUnitでのテストの書き方いろいろ

サンプルテストを実行する

テストを実行するには、コマンドラインで実行する方法と、PhpStormで実行する方法があります。

コマンドラインで

  • すべてのテストを実行する場合
$ docker-compose run --rm php vendor/bin/phpunit
  • ファイル名指定して実行する場合
$ docker-compose run --rm php vendor/bin/phpunit tests/SampleTest.php

image.png

  • groupを指定して実行する場合
    テストの数が多くて時間がかかるなど、特定のテストだけ実行したい場合などに使われるようです。
    テストメソッドに、@groupアサーションと、group名を追加し、
    /**
     * @group google
     */
    public function testGoogle()
    {
    	// 省略
    }

次のように--groupフラグに実行したいグループ名を付けて実行

$ docker-compose run --rm php vendor/bin/phpunit --group="google"

PhpStormで

Edit Configurations... を選択
image.png

②+マークからPHPUnitを選択し、新しいテンプレートを追加
image.png

③ディレクトリ/クラス/メソッド単位で実行したいテストの範囲を定め、OKをクリック
また、コマンドでオプション指定するのと同様にオプションを指定したいときは、Test Runner options: に入力します。
image.png

このテスト実行時の設定をGitなどでバージョン管理したい場合、右上のチェックボックスShare through VSCにチェックを入れます。
もし「このファイルまたはディレクトリは無視されているからVCSに追加できない」というエラーが出たら、
画面の指示にしたがってRemove from ignored filesをクリックすればバージョン管理できるように自動で.gitignoreを書き換えてくれます。

手動で書き換える場合、次のように記述すると.ideaディレクトリの他のファイルは無視しつつ、テスト実行設定ファイルだけを管理できます。

/.idea/*
!/.idea/runConfigurations/*

④作成したテスト実行テンプレートを選択した状態で再生ボタンクリック
image.png

これでテストが実行されます。
image.png

参考

.gitignore の書き方。ファイル/ディレクトリの除外と反映方法
[Git] .gitignoreの仕様詳解

おまけ VNCで爆速で入力している様子が見れる

  • UltraVNCをDL

  • インストールとか諸々をこなす

  • localhost:5900にConnect ※ポート番号はdocker-compose.ymlでnodeコンテナに割り当てた番号

  • パスワード入力(初期値はsecret)

  • この状態になったら待機
    image.png

  • テスト実行すると、Seleniumが働いている様子が見れる。デザインなど目視したい場合などに使うもの...?
    私の場合、ブラウザテストを書いていてエラーの原因がわからないときに稀に参考になった。

  • この記事では特に活用しないので楽しい気分を味わいたい人向け。
    image.png

テストの書き方tips

HTTP通信ヘッダ情報は、Seleniumで取得できない

Selenium実践入門 自動化による継続的なブラウザテスト p111

によると、

HTTP通信ヘッダ情報は、そのままではWebDriverプログラムで取得できません。
HTTPヘッダの内容を取得するには、HTTPプロキシを利用するのがお勧めです。

ということらしいので、
ステータスコードの確認などをしたい場合、curlなどを利用する必要がある。

「なぜHTTPヘッダーを返すメソッドがないのか」という問いに対する公式回答はこれっぽい。
全部は読んでいないが、一番最後に

Browsers can check that data via js code, so I do not see why the webdriver should
be more limited / not expose it

ブラウザはjsコードを介してデータをチェックできるので、なぜWeb Driverがそうするべきかわからない

とある。他の方法でできるんだから実装しなくていいでしょ、ってことらしい。

新しいタブ/ウィンドウに切り替える方法

ボタンをクリックすると別タブが開かれ、別タブのほうをSeleniumで操作したいときは、操作対象を明示する必要があります。

次のコードでは、任意の操作の前後で全てのウィンドウを取得し、その差分を新しいウィンドウとしています。

    $windowHandlesBefore = $this->driver->getWindowHandles();

	// クリックなどの任意の操作
    $this->driver->findElement(WebDriverBy::cssSelector('.ok-button'))->click();

    $windowHandlesAfter = $this->driver->getWindowHandles();
    $newWindowHandle = array_diff($windowHandlesAfter, $windowHandlesBefore);
    $this->driver->switchTo()->window($newWindowHandle);

この方法は、公式のwikiに倣った。(注意点などいろいろと記載されているので不明点があったらまず読むのがおすすめ)

Note: Do not use end($handles) to find the last open window, because getWindowHandles() does not necessarily return window in the order in which they were opened.

注:最後に開いたウィンドウを見つけるためにend($handles)を使用しないでください。
なぜなら、getWindowHandles()は、ウィンドウが開かれた順に返されるとは限らないから。

WebDriverBy::cssSelector()メソッドを使えばid属性やclass属性がなくても要素を指定できる

class属性、id属性、name属性、が何も指定されていない。
HTMLタグで指定しようとすると重複して特定の要素を指定できない。
CSSやJSが効いているので、テストのためだけに実装側を変える時間はない。
for属性ならあるけど...

という要素に対して操作したい。
そんなときは、WebDriverBy::cssSelector()メソッドを使い、CSSやJSでDOM要素を指定するように要素を指定すればOK。

$this->driver->findElement(WebDriverBy::cssSelector('.form-check-list li label[for="apple"]'))->click();

id()メソッド、className()メソッド、xpath()メソッドなど他にもメソッドは存在するが、できるだけ統一するようにプロジェクト内でルールを決めたほうが見やすい。

エラー対策

Facebook\WebDriver\Exception\ElementNotVisibleException : element not interactable

ざっくりいうと、 display: none; によって要素が見えていないとclick()などの操作ができない、というエラー。
「画面外に存在していて、スクロールしないと見えない要素」のことではない。画面外にあろうがSeleniumで操作を行える。

チェックボックスやラジオボタンなど、input属性要素を隠して上に別の要素を表示していると、
モーダルなどと同様、非表示状態であるとみなされるため起きているエラーだった。

そのため、本当に表示されている要素、display: block;な要素を選択してclick()メソッドを呼ぶ必要がある。
cssSelector()メソッドを使い、細かく指定すれば可能。

Selenium 日本語ドキュメント Seleniumのコマンド

Facebook\WebDriver\Exception\WebDriverCurlException : Curl error thrown for http POST to /session with params

Facebook\WebDriver\Exception\WebDriverCurlException : Curl error thrown for http POST to /session with params: {"desiredCapabilities":{"browserName":"chrome","platform":"LINUX","chromeOptions":{"w3c":false,"binary":"","args":["--disable-gpu","--no-sandbox"]}}}

または

Facebook\WebDriver\Exception\WebDriverCurlException : Curl error thrown for http POST to /session/{ランダムの文字列(たぶんセッションのhash)}/url with params:(省略)

というエラーに何度も出くわした。
また、テストが成功したり失敗したり不安定だった。

解決するために下記を試したが、正しい解決方法はわからない。

  • ドライバをヘッドレスで実行する。

  • phpunitのtearDown()メソッドを用い、テスト成功/失敗に関係なく、必ずウィンドウプロセスを終了させる。

        protected function tearDown(): void
        {
            $this->driver->quit();
        }
    

以下、 Selenium実践入門 自動化による継続的なブラウザテスト p43 より

quitメソッドを呼び出さずにテストプロセスを終了すると、ブラウザのウィンドウは閉じられず、
Chrome Driverサーバ・IEDriverサーバ・PhantomJSのプロセスも終了されずに残ってしまいます。
新たにWebDriverインスタンスを生成しても、これらのウィンドウプロセスは再利用されないため、
残ってしまったものは手動で終了させる必要があります。テストプログラムを途中で強制終了した場合も、これらのウィンドウ・プロセスは当然残ってしまいます。
たとえばデバッグ中のテストをEclipseの「停止」ボタンで終了させた場合も、quitメソッドは呼び出されず、
ウィンドウ・プロセスが残ってしまいます。

  • ドライバ生成時にタイムアウトの時間(ms)を指定できるが、0を指定しない。また、制限時間が短すぎると、ページのロードが終わる前にタイムアウトになりテストが実行されないので、余裕をもった時間を指定する。

    $this->driver = RemoteWebDriver::create($seleniumServerHost, $desiredCapabilities, 80000, 80000);
    

このエラーは出にくくなったが、
テストが成功することもあれば、
WebDriverCurlExceptionが再発生したり、TimeOutExceptionが発生したりしている。

明示的な待機wait()~until()か、
暗黙的な待機implicitlyWait()を使う必要がありそう。調査中。
https://github.com/facebook/php-webdriver/wiki/HowTo-Wait#implicit-waits
Selenium で別ウィンドウが開くまで待つ

どうやらSeleniumで別タブを開くこと自体が不安定になる原因のよう。
全国のSeleniumer必読?より

別タブで開くことの検証別タブで開くことの検証

  • 別タブ/ウィンドウで開くことのテストは可能
  • ボタン等をクリックすると新しいウィンドウが「ぽこっ」と出て、waitForPopUpなりselectWindowなりをかましてカレントウィンドウにするとか
  • ただ、できればそのやり方はしないほうが良い
  • ウィンドウが立ち上がるまでに時間がかかって不安定のなテストの原因となりかねない
  • そもそも別タブ/ウィンドウで開くことはHTMLレベルで判断でき、ブラウザの機能として提供されているため、わざわざSeleniumでその挙動まで見る必要はない

hiddenパラメータでデータを送信しなければいけない場合はどうしたらいいのか...。

【追記 2020/01/09】
テストの結果が不安定だったのは、wait()~until()を使い、
「クリックできる状態になるまで待機してからクリックする」「画面に表示されるまで待機してから操作する」という処理を入れていなかったため。
(私の場合、セレクトボックスを選択する前に「クリックできる状態になるまで待機」という処理を入れるとテストが安定した。)
↓下記記事とそのリンク先は、wait()~until()メソッドを使うとき、どれを使ったらいいのか参考になった。
ちゃんと動いていたSeleniumがたまに失敗するケースまとめ

例えばvisibilityOfElementLocated()は表示されるまで待ってくれるが、presenceOfElementLocated()は表示/非表示にかかわらずDOM要素の存在だけしか待たない。

もちろん、テスト失敗時や、テスト実行を自分で止めた場合などに、確実にプロセスを停止させることも必要。
DockerでSelenumの環境を作っておくと、
(テストを実行してからテストコードの誤りに気付いたときなどに)テスト実行を止めてDockerコンテナ終了→起動→すぐテスト再開 ができるので便利だなと思った。
むしろ、パソコンのホスト上でやったりした場合、どうやってプロセス終了させるんだろう・・・。いちいち$driver->quit()を呼び出すんだろうか。
【追記ここまで】

参考

Selenium実践入門 自動化による継続的なブラウザテスト
全国のSeleniumer必読
なぜSeleniumテストは不安定なのかとその対策
Selenium で別ウィンドウが開くまで待つ

メモ

Laravel+TDDの基礎
Laravel dusk使ってて面白そう

5
11
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
11

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?