| 
									
										
										
										
											2024-11-04 10:49:10 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace app\common\library\token\driver; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use app\common\library\token\Driver; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Token操作类 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class Redis extends Driver | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     protected $options = [ | 
					
						
							|  |  |  |         'host'        => '127.0.0.1', | 
					
						
							|  |  |  |         'port'        => 6379, | 
					
						
							|  |  |  |         'password'    => '', | 
					
						
							|  |  |  |         'select'      => 0, | 
					
						
							|  |  |  |         'timeout'     => 0, | 
					
						
							|  |  |  |         'expire'      => 0, | 
					
						
							|  |  |  |         'persistent'  => false, | 
					
						
							|  |  |  |         'userprefix'  => 'up:', | 
					
						
							|  |  |  |         'tokenprefix' => 'tp:', | 
					
						
							|  |  |  |     ]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 构造函数 | 
					
						
							|  |  |  |      * @param array $options 缓存参数 | 
					
						
							|  |  |  |      * @throws \BadFunctionCallException | 
					
						
							|  |  |  |      * @access public | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct($options = []) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!extension_loaded('redis')) { | 
					
						
							|  |  |  |             throw new \BadFunctionCallException('not support: redis'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!empty($options)) { | 
					
						
							|  |  |  |             $this->options = array_merge($this->options, $options); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $this->handler = new \Redis; | 
					
						
							|  |  |  |         if ($this->options['persistent']) { | 
					
						
							|  |  |  |             $this->handler->pconnect($this->options['host'], $this->options['port'], $this->options['timeout'], 'persistent_id_' . $this->options['select']); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $this->handler->connect($this->options['host'], $this->options['port'], $this->options['timeout']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ('' != $this->options['password']) { | 
					
						
							|  |  |  |             $this->handler->auth($this->options['password']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (0 != $this->options['select']) { | 
					
						
							|  |  |  |             $this->handler->select($this->options['select']); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 获取加密后的Token | 
					
						
							|  |  |  |      * @param string $token Token标识 | 
					
						
							|  |  |  |      * @return string | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getEncryptedToken($token) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $config = \think\Config::get('token'); | 
					
						
							|  |  |  |         $token = $token ?? ''; // 为兼容 php8
 | 
					
						
							|  |  |  |         return $this->options['tokenprefix'] . hash_hmac($config['hashalgo'], $token, $config['key']); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 获取会员的key | 
					
						
							|  |  |  |      * @param $user_id | 
					
						
							|  |  |  |      * @return string | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getUserKey($user_id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return $this->options['userprefix'] . $user_id; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 存储Token | 
					
						
							|  |  |  |      * @param   string $token   Token | 
					
						
							|  |  |  |      * @param   int    $user_id 会员ID | 
					
						
							|  |  |  |      * @param   int    $expire  过期时长,0表示无限,单位秒 | 
					
						
							|  |  |  |      * @return bool | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function set($token, $user_id, $expire = 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (is_null($expire)) { | 
					
						
							|  |  |  |             $expire = $this->options['expire']; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if ($expire instanceof \DateTime) { | 
					
						
							|  |  |  |             $expire = $expire->getTimestamp() - time(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $key = $this->getEncryptedToken($token); | 
					
						
							|  |  |  |         if ($expire) { | 
					
						
							|  |  |  |             $result = $this->handler->setex($key, $expire, $user_id); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             $result = $this->handler->set($key, $user_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         //写入会员关联的token
 | 
					
						
							|  |  |  |         $this->handler->sAdd($this->getUserKey($user_id), $key); | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 获取Token内的信息 | 
					
						
							|  |  |  |      * @param   string $token | 
					
						
							|  |  |  |      * @return  array | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function get($token) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $key = $this->getEncryptedToken($token); | 
					
						
							|  |  |  |         $value = $this->handler->get($key); | 
					
						
							|  |  |  |         if (is_null($value) || false === $value) { | 
					
						
							|  |  |  |             return []; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         //获取有效期
 | 
					
						
							|  |  |  |         $expire = $this->handler->ttl($key); | 
					
						
							|  |  |  |         $expire = $expire < 0 ? 365 * 86400 : $expire; | 
					
						
							|  |  |  |         $expiretime = time() + $expire; | 
					
						
							|  |  |  |         //解决使用redis方式储存token时api接口Token刷新与检测因expires_in拼写错误报错的BUG
 | 
					
						
							| 
									
										
										
										
											2025-01-16 18:00:46 +08:00
										 |  |  |         $result = ['token' => $token, 'user_id' => $value,'id' => $value, 'expiretime' => $expiretime, 'expires_in' => $expire]; | 
					
						
							| 
									
										
										
										
											2024-11-04 10:49:10 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $result; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 判断Token是否可用 | 
					
						
							|  |  |  |      * @param   string $token   Token | 
					
						
							|  |  |  |      * @param   int    $user_id 会员ID | 
					
						
							|  |  |  |      * @return  boolean | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function check($token, $user_id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $data = self::get($token); | 
					
						
							|  |  |  |         return $data && $data['user_id'] == $user_id ? true : false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 删除Token | 
					
						
							|  |  |  |      * @param   string $token | 
					
						
							|  |  |  |      * @return  boolean | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function delete($token) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $data = $this->get($token); | 
					
						
							|  |  |  |         if ($data) { | 
					
						
							|  |  |  |             $key = $this->getEncryptedToken($token); | 
					
						
							|  |  |  |             $user_id = $data['user_id']; | 
					
						
							|  |  |  |             $this->handler->del($key); | 
					
						
							|  |  |  |             $this->handler->sRem($this->getUserKey($user_id), $key); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * 删除指定用户的所有Token | 
					
						
							|  |  |  |      * @param   int $user_id | 
					
						
							|  |  |  |      * @return  boolean | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function clear($user_id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $keys = $this->handler->sMembers($this->getUserKey($user_id)); | 
					
						
							|  |  |  |         $this->handler->del($this->getUserKey($user_id)); | 
					
						
							|  |  |  |         $this->handler->del($keys); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |