自动化测试

在 Hyperf 里测试默认通过 phpunit 来实现,但由于 Hyperf 是一个协程框架,所以默认的 phpunit 并不能很好的工作,因此我们提供了一个 co-phpunit 脚本来进行适配,您可直接调用脚本或者使用对应的 composer 命令来运行。自动化测试没有特定的组件,但是在 Hyperf 提供的骨架包里都会有对应实现。

  1. composer require hyperf/testing
  1. "scripts": {
  2. "test": "./test/co-phpunit -c phpunit.xml --colors=always"
  3. },

Bootstrap

Hyperf 提供了默认的 bootstrap.php 文件,它让用户在运行单元测试时,扫描并加载对应的库到内存里。

  1. <?php
  2. declare(strict_types=1);
  3. error_reporting(E_ALL);
  4. ! defined('BASE_PATH') && define('BASE_PATH', dirname(__DIR__, 1));
  5. \Swoole\Runtime::enableCoroutine(true);
  6. require BASE_PATH . '/vendor/autoload.php';
  7. require BASE_PATH . '/config/container.php';

当用户修改的代码需要重新生成代理类时,需要主动运行一下脚本。因为你单元测试运行时,并不会重置代理类。

  1. # 重新生成代理类
  2. php bin/hyperf.php di:init-proxy
  3. # 运行单元测试
  4. composer test

模拟 HTTP 请求

在开发接口时,我们通常需要一段自动化测试脚本来保证我们提供的接口按预期在运行,Hyperf 框架下提供了 Hyperf\Testing\Client 类,可以让您在不启动 Server 的情况下,模拟 HTTP 服务的请求:

  1. <?php
  2. use Hyperf\Testing\Client;
  3. $client = make(Client::class);
  4. $result = $client->get('/');

因为 Hyperf 支持多端口配置,除了验证默认的端口接口外,如果验证其他端口的接口呢?

  1. <?php
  2. use Hyperf\Testing\Client;
  3. $client = make(Client::class,['server' => 'adminHttp']);
  4. $result = $client->json('/user/0',[
  5. 'nickname' => 'Hyperf'
  6. ]);

示例

让我们写个小 DEMO 来测试一下。

  1. <?php
  2. declare(strict_types=1);
  3. namespace HyperfTest\Cases;
  4. use Hyperf\Testing\Client;
  5. use PHPUnit\Framework\TestCase;
  6. /**
  7. * @internal
  8. * @coversNothing
  9. */
  10. class ExampleTest extends TestCase
  11. {
  12. /**
  13. * @var Client
  14. */
  15. protected $client;
  16. public function __construct($name = null, array $data = [], $dataName = '')
  17. {
  18. parent::__construct($name, $data, $dataName);
  19. $this->client = make(Client::class);
  20. }
  21. public function testExample()
  22. {
  23. $this->assertTrue(true);
  24. $res = $this->client->get('/');
  25. $this->assertSame(0, $res['code']);
  26. $this->assertSame('Hello Hyperf.', $res['data']['message']);
  27. $this->assertSame('GET', $res['data']['method']);
  28. $this->assertSame('Hyperf', $res['data']['user']);
  29. $res = $this->client->get('/', ['user' => 'developer']);
  30. $this->assertSame(0, $res['code']);
  31. $this->assertSame('developer', $res['data']['user']);
  32. $res = $this->client->post('/', [
  33. 'user' => 'developer',
  34. ]);
  35. $this->assertSame('Hello Hyperf.', $res['data']['message']);
  36. $this->assertSame('POST', $res['data']['method']);
  37. $this->assertSame('developer', $res['data']['user']);
  38. $res = $this->client->json('/', [
  39. 'user' => 'developer',
  40. ]);
  41. $this->assertSame('Hello Hyperf.', $res['data']['message']);
  42. $this->assertSame('POST', $res['data']['method']);
  43. $this->assertSame('developer', $res['data']['user']);
  44. $res = $this->client->file('/', ['name' => 'file', 'file' => BASE_PATH . '/README.md']);
  45. $this->assertSame('Hello Hyperf.', $res['data']['message']);
  46. $this->assertSame('POST', $res['data']['method']);
  47. $this->assertSame('README.md', $res['data']['file']);
  48. }
  49. }

调试代码

在FPM场景下,我们通常改完代码,然后打开浏览器访问对应接口,所以我们通常会需要两个函数 dddump,但 Hyperf 跑在 CLI 模式下,就算提供了这两个函数,也需要在 CLI 中重启 Server,然后再到浏览器中调用对应接口查看结果。这样其实并没有简化流程,反而更麻烦了。

接下来,我来介绍如何通过配合 testing,来快速调试代码,顺便完成单元测试。

假设我们在 UserDao 中实现了一个查询用户信息的函数

  1. namespace App\Service\Dao;
  2. use App\Constants\ErrorCode;
  3. use App\Exception\BusinessException;
  4. use App\Model\User;
  5. class UserDao extends Dao
  6. {
  7. /**
  8. * @param $id
  9. * @param bool $throw
  10. * @return
  11. */
  12. public function first($id, $throw = true)
  13. {
  14. $model = User::query()->find($id);
  15. if ($throw && empty($model)) {
  16. throw new BusinessException(ErrorCode::USRE_NOT_EXIST);
  17. }
  18. return $model;
  19. }
  20. }

那我们编写对应的单元测试

  1. namespace HyperfTest\Cases;
  2. use HyperfTest\HttpTestCase;
  3. use App\Service\Dao\UserDao;
  4. /**
  5. * @internal
  6. * @coversNothing
  7. */
  8. class UserTest extends HttpTestCase
  9. {
  10. public function testUserDaoFirst()
  11. {
  12. $model = \Hyperf\Utils\ApplicationContext::getContainer()->get(UserDao::class)->first(1);
  13. var_dump($model);
  14. $this->assertSame(1, $model->id);
  15. }
  16. }

然后执行我们的单测

  1. composer test -- --filter=testUserDaoFirst