Laravelに限らずですが外部のAPIを通信したり、重い処理を実行しているの部分があるのでテストの時はモックにしたい、といった場面は多々あると思います。
今回はServiceコンテナをモック化する方法をまとめていきます。
Serviceコンテナが使われているコード
public function store(Request $request, User $user)
{
// ScrapeManagerの生成
$scrapers = resolve('app.bookInfo.scrapeManager');
// ScrapeManagerの実行
$newBook = $scrapers->searchByIsbn((string)$isbn);
// ......
Serviceコンテナのキーを使ってresolveで生成しています。
この中身をモック化できればいい感じにテストができそうです。
(Managerなんて責務の大きい名前を使ってんじゃねぇ、という意見は却下します)
テスト内でモック化する
public function setUp()
{
// モックを作成
$mock = \Mockery::mock(ScrapeManager::class)
->shouldReceive('searchByIsbn')
->andReturn($this->book)
->getMock();
// モックを適用
$this->app->bind('app.bookInfo.scrapeManager', function() use ($mock) {
return $mock;
});
}
実はこれだけです。
Mockery::mock
でモック化対象のクラスを指定します。
shouldReceive
で対象メソッドを指定します。
andReturn
で対象メソッドの戻り値を設定します。
あとは最後にgetMock
を実行してモックのインスタンスを獲得することを忘れないようにしてください。
次に作成したモックをresolveで解決できる形に設定する必要があります。
$this->app->bind
を使います。
第一引数にresolve時に使うキーを指定し、第二引数にはモックを返す関数を設定します。
テスト終了後にMockery::close()
を実行することを忘れないようにしましょう。
あとはただただテストをするだけです。
そうすれば処理の中でモック化したキーのresolveが使われれば、モックオブジェクトが呼び出されて任意の処理になります。
テストについて
Laravelでコントローラーテストをする場合にはこちらの記事をご覧ください。
Laravel5.7でAPIコントローラーをメソッドとしてテストする方法
まとめ
処理の重い部分、特に外部に影響を与える部分はちゃんとモック化しましょう!!