Laravelでテスト駆動開発を実際にやってみて、感想や疑問に思ったことをディスカッションした。
テスト駆動開発について
フィーチャーテストってプレゼンテーション層のテストって認識でいいの?
- Laravelではコントローラーのテストをフィーチャーテストとして行っている
- Laravelの場合はフレームワークに依存するテストはフィーチャーテスト、依存しないテストはUnitテストにかくことが多い
- Laravelではこうすることが多いってだけで、フレームワークに依存するからフィーチャーテストってわけじゃない
- 複数のクラスをつなぎ合わせてテストするのがフィーチャーテスト
- 部品をつなぎ合わせてテストを行うことが目的
- 単体のクラスやメソッドをテストするのがUnitテスト
- コントローラーのテストとかモデルのテストとかではなく、機能のテストとか繋ぎ込みのテストとかが観点となる
引数を1つずつ追加してテストしながら実装をしていたけど、最初に設計ができていればある程度メソッドを実装した上でテストできそうだけど、どうなの?
- 細かくテストするのは別にいい。細かい単位でテストしていくことで前にテストが通った実装が保証される
- 設計を考えながらやっていくような時に、一個一個テストしながらしたりする
今はモックのDBとか準備するのが面倒なので基本Unitテストしか書いてないけど、実際テストはどの程度書くものなの?
テストピラミッド
テストは末端の方が低コストなので、費用対効果で考えると良さげ
DB テストサボってブラウザ操作をリリースのたびに超しなきゃいけないなら書く、とか
- バグには出るべきフェーズがある
- basic auth の設定間違いにブラウザテストで気づくのは妥当
- 料金計算が間違ってるのにブラウザテストで気づくのは遅い
- 妥当なフェーズで妥当なコストでやろうって考えると良さげ
ダミーデータを毎回入れてくれる機能があればやってる
Laravel8で完成されたModelFactoryの使い方
- 全て書くのに越したことはないが、全部が全部書いていくことは難しいので、費用対効果が高いテストを書くのがいい。最悪の最悪ユースケースのフィーチャーテストだけでも書いた方がいい
フィーチャーテストって基本はUseCaseに書くイメージであってる??
- Controllerにかくことが多い
- 開発中にフィーチャーテストをすると重たくなることが多いので、開発中はUnitテストをして、プルリク出すときに1回だけ実行することが多い
- DIすればUseCaseの単体テストもできる。
- ほとんど書くことなくてスカスカになりがちだけど
みんなは実際にコードを書くときにテスト駆動開発を行っている?
- テストは書いているけど、テスト駆動開発と呼べるものは行っていない
- 最初にコードを全部書いて、その後にテストを書いている
- コードを書いて、テストを書く、を繰り返している。今書いたコードが間違っていないことをテストで保証して、次のコードを描き始めるイメージ。
PHPのinvokeメソッドの使い所かわからない
Laravelでinvokeを使う事でシングルアクションコントローラーになる
PHPの「マジックメソッド」とは――「set()」「get()」「__invoke()」の使い方
-
invokeメソッドがある = シングルアクションコントローラーであることが明示される
- invokeに強制力はないけど(invokeがあっても他のメソッドも追加できてしまう)
class Doubler { public function __invoke($n) { return $n * 2; } }
$doubler = new Doubler(); $this->assertEquals( [2, 4, 6], array_map($doubler, [1, 2, 3]) );
array_map
とかにはまりやすいというメリットはある↓ をシンプルにかける
$this->assertEquals( [2, 4, 6], array_map(function($n) { return $n * 2; }, [1, 2, 3]) );
憶測だけど、無名関数の文法ができる前には価値があったんじゃあないかな
↓ 今は再利用したいなら別にこうやって無名関数を束縛すれば、別にいちいちクラスを作らなくても済むし、という意味
$doubler = function($n) { return $n * 2; };
-
Laravelの場合、 __invoke をコントローラで書くとルーティングがシンプルになる
// Laravel側で __invoke メソッドが呼ばれる Route::post('/price', PriceController::class)->name('price'); // Laravel側で price メソッドが呼ばれる Route::post('/price', [PriceController::class, 'price'])->name('price');
https://qiita.com/mitsuru793/items/e25b4351e8a63a229b8a#invokeしかできないこと
-
マジックメソッド
- PHPが準備してくれている関数 + PHPが勝手に呼び出してくれる関数
-
magic method は
__toString
が一番イメージしやすいかもclass Foo { public function __toString() { return "I'm hoge-san."; } }
$foo = new Foo(); $this->assertEquals("Hello, $foo", "Hello, I'm hoge-san.");