例
テスト環境の設定方法とアサーションの作成方法を説明しましたが、
まだまだ戦いは続きます。それでは、
実際のテストシナリオをもとにテストの方法を確認していきましょう。
UserController のテスト
ウェブサイトの一般的なタスクである、
ユーザ認証とユーザ登録について考えてみましょう。
今回の例では UserController でこれらを処理することにします。
要件は次のとおりです。
ユーザがまだ認証を済ませていない場合は、
どんなアクションが指定されたかにかかわらず
常にコントローラのログインページにリダイレクトされる。
ログインフォームのページには、
ログインフォームと新規登録フォームの両方が表示される。
間違った認証情報を入力すると、
ログインフォームに戻る。
正しい認証情報を入力すると、
ユーザのプロファイルページにリダイレクトされる。
プロファイルページには、そのユーザのユーザ名が表示される。
認証済みのユーザがログインフォームを訪れると、
そのユーザのプロファイルページにリダイレクトされる。
ログアウトしたら、ログインページにリダイレクトされる。
無効なデータが渡された場合は、登録に失敗する。
もちろんこれら以外にも別のテストも必要でしょうが、
今のところはひとまずこれだけにしておきます。
今回のアプリケーションでは、プラグイン 'Initialize'
を定義してそれを routeStartup() で実行します。
これによって起動処理をオブジェクト指向でカプセル化することができ、
コールバックを提供しやすくなります。
それではまず、このクラスの基本部分を見ていきましょう。
_setEnv($env);
if (null === $root) {
$root = realpath(dirname(__FILE__) . '/../../../');
}
$this->_root = $root;
$this->initPhpConfig();
$this->_front = Zend_Controller_Front::getInstance();
}
/**
* ルートの開始処理
*
* @return void
*/
public function routeStartup(Zend_Controller_Request_Abstract $request)
{
$this->initDb();
$this->initHelpers();
$this->initView();
$this->initPlugins();
$this->initRoutes();
$this->initControllers();
}
// この後にメソッド定義が続きます...
}
]]>
これで、起動用コールバックを次のように作れるようになります。
getFrontController();
$controller->registerPlugin(
new Bugapp_Plugin_Initialize('development')
);
}
public function setUp()
{
$this->bootstrap = array($this, 'appBootstrap');
parent::setUp();
}
// ...
}
]]>
ここまでできたら、テストを書くことができます。
しかし、ユーザがログインした状態でのテストはどのように書けばいいでしょう?
簡単な方法は、アプリケーションのロジックを利用する方法です。
resetRequest() メソッドや
resetResponse() メソッドを使ってちょっとした細工を行い、
別のリクエストをディスパッチさせます。
request->setMethod('POST')
->setPost(array(
'username' => $user,
'password' => $password,
));
$this->dispatch('/user/login');
$this->assertRedirectTo('/user/view');
$this->resetRequest()
->resetResponse();
$this->request->setPost(array());
// ...
}
// ...
}
]]>
ではテストを書いてみましょう。
dispatch('/user');
$this->assertController('user');
$this->assertAction('index');
}
public function testLoginFormShouldContainLoginAndRegistrationForms()
{
$this->dispatch('/user');
$this->assertQueryCount('form', 2);
}
public function testInvalidCredentialsShouldResultInRedisplayOfLoginForm()
{
$request = $this->getRequest();
$request->setMethod('POST')
->setPost(array(
'username' => 'bogus',
'password' => 'reallyReallyBogus',
));
$this->dispatch('/user/login');
$this->assertNotRedirect();
$this->assertQuery('form');
}
public function testValidLoginShouldRedirectToProfilePage()
{
$this->loginUser('foobar', 'foobar');
}
public function testAuthenticatedUserShouldHaveCustomizedProfilePage()
{
$this->loginUser('foobar', 'foobar');
$this->request->setMethod('GET');
$this->dispatch('/user/view');
$this->assertNotRedirect();
$this->assertQueryContentContains('h2', 'foobar');
}
public function
testAuthenticatedUsersShouldBeRedirectedToProfileWhenVisitingLogin()
{
$this->loginUser('foobar', 'foobar');
$this->request->setMethod('GET');
$this->dispatch('/user');
$this->assertRedirectTo('/user/view');
}
public function testUserShouldRedirectToLoginPageOnLogout()
{
$this->loginUser('foobar', 'foobar');
$this->request->setMethod('GET');
$this->dispatch('/user/logout');
$this->assertRedirectTo('/user');
}
public function testRegistrationShouldFailWithInvalidData()
{
$data = array(
'username' => 'This will not work',
'email' => 'this is an invalid email',
'password' => 'Th1s!s!nv@l1d',
'passwordVerification' => 'wrong!',
);
$request = $this->getRequest();
$request->setMethod('POST')
->setPost($data);
$this->dispatch('/user/register');
$this->assertNotRedirect();
$this->assertQuery('form .errors');
}
}
]]>
これらは簡潔なものであり、大半は実際の中身までは見ていないことに注意しましょう。
その代わりに、レスポンスコードやヘッダ、そして DOM ノードを見ています。
これにより、期待通りの構造になっているかどうかを検証できるようになり、
新たなコンテンツが追加されるたびにテストを実行しなおすことが避けられます。
ドキュメントの構造を使用してテストを行なっていることに注目しましょう。
たとえば最後のテストでは、"errors"
というクラスが指定されているノードをフォームから探しました。
これにより、単にフォームの検証エラーが発生したかどうかだけを確認することができ、
どんなエラーが発生したのかという中身までは気にしなくてすむのです。
このアプリケーションでは、データベースを使うことがあるかもしれません。
そんな場合は、何らかの scaffold を使用してデータベースの初期状態を作成し、
テスト用の設定を行うという作業が各テストの最初に発生します。
PHPUnit にはそのための機能が既に用意されています。
PHPUnit のドキュメントを参照ください。
テスト時と実運用時には別のデータベースを使用することを推奨します。
また、特に (ファイルあるいはインメモリ形式の) SQLite
を使うことを推奨します。どちらも別のサーバを必要とせず、
大半の SQL 構文を使用できます。