580 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			580 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | ThinkPHP [ WE CAN DO IT JUST THINK ]
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Copyright (c) 2006~2018 http://thinkphp.cn All rights reserved.
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Author: liu21st <liu21st@gmail.com>
 | ||
| // +----------------------------------------------------------------------
 | ||
| 
 | ||
| namespace think;
 | ||
| 
 | ||
| use think\exception\ClassNotFoundException;
 | ||
| 
 | ||
| class Session
 | ||
| {
 | ||
|     /**
 | ||
|      * 配置参数
 | ||
|      * @var array
 | ||
|      */
 | ||
|     protected $config = [];
 | ||
| 
 | ||
|     /**
 | ||
|      * 前缀
 | ||
|      * @var string
 | ||
|      */
 | ||
|     protected $prefix = '';
 | ||
| 
 | ||
|     /**
 | ||
|      * 是否初始化
 | ||
|      * @var bool
 | ||
|      */
 | ||
|     protected $init = null;
 | ||
| 
 | ||
|     /**
 | ||
|      * 锁驱动
 | ||
|      * @var object
 | ||
|      */
 | ||
|     protected $lockDriver = null;
 | ||
| 
 | ||
|     /**
 | ||
|      * 锁key
 | ||
|      * @var string
 | ||
|      */
 | ||
|     protected $sessKey = 'PHPSESSID';
 | ||
| 
 | ||
|     /**
 | ||
|      * 锁超时时间
 | ||
|      * @var integer
 | ||
|      */
 | ||
|     protected $lockTimeout = 3;
 | ||
| 
 | ||
|     /**
 | ||
|      * 是否启用锁机制
 | ||
|      * @var bool
 | ||
|      */
 | ||
|     protected $lock = false;
 | ||
| 
 | ||
|     public function __construct(array $config = [])
 | ||
|     {
 | ||
|         $this->config = $config;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 设置或者获取session作用域(前缀)
 | ||
|      * @access public
 | ||
|      * @param  string $prefix
 | ||
|      * @return string|void
 | ||
|      */
 | ||
|     public function prefix($prefix = '')
 | ||
|     {
 | ||
|         empty($this->init) && $this->boot();
 | ||
| 
 | ||
|         if (empty($prefix) && null !== $prefix) {
 | ||
|             return $this->prefix;
 | ||
|         } else {
 | ||
|             $this->prefix = $prefix;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     public static function __make(Config $config)
 | ||
|     {
 | ||
|         return new static($config->pull('session'));
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 配置
 | ||
|      * @access public
 | ||
|      * @param  array $config
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function setConfig(array $config = [])
 | ||
|     {
 | ||
|         $this->config = array_merge($this->config, array_change_key_case($config));
 | ||
| 
 | ||
|         if (isset($config['prefix'])) {
 | ||
|             $this->prefix = $config['prefix'];
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['use_lock'])) {
 | ||
|             $this->lock = $config['use_lock'];
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 设置已经初始化
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function inited()
 | ||
|     {
 | ||
|         $this->init = true;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session初始化
 | ||
|      * @access public
 | ||
|      * @param  array $config
 | ||
|      * @return void
 | ||
|      * @throws \think\Exception
 | ||
|      */
 | ||
|     public function init(array $config = [])
 | ||
|     {
 | ||
|         $config = $config ?: $this->config;
 | ||
| 
 | ||
|         $isDoStart = false;
 | ||
|         if (isset($config['use_trans_sid'])) {
 | ||
|             ini_set('session.use_trans_sid', $config['use_trans_sid'] ? 1 : 0);
 | ||
|         }
 | ||
| 
 | ||
|         // 启动session
 | ||
|         if (!empty($config['auto_start']) && PHP_SESSION_ACTIVE != session_status()) {
 | ||
|             ini_set('session.auto_start', 0);
 | ||
|             $isDoStart = true;
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['prefix'])) {
 | ||
|             $this->prefix = $config['prefix'];
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['use_lock'])) {
 | ||
|             $this->lock = $config['use_lock'];
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['var_session_id']) && isset($_REQUEST[$config['var_session_id']])) {
 | ||
|             session_id($_REQUEST[$config['var_session_id']]);
 | ||
|         } elseif (isset($config['id']) && !empty($config['id'])) {
 | ||
|             session_id($config['id']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['name'])) {
 | ||
|             session_name($config['name']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['path'])) {
 | ||
|             session_save_path($config['path']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['domain'])) {
 | ||
|             ini_set('session.cookie_domain', $config['domain']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['expire'])) {
 | ||
|             ini_set('session.gc_maxlifetime', $config['expire']);
 | ||
|             ini_set('session.cookie_lifetime', $config['expire']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['secure'])) {
 | ||
|             ini_set('session.cookie_secure', $config['secure']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['httponly'])) {
 | ||
|             ini_set('session.cookie_httponly', $config['httponly']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['use_cookies'])) {
 | ||
|             ini_set('session.use_cookies', $config['use_cookies'] ? 1 : 0);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['cache_limiter'])) {
 | ||
|             session_cache_limiter($config['cache_limiter']);
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['cache_expire'])) {
 | ||
|             session_cache_expire($config['cache_expire']);
 | ||
|         }
 | ||
| 
 | ||
|         if (!empty($config['type'])) {
 | ||
|             // 读取session驱动
 | ||
|             $class = false !== strpos($config['type'], '\\') ? $config['type'] : '\\think\\session\\driver\\' . ucwords($config['type']);
 | ||
| 
 | ||
|             // 检查驱动类
 | ||
|             if (!class_exists($class) || !session_set_save_handler(new $class($config))) {
 | ||
|                 throw new ClassNotFoundException('error session handler:' . $class, $class);
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         if ($isDoStart) {
 | ||
|             $this->start();
 | ||
|         } else {
 | ||
|             $this->init = false;
 | ||
|         }
 | ||
| 
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session自动启动或者初始化
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function boot()
 | ||
|     {
 | ||
|         if (is_null($this->init)) {
 | ||
|             $this->init();
 | ||
|         }
 | ||
| 
 | ||
|         if (false === $this->init) {
 | ||
|             if (PHP_SESSION_ACTIVE != session_status()) {
 | ||
|                 $this->start();
 | ||
|             }
 | ||
|             $this->init = true;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session设置
 | ||
|      * @access public
 | ||
|      * @param  string        $name session名称
 | ||
|      * @param  mixed         $value session值
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function set($name, $value, $prefix = null)
 | ||
|     {
 | ||
|         $this->lock();
 | ||
| 
 | ||
|         empty($this->init) && $this->boot();
 | ||
| 
 | ||
|         $prefix = !is_null($prefix) ? $prefix : $this->prefix;
 | ||
| 
 | ||
|         if (strpos($name, '.')) {
 | ||
|             // 二维数组赋值
 | ||
|             list($name1, $name2) = explode('.', $name);
 | ||
|             if ($prefix) {
 | ||
|                 $_SESSION[$prefix][$name1][$name2] = $value;
 | ||
|             } else {
 | ||
|                 $_SESSION[$name1][$name2] = $value;
 | ||
|             }
 | ||
|         } elseif ($prefix) {
 | ||
|             $_SESSION[$prefix][$name] = $value;
 | ||
|         } else {
 | ||
|             $_SESSION[$name] = $value;
 | ||
|         }
 | ||
| 
 | ||
|         $this->unlock();
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session获取
 | ||
|      * @access public
 | ||
|      * @param  string        $name session名称
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return mixed
 | ||
|      */
 | ||
|     public function get($name = '', $prefix = null)
 | ||
|     {
 | ||
|         $this->lock();
 | ||
| 
 | ||
|         empty($this->init) && $this->boot();
 | ||
| 
 | ||
|         $prefix = !is_null($prefix) ? $prefix : $this->prefix;
 | ||
| 
 | ||
|         $value = $prefix ? (!empty($_SESSION[$prefix]) ? $_SESSION[$prefix] : []) : $_SESSION;
 | ||
| 
 | ||
|         if ('' != $name) {
 | ||
|             $name = explode('.', $name);
 | ||
| 
 | ||
|             foreach ($name as $val) {
 | ||
|                 if (isset($value[$val])) {
 | ||
|                     $value = $value[$val];
 | ||
|                 } else {
 | ||
|                     $value = null;
 | ||
|                     break;
 | ||
|                 }
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         $this->unlock();
 | ||
| 
 | ||
|         return $value;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session 读写锁驱动实例化
 | ||
|      */
 | ||
|     protected function initDriver()
 | ||
|     {
 | ||
|         $config = $this->config;
 | ||
| 
 | ||
|         if (!empty($config['type']) && isset($config['use_lock']) && $config['use_lock']) {
 | ||
|             // 读取session驱动
 | ||
|             $class = false !== strpos($config['type'], '\\') ? $config['type'] : '\\think\\session\\driver\\' . ucwords($config['type']);
 | ||
| 
 | ||
|             // 检查驱动类及类中是否存在 lock 和 unlock 函数
 | ||
|             if (class_exists($class) && method_exists($class, 'lock') && method_exists($class, 'unlock')) {
 | ||
|                 $this->lockDriver = new $class($config);
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         // 通过cookie获得session_id
 | ||
|         if (isset($config['name']) && $config['name']) {
 | ||
|             $this->sessKey = $config['name'];
 | ||
|         }
 | ||
| 
 | ||
|         if (isset($config['lock_timeout']) && $config['lock_timeout'] > 0) {
 | ||
|             $this->lockTimeout = $config['lock_timeout'];
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session 读写加锁
 | ||
|      * @access protected
 | ||
|      * @return void
 | ||
|      */
 | ||
|     protected function lock()
 | ||
|     {
 | ||
|         if (empty($this->lock)) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $this->initDriver();
 | ||
| 
 | ||
|         if (null !== $this->lockDriver && method_exists($this->lockDriver, 'lock')) {
 | ||
|             $t = time();
 | ||
|             // 使用 session_id 作为互斥条件,即只对同一 session_id 的会话互斥。第一次请求没有 session_id
 | ||
|             $sessID = isset($_COOKIE[$this->sessKey]) ? $_COOKIE[$this->sessKey] : '';
 | ||
| 
 | ||
|             do {
 | ||
|                 if (time() - $t > $this->lockTimeout) {
 | ||
|                     $this->unlock();
 | ||
|                 }
 | ||
|             } while (!$this->lockDriver->lock($sessID, $this->lockTimeout));
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session 读写解锁
 | ||
|      * @access protected
 | ||
|      * @return void
 | ||
|      */
 | ||
|     protected function unlock()
 | ||
|     {
 | ||
|         if (empty($this->lock)) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $this->pause();
 | ||
| 
 | ||
|         if ($this->lockDriver && method_exists($this->lockDriver, 'unlock')) {
 | ||
|             $sessID = isset($_COOKIE[$this->sessKey]) ? $_COOKIE[$this->sessKey] : '';
 | ||
|             $this->lockDriver->unlock($sessID);
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session获取并删除
 | ||
|      * @access public
 | ||
|      * @param  string        $name session名称
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return mixed
 | ||
|      */
 | ||
|     public function pull($name, $prefix = null)
 | ||
|     {
 | ||
|         $result = $this->get($name, $prefix);
 | ||
| 
 | ||
|         if ($result) {
 | ||
|             $this->delete($name, $prefix);
 | ||
|             return $result;
 | ||
|         } else {
 | ||
|             return;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * session设置 下一次请求有效
 | ||
|      * @access public
 | ||
|      * @param  string        $name session名称
 | ||
|      * @param  mixed         $value session值
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function flash($name, $value)
 | ||
|     {
 | ||
|         $this->set($name, $value);
 | ||
| 
 | ||
|         if (!$this->has('__flash__.__time__')) {
 | ||
|             $this->set('__flash__.__time__', $_SERVER['REQUEST_TIME_FLOAT']);
 | ||
|         }
 | ||
| 
 | ||
|         $this->push('__flash__', $name);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 清空当前请求的session数据
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function flush()
 | ||
|     {
 | ||
|         if (!$this->init) {
 | ||
|             return;
 | ||
|         }
 | ||
| 
 | ||
|         $item = $this->get('__flash__');
 | ||
| 
 | ||
|         if (!empty($item)) {
 | ||
|             $time = $item['__time__'];
 | ||
| 
 | ||
|             if ($_SERVER['REQUEST_TIME_FLOAT'] > $time) {
 | ||
|                 unset($item['__time__']);
 | ||
|                 $this->delete($item);
 | ||
|                 $this->set('__flash__', []);
 | ||
|             }
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 删除session数据
 | ||
|      * @access public
 | ||
|      * @param  string|array  $name session名称
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function delete($name, $prefix = null)
 | ||
|     {
 | ||
|         empty($this->init) && $this->boot();
 | ||
| 
 | ||
|         $prefix = !is_null($prefix) ? $prefix : $this->prefix;
 | ||
| 
 | ||
|         if (is_array($name)) {
 | ||
|             foreach ($name as $key) {
 | ||
|                 $this->delete($key, $prefix);
 | ||
|             }
 | ||
|         } elseif (strpos($name, '.')) {
 | ||
|             list($name1, $name2) = explode('.', $name);
 | ||
|             if ($prefix) {
 | ||
|                 unset($_SESSION[$prefix][$name1][$name2]);
 | ||
|             } else {
 | ||
|                 unset($_SESSION[$name1][$name2]);
 | ||
|             }
 | ||
|         } else {
 | ||
|             if ($prefix) {
 | ||
|                 unset($_SESSION[$prefix][$name]);
 | ||
|             } else {
 | ||
|                 unset($_SESSION[$name]);
 | ||
|             }
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 清空session数据
 | ||
|      * @access public
 | ||
|      * @param  string|null   $prefix 作用域(前缀)
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function clear($prefix = null)
 | ||
|     {
 | ||
|         empty($this->init) && $this->boot();
 | ||
|         $prefix = !is_null($prefix) ? $prefix : $this->prefix;
 | ||
| 
 | ||
|         if ($prefix) {
 | ||
|             unset($_SESSION[$prefix]);
 | ||
|         } else {
 | ||
|             $_SESSION = [];
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 判断session数据
 | ||
|      * @access public
 | ||
|      * @param  string        $name session名称
 | ||
|      * @param  string|null   $prefix
 | ||
|      * @return bool
 | ||
|      */
 | ||
|     public function has($name, $prefix = null)
 | ||
|     {
 | ||
|         empty($this->init) && $this->boot();
 | ||
| 
 | ||
|         $prefix = !is_null($prefix) ? $prefix : $this->prefix;
 | ||
|         $value  = $prefix ? (!empty($_SESSION[$prefix]) ? $_SESSION[$prefix] : []) : $_SESSION;
 | ||
| 
 | ||
|         $name = explode('.', $name);
 | ||
| 
 | ||
|         foreach ($name as $val) {
 | ||
|             if (!isset($value[$val])) {
 | ||
|                 return false;
 | ||
|             } else {
 | ||
|                 $value = $value[$val];
 | ||
|             }
 | ||
|         }
 | ||
| 
 | ||
|         return true;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 添加数据到一个session数组
 | ||
|      * @access public
 | ||
|      * @param  string  $key
 | ||
|      * @param  mixed   $value
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function push($key, $value)
 | ||
|     {
 | ||
|         $array = $this->get($key);
 | ||
| 
 | ||
|         if (is_null($array)) {
 | ||
|             $array = [];
 | ||
|         }
 | ||
| 
 | ||
|         $array[] = $value;
 | ||
| 
 | ||
|         $this->set($key, $array);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 启动session
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function start()
 | ||
|     {
 | ||
|         session_start();
 | ||
| 
 | ||
|         $this->init = true;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 销毁session
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function destroy()
 | ||
|     {
 | ||
|         if (!empty($_SESSION)) {
 | ||
|             $_SESSION = [];
 | ||
|         }
 | ||
| 
 | ||
|         session_unset();
 | ||
|         session_destroy();
 | ||
| 
 | ||
|         $this->init       = null;
 | ||
|         $this->lockDriver = null;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 重新生成session_id
 | ||
|      * @access public
 | ||
|      * @param  bool $delete 是否删除关联会话文件
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function regenerate($delete = false)
 | ||
|     {
 | ||
|         session_regenerate_id($delete);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 暂停session
 | ||
|      * @access public
 | ||
|      * @return void
 | ||
|      */
 | ||
|     public function pause()
 | ||
|     {
 | ||
|         // 暂停session
 | ||
|         session_write_close();
 | ||
|         $this->init = false;
 | ||
|     }
 | ||
| }
 |