EasySwoole Session 组件

由于在 Swoole 协程下,php 自带的 session 函数是不能使用的。为此,EasySwoole 提供了独立的 session 组件,实现 phpsession 功能。

Session 组件目前最新稳定版本为 3.x。针对 2.x 版本的组件使用文档请看 Session 2.x,其他旧版本的组件使用文档请以 Github 为准。

组件要求

  • php: >=7.1.0
  • easyswoole/spl: ^1.3
  • easyswoole/utility: ^1.1
  • easyswoole/component: ^2.1

安装方法

从框架 3.4.4 版本开始,框架默认自带该组件,不用再次再装,其他版本请使用 composer 安装,安装方法如下所示。

composer require easyswoole/session=3.x

仓库地址

easyswoole/session=3.x

基本使用

注册 session handler

使用 session 前,需要先注册 session handler。接下来的示例使用的 session handlerEasySwoole 内置的 session handler,开箱即用。

注册步骤如下:

  1. 首先我们定义一个 session 工具类继承自 session 组件的 \EasySwoole\EasySwoole\Session 类。用户可以自行定义类继承并实现。下面为提供的一个参考工具类。

新增 App\Tools\Session.php,内容如下:

  1. <?php
  2. namespace App\Tools;
  3. use EasySwoole\Component\Singleton;
  4. class Session extends \EasySwoole\Session\Session
  5. {
  6. use Singleton;
  7. }
  1. 注册 session handler。修改 EasySwoole 全局 event 文件(即框架根目录的 EasySwooleEvent.php 文件),在 mainServerCreate 全局事件和 HTTP 的 全局 HTTP_GLOBAL_ON_REQUEST 事件中注册 session handler

具体实现代码如下:

  1. <?php
  2. namespace EasySwoole\EasySwoole;
  3. use App\Tools\Session;
  4. use EasySwoole\Component\Di;
  5. use EasySwoole\EasySwoole\AbstractInterface\Event;
  6. use EasySwoole\EasySwoole\Swoole\EventRegister;
  7. use EasySwoole\Http\Request;
  8. use EasySwoole\Http\Response;
  9. use EasySwoole\Session\FileSession;
  10. use EasySwoole\Utility\Random;
  11. class EasySwooleEvent implements Event
  12. {
  13. public static function initialize()
  14. {
  15. date_default_timezone_set('Asia/Shanghai');
  16. // 可以自己实现一个标准的 session handler,下面使用组件内置实现的 session handler
  17. // 基于文件存储,传入 EASYSWOOLE_TEMP_DIR . '/Session' 目录作为 session 数据文件存储位置
  18. Session::getInstance(new FileSession(EASYSWOOLE_TEMP_DIR . '/Session'));
  19. Di::getInstance()->set(SysConst::HTTP_GLOBAL_ON_REQUEST, function (Request $request, Response $response) {
  20. // TODO: 注册 HTTP_GLOBAL_ON_REQUEST 回调,相当于原来的 onRequest 事件
  21. // 获取客户端 Cookie 中 easy_session 参数
  22. $sessionId = $request->getCookieParams('easy_session');
  23. if (!$sessionId) {
  24. $sessionId = Random::character(32); // 生成 sessionId
  25. // 设置向客户端响应 Cookie 中 easy_session 参数
  26. $response->setCookie('easy_session', $sessionId);
  27. }
  28. // 存储 sessionId 方便调用,也可以通过其它方式存储
  29. $request->withAttribute('easy_session', $sessionId);
  30. Session::getInstance()->create($sessionId); // 创建并返回该 sessionId 的 context
  31. });
  32. Di::getInstance()->set(SysConst::HTTP_GLOBAL_AFTER_REQUEST, function (Request $request, Response $response) {
  33. // TODO: 注册 HTTP_GLOBAL_AFTER_REQUEST 回调,相当于原来的 afterRequest 事件
  34. // session 数据落地【必不可少这一步】
  35. Session::getInstance()->close($request->getAttribute('easy_session'));
  36. // gc 会清除所有 session,切勿操作
  37. // Session::getInstance()->gc(time());
  38. });
  39. }
  40. public static function mainServerCreate(EventRegister $register)
  41. {
  42. }
  43. }

EasySwoole 中使用 session

注册 session handler 之后,我们就可以在 EasySwoole 控制器 的任意位置使用了。

简单使用示例代码如下:

  1. <?php
  2. namespace App\HttpController;
  3. use EasySwoole\Http\AbstractInterface\Controller;
  4. use EasySwoole\Session\Context;
  5. class Session extends Controller
  6. {
  7. protected function session(): ?Context
  8. {
  9. // 封装一个方法,方便我们快速获取 session context
  10. $sessionId = $this->request()->getAttribute('easy_session');
  11. return \App\Tools\Session::getInstance()->create($sessionId);
  12. }
  13. // 将值保存在 session 中
  14. public function set()
  15. {
  16. // $this->session()->set('key', 'value');
  17. // 把 'test_session_key' 作为键,time() 的值作为值,保存在 session 中
  18. $this->session()->set('test_session_key', time());
  19. // 响应客户端
  20. $this->writeJson(200, 'success!');
  21. }
  22. // 获取 session 中的值
  23. public function get()
  24. {
  25. // $this->session()->get('key');
  26. // 从 session 中获取 key 为 'test_session_key' 的值
  27. $ret = $this->session()->get('test_session_key');
  28. // 响应客户端
  29. $this->writeJson(200, $ret);
  30. }
  31. // 获取 session 中所有数据
  32. public function all()
  33. {
  34. // 获取 session 中所有数据
  35. $ret = $this->session()->allContext();
  36. // 响应客户端
  37. $this->writeJson(200, $ret);
  38. }
  39. // 删除 session 中的值
  40. public function del()
  41. {
  42. // $this->session()->del('key');
  43. // 删除 session 中 key 为 'test_session_key' 的值
  44. $this->session()->del('test_session_key');
  45. // 再次获取 session 中所有数据并响应客户端
  46. $this->writeJson(200, $this->session()->allContext());
  47. }
  48. // 清空 session 中所有数据
  49. public function flush()
  50. {
  51. // 清空 session 中所有数据
  52. $this->session()->flush();
  53. // 再次获取 session 中所有数据并响应客户端
  54. $this->writeJson(200, $this->session()->allContext());
  55. }
  56. // 重新设置(覆盖) session 中的数据
  57. public function setData()
  58. {
  59. // 重新设置(覆盖) session 中的数据
  60. $ret = $this->session()->setData([
  61. 'test_session_key' => 1,
  62. 'test_session_key1' => 2
  63. ]);
  64. // 再次获取 session 中所有数据并响应给客户端
  65. $this->writeJson(200, $ret->allContext());
  66. }
  67. }

然后访问 http://127.0.0.1:9501/session/set (示例请求地址)就可以进行测试设置 session,访问 http://127.0.0.1:9501/session/flush (示例请求地址)就可以清空所有 session 数据。其他示例请用户自行测试。