You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

134 lines
3.7 KiB

  1. <?php
  2. declare(strict_types=1);
  3. namespace App\Middleware\Auth;
  4. use App\Model\v3\User;
  5. use App\TaskWorker\SSDBTask;
  6. use Hashids\Hashids;
  7. use Hyperf\HttpServer\Contract\RequestInterface as HttpRequest;
  8. use Hyperf\HttpServer\Contract\ResponseInterface as HttpResponse;
  9. use Hyperf\Utils\ApplicationContext;
  10. use Psr\Container\ContainerInterface;
  11. use Psr\Http\Message\ResponseInterface;
  12. use Psr\Http\Server\MiddlewareInterface;
  13. use Psr\Http\Message\ServerRequestInterface;
  14. use Psr\Http\Server\RequestHandlerInterface;
  15. class ApiMiddleware implements MiddlewareInterface
  16. {
  17. /**
  18. * @var ContainerInterface
  19. */
  20. protected $container;
  21. /**
  22. * @var HttpResponse
  23. */
  24. protected $response;
  25. /**
  26. * @var HttpRequest
  27. */
  28. protected $request;
  29. public function __construct(ContainerInterface $container, HttpResponse $response, HttpRequest $request)
  30. {
  31. $this->container = $container;
  32. $this->response = $response;
  33. $this->request = $request;
  34. }
  35. public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
  36. {
  37. // 签名校验
  38. # 获取参数
  39. $params = $this->request->all();
  40. # 必须参数,签名、时间戳、随机数
  41. if (!(
  42. isset($params['sign'])
  43. &&isset($params['timestamp'])
  44. &&isset($params['rand'])
  45. ) && env('APP_ENV') == 'prod') {
  46. $content = [
  47. "status" => 'ok',
  48. "code" => 9001,
  49. "result" => [],
  50. "message" => '接口验签失败:缺少参数'
  51. ];
  52. return $this->response->json($content);
  53. }
  54. if (!$this->checkSign($params)) {
  55. $content = [
  56. "status" => 'ok',
  57. "code" => 9002,
  58. "result" => [],
  59. "message" => '接口验签失败:签名错误或已失效'
  60. ];
  61. return $this->response->json($content);
  62. }
  63. $this->request->user = null;
  64. $userToken = $params['user_token'] ?? '';
  65. if ($userToken) {
  66. $ssdb = ApplicationContext::getContainer()->get(SSDBTask::class);
  67. $exists = $ssdb->exec('exists', $userToken);
  68. if ($exists) {
  69. $hashIds = ApplicationContext::getContainer()->get(Hashids::class);
  70. $user = $hashIds->decode($userToken);
  71. $userModel = User::query()->find($user[0]);
  72. $userModel->userToken = $userToken;
  73. $this->request->user = $userModel;
  74. }
  75. }
  76. // 处理全局默认值
  77. $request = \Hyperf\Utils\Context::override(ServerRequestInterface::class, function (ServerRequestInterface $request)
  78. {
  79. $preDatas = $request->getParsedBody();
  80. if (isset($preDatas['market_id']) && $preDatas['market_id'] == -1) {
  81. $preDatas['market_id'] = 1;
  82. }
  83. return $request->withParsedBody($preDatas);
  84. });
  85. return $handler->handle($request);
  86. }
  87. private function checkSign($params)
  88. {
  89. if (env('APP_ENV') != 'prod') {
  90. return true;
  91. }
  92. $sign = $params['sign'];
  93. unset($params['sign']);
  94. $timestamp = $params['timestamp'];
  95. if (empty($sign) || ($timestamp+config('auth.api.sign.expire_time')) < time()) {
  96. return false;
  97. }
  98. return $sign == $this->signature($params);
  99. }
  100. private function signature($params)
  101. {
  102. ksort($params);
  103. $http_query = [];
  104. foreach ($params as $key => $value) {
  105. $http_query[] = $key.'='.$value;
  106. }
  107. return sha1(md5(implode('&', $http_query)).config('auth.api.sign.secret_key'));
  108. }
  109. }