ホームページ >バックエンド開発 >PHPチュートリアル >[Lumen 5.2 ドキュメント] その他の機能 -- 単体テスト
ExampleTest.php ファイルは、tests ディレクトリに提供されます。新しい Lumen アプリケーションをインストールした後、コマンドラインで phpunit を実行するだけでテストを実行できます。
1.1 テスト環境
必要に応じて、他のテスト環境構成を自由に作成できます。テスト環境変数は phpunit.xml ファイルで構成できます。
1.2 テストの定義と実行
<?phpclass FooTest extends TestCase{ public function testSomethingIsTrue() { $this->assertTrue(true); }}
注: テスト クラスで独自の setUp メソッドを定義する場合は、必ずその中でparent::setUp を呼び出してください。
2.1 JSON API のテスト
<?phpclass ExampleTest extends TestCase{ /** * 基本功能测试示例 * * @return void */ public function testBasicExample() { $this->post('/user', ['name' => 'Sally']) ->seeJson([ 'created' => true, ]); }}seeJson メソッドは、指定された配列を JSON に変換し、JSON 応答全体内の JSON フラグメントを検証します。アプリケーションによって返されます。したがって、JSON 応答に他の属性がある場合でも、指定されたフラグメントが存在する限りテストは合格します。
JSON の一致を正確に検証する
指定された配列とアプリケーションによって返された JSON が正確に一致することを検証したい場合は、seeJsonEquals メソッドを使用します:
<?phpclass ExampleTest extends TestCase{ /** * 基本功能测试示例 * * @return void */ public function testBasicExample() { $this->post('/user', ['name' => 'Sally']) ->seeJsonEquals([ 'created' => true, ]); }}
2.2 認証
<?phpclass ExampleTest extends TestCase{ public function testApplication() { $user = factory('App\User')->create(); $this->actingAs($user) ->get('/user') }}
2.3 カスタム HTTP リクエスト
public function testApplication(){ $response = $this->call('GET', '/'); $this->assertEquals(200, $response->status());}
POST、PUT、または PATCH リクエストを生成します。 入力データ配列はリクエストで渡すことができ、ルートまたはコントローラーの Request インスタンスを通じてリクエスト データにアクセスできます:
$response = $this->call('POST', '/user', ['name' => 'Taylor']);
3. データベースの処理
public function testDatabase(){ // 调用应用... $this->seeInDatabase('users', ['email' => 'sally@foo.com']);}
もちろん、seeInDatabase メソッドなどを使用します。同様のヘルパー関数は便宜上の目的であり、PHPUnit のすべての組み込みアサーション関数を使用してテストを補足することもできます。
3.1 各テスト後にデータベースをリセットする
移行を使用する
1 つの方法は、各テスト後にデータベースをロールバックし、次のテストの前に再度移行することです。 Lumen は、これを自動的に処理するための単純な DatabaseMigrationstrait を提供します。次のようにテスト クラスでこのトレイトを使用するだけです:
<?phpuse Laravel\Lumen\Testing\DatabaseMigrations;use Laravel\Lumen\Testing\DatabaseTransactions;class ExampleTest extends TestCase{ use DatabaseMigrations; /** * A basic functional test example. * * @return void */ public function testBasicExample() { $this->get('/foo'); }}
トランザクションを使用する
別の方法は、各テスト ケースをデータベース トランザクションにラップすることです。Lumen は、それを自動的に処理する便利な DatabaseTransactionstrait を提供します:
<?phpuse Laravel\Lumen\Testing\DatabaseMigrations;use Laravel\Lumen\Testing\DatabaseTransactions;class ExampleTest extends TestCase{ use DatabaseTransactions; /** * A basic functional test example. * * @return void */ public function testBasicExample() { $this->get('/foo'); }}
3.2 Model Factory
$factory->define(App\User::class, function ($faker) { return [ 'name' => $faker->name, 'email' => $faker->email, ];});
クロージャでは、ファクトリ定義として、すべてのプロパティのデフォルトのテスト値を返します。モデル。このクロージャは PHP ライブラリ Faker のインスタンスを受け取り、テスト用にさまざまなタイプのランダム データを簡単に生成できるようにします。
もちろん、ModelFactory.php ファイルにさらにファクトリを追加することもできます。
複数のファクトリ タイプ
同じ Eloquent モデル クラスに対して複数のファクトリを生成したい場合があります。たとえば、通常のユーザーに加えて「admin」ユーザー用のファクトリを生成したい場合は、defineAs を使用できます。メソッド これらのファクトリを定義します:
$factory->defineAs(App\User::class, 'admin', function ($faker) { return [ 'name' => $faker->name, 'email' => $faker->email, 'admin' => true, ];});
基本的なユーザー ファクトリのすべてのプロパティを繰り返すことなく、生のメソッドを使用して基本的なプロパティを取得できます。これらのプロパティを取得した後、必要な追加の値を追加するだけです。
$factory->defineAs(App\User::class, 'admin', function ($faker) use ($factory) { $user = $factory->raw(App\User::class); return array_merge($user, ['admin' => true]);});
测试中使用工厂
定义好工厂后,可以在测试或数据库填充文件中通过全局的 factory方法使用它们来生成模型实例,所以,让我们看一些生成模型的例子,首先,我们使用 make方法,该方法创建模型但不将其保存到数据库:
public function testDatabase(){ $user = factory(App\User::class)->make(); // 用户模型测试...}
如果你想要覆盖模型的一些默认值,可以传递数组值到 make方法。只有指定值被替换,其他数据保持不变:
$user = factory(App\User::class)->make([ 'name' => 'Abigail',]);
还可以创建多个模型集合或者创建给定类型的集合:
// 创建3个 App\User 实例...$users = factory(App\User::class, 3)->make();// 创建1个 App\User "admin" 实例...$user = factory(App\User::class, 'admin')->make();// 创建3个 App\User "admin" 实例...$users = factory(App\User::class, 'admin', 3)->make();
持久化工厂模型
create方法不仅能创建模型实例,还可以使用Eloquent的 save方法将它们保存到数据库:
public function testDatabase(){ $user = factory(App\User::class)->create(); //用户模型测试...}
你仍然可以通过传递数组到create方法覆盖模型上的属性:
$user = factory(App\User::class)->create([ 'name' => 'Abigail',]);
添加关联关系到模型
你甚至可以持久化多个模型到数据库。在本例中,我们添加一个关联到创建的模型,使用 create方法创建多个模型的时候,会返回一个Eloquent集合实例,从而允许你使用集合提供的所有便利方法,例如 each:
$users = factory(App\User::class, 3) ->create() ->each(function($u) { $u->posts()->save(factory(App\Post::class)->make()); });
如果你在重度使用Lumen的事件系统,可能想要在测试时模拟特定事件。例如,如果你在测试用户注册,你可能不想所有 UserRegistered的时间处理器都被触发,因为这可能会发送欢迎邮件,等等。
Lumen提供可一个方便的 expectsEvents方法来验证期望的事件被触发,但同时阻止该事件的其它处理器运行:
<?phpclass ExampleTest extends TestCase{ public function testUserRegistration() { $this->expectsEvents(App\Events\UserRegistered::class); // 测试用户注册代码... }}
如果你想要阻止所有事件运行,可以使用 withoutEvents方法:
<?phpclass ExampleTest extends TestCase{ public function testUserRegistration() { $this->withoutEvents(); // 测试用户注册代码... }}
有时候,你可能想要在请求时简单测试控制器分发的指定任务,这允许你孤立的测试路由/控制器——将其从任务逻辑中分离出去,当然,接下来你可以在一个独立测试类中测试任务本身。
Lumen提供了一个方便的 expectsJobs方法来验证期望的任务被分发,但该任务本身不会被测试:
<?phpclass ExampleTest extends TestCase{ public function testPurchasePodcast() { $this->expectsJobs(App\Jobs\PurchasePodcast::class); // 测试购买播客代码... }}
注意:这个方法只检查通过全局辅助函数dispatch或路由/控制器中调用$this->dispatch分发的任务,并不检查直接通过 Queue::push分发的任务。
测试的时候,你可能经常想要模拟Lumen门面的调用,例如,看看下面的控制器动作:
<?phpnamespace App\Http\Controllers;use Cache;use Illuminate\Routing\Controller;class UserController extends Controller{ /** * 显示应用用户列表 * * @return Response */ public function index() { $value = Cache::get('key'); // }}
我们可以通过使用 shouldReceive方法模拟 Cache门面的调用,该方法返回一个 Mockery模拟的实例,由于门面通过Lumen服务容器解析和管理,它们比通常的静态类更具有可测试性。例如,我们来模拟 Cache门面的调用:
<?phpclass FooTest extends TestCase{ public function testGetIndex() { Cache::shouldReceive('get') ->once() ->with('key') ->andReturn('value'); $this->visit('/users'); }}
注意:不要模拟 Request门面,取而代之地,在测试时传递输入到HTTP辅助函数如 call和 post。