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.

159 lines
4.6 KiB

5 years ago
  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\HttpServer\Router\Router;
  10. use Hyperf\Utils\ApplicationContext;
  11. use Psr\Container\ContainerInterface;
  12. use Psr\Http\Message\ResponseInterface;
  13. use Psr\Http\Server\MiddlewareInterface;
  14. use Psr\Http\Message\ServerRequestInterface;
  15. use Psr\Http\Server\RequestHandlerInterface;
  16. class ApiMiddleware implements MiddlewareInterface
  17. {
  18. /**
  19. * @var ContainerInterface
  20. */
  21. protected $container;
  22. /**
  23. * @var HttpResponse
  24. */
  25. protected $response;
  26. /**
  27. * @var HttpRequest
  28. */
  29. protected $request;
  30. public function __construct(ContainerInterface $container, HttpResponse $response, HttpRequest $request)
  31. {
  32. $this->container = $container;
  33. $this->response = $response;
  34. $this->request = $request;
  35. }
  36. public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
  37. {
  38. // 拦截v1路由
  39. $routePath = $this->request->getPathInfo();
  40. $routePath = explode('/', $routePath);
  41. if ($routePath[1] == 'v1') {
  42. $content = [
  43. "status" => 'ok',
  44. "code" => 99999,
  45. "result" => [],
  46. "message" => '服务已停止升级,稍后请升级更新小程序(如不能自动更新,请删除小程序后重新进入小程序)'
  47. ];
  48. return $this->response->json($content);
  49. }
  50. // 签名校验
  51. # 获取参数
  52. $params = $this->request->all();
  53. # 必须参数,签名、时间戳、随机数
  54. if (!(
  55. isset($params['sign'])
  56. &&isset($params['timestamp'])
  57. &&isset($params['rand'])
  58. ) && env('APP_ENV') == 'prod') {
  59. $content = [
  60. "status" => 'ok',
  61. "code" => 9001,
  62. "result" => [],
  63. "message" => '接口验签失败:缺少参数'
  64. ];
  65. return $this->response->json($content);
  66. }
  67. if (!$this->checkSign($params)) {
  68. $content = [
  69. "status" => 'ok',
  70. "code" => 9002,
  71. "result" => [],
  72. "message" => '接口验签失败:签名错误或已失效'
  73. ];
  74. return $this->response->json($content);
  75. }
  76. $this->request->user = null;
  77. $userToken = $params['user_token'] ?? '';
  78. if ($userToken) {
  79. $ssdb = ApplicationContext::getContainer()->get(SSDBTask::class);
  80. $exists = $ssdb->exec('exists', $userToken);
  81. // TODO 临时进行登录状态续
  82. if (!$exists) {
  83. $ssdb->exec('setnx', $userToken, 1);
  84. $loginExpired = config('auth.user.expire_time');
  85. if (isset($loginExpired) && $loginExpired) {
  86. $ssdb->exec('expire', $hashIds, $loginExpired);
  87. }
  88. } else {
  89. $hashIds = ApplicationContext::getContainer()->get(Hashids::class);
  90. $user = $hashIds->decode($userToken);
  91. $userModel = User::query()->find($user[0]);
  92. $userModel->userToken = $userToken;
  93. $this->request->user = $userModel;
  94. }
  95. }
  96. // 处理全局默认值
  97. $request = \Hyperf\Utils\Context::override(ServerRequestInterface::class, function (ServerRequestInterface $request)
  98. {
  99. $preDatas = $request->getParsedBody();
  100. if (isset($preDatas['market_id']) && $preDatas['market_id'] == -1) {
  101. $preDatas['market_id'] = 1;
  102. }
  103. return $request->withParsedBody($preDatas);
  104. });
  105. return $handler->handle($request);
  106. }
  107. private function checkSign($params)
  108. {
  109. if (env('APP_ENV') != 'prod') {
  110. return true;
  111. }
  112. $sign = $params['sign'];
  113. unset($params['sign']);
  114. $timestamp = $params['timestamp'];
  115. if (empty($sign) || ($timestamp+config('auth.api.sign.expire_time')) < time()) {
  116. return false;
  117. }
  118. return $sign == $this->signature($params);
  119. }
  120. private function signature($params)
  121. {
  122. ksort($params);
  123. $http_query = [];
  124. foreach ($params as $key => $value) {
  125. $http_query[] = $key.'='.$value;
  126. }
  127. return sha1(md5(implode('&', $http_query)).config('auth.api.sign.secret_key'));
  128. }
  129. }