Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. <?php
  2. namespace crmeb\utils;
  3. use crmeb\exceptions\AdminException;
  4. use crmeb\services\CacheService;
  5. use Firebase\JWT\JWT;
  6. use think\facade\Env;
  7. /**
  8. * Jwt
  9. * Class JwtAuth
  10. */
  11. class JwtAuth
  12. {
  13. /**
  14. * token
  15. * @var string
  16. */
  17. protected $token;
  18. /**
  19. * @var string
  20. */
  21. protected $app_key = 'crmeb_app_key';
  22. /**
  23. * 获取token
  24. * @param int $id
  25. * @param string $type
  26. * @param array $params
  27. * @return array
  28. */
  29. public function getToken(int $id, string $type, array $params = []): array
  30. {
  31. $host = app()->request->host();
  32. $time = time();
  33. $exp_time = strtotime('+ 7day');
  34. if (app()->request->isApp()) {
  35. $exp_time = strtotime('+ 30day');
  36. }
  37. if ($type == 'out') {
  38. $exp_time = strtotime('+ 1day');
  39. }
  40. $params += [
  41. 'iss' => $host,
  42. 'aud' => $host,
  43. 'iat' => $time,
  44. 'nbf' => $time,
  45. 'exp' => $exp_time,
  46. ];
  47. $params['jti'] = compact('id', 'type');
  48. $token = JWT::encode($params, Env::get('app.app_key', $this->app_key) ?: $this->app_key);
  49. return compact('token', 'params');
  50. }
  51. /**
  52. * 解析token
  53. * @param string $jwt
  54. * @return array
  55. */
  56. public function parseToken(string $jwt): array
  57. {
  58. $this->token = $jwt;
  59. list($headb64, $bodyb64, $cryptob64) = explode('.', $this->token);
  60. $payload = JWT::jsonDecode(JWT::urlsafeB64Decode($bodyb64));
  61. return [$payload->jti->id, $payload->jti->type];
  62. }
  63. /**
  64. * 验证token
  65. */
  66. public function verifyToken()
  67. {
  68. JWT::$leeway = 60;
  69. JWT::decode($this->token, Env::get('app.app_key', $this->app_key) ?: $this->app_key, array('HS256'));
  70. $this->token = null;
  71. }
  72. /**
  73. * 获取token并放入令牌桶
  74. * @param int $id
  75. * @param string $type
  76. * @param array $params
  77. * @return array
  78. */
  79. public function createToken(int $id, string $type, array $params = [])
  80. {
  81. $tokenInfo = $this->getToken($id, $type, $params);
  82. $exp = $tokenInfo['params']['exp'] - $tokenInfo['params']['iat'] + 60;
  83. $res = CacheService::setTokenBucket(md5($tokenInfo['token']), ['uid' => $id, 'type' => $type, 'token' => $tokenInfo['token'], 'exp' => $exp], (int)$exp, $type);
  84. if (!$res) {
  85. throw new AdminException(ApiErrorCode::ERR_SAVE_TOKEN);
  86. }
  87. return $tokenInfo;
  88. }
  89. }