1、Http类设计
workerman作者在设计Http类时,还引入了另外一个类HttpCache,用于辅助设置Http的报头、状态码、会话等操作。HttpCache类是在载入Http类所在的类文件时,进行调用。调用方法为: HttpCache::init()
注意,此方法仅在首次载入Http类文件时,进行调用。
2、HttpCache辅助类
2.1、类概要
2.2、核心方法解析
init()方法的作用是把php.ini配置文件中关于session的配置参数值,赋值给HttpCache类的静态属性如:
session名称、session文件保存路径、sessiongc设置、session最大生存时间
3、Http类
在实现上,并不implements协议接口,只是按照其接口规范进行实现
3.1、类概要设计
3.2、核心方法解析
- input静态方法
判断是否存在 \\r\\n\\r\\n 报头与报体分隔符,不存在则返回0,说明接收数据还不够。但如果接收的长度大于一条链接的最大包长度,则认为是非法,因关闭此链接
存在分隔符,则进行分割,获取报头的信息
从报头信息中,获取当前的请求行中的请求方法,执行getRequestSize方法
getRequestSize方法进行如下判断:
当请求方法为GET/OPTIONS/HEAD/DELETE这些方法时,则直接计算分割后的报头长度+报头与报体分隔符
当请求方法为POST,则执行正则匹配获取Content-Length的值,然后计算Content-Length的值+分割后的报头长度+报头与报体分隔符
- decode静态方法
- 重置php中的关于http请求的全局变量如
$_POST、$_GET、$_COOKIE、$_REQUEST、$_SESSION、$_FILES、$GLOBALS['HTTP_RAW_POST_DATA']
- 实例化HttpCache()对象,并赋值给其静态属性$instance
- 初始化全局$_SERVER变量
- 以分隔符 \\r\\n\\r\\n分割http的报头$http_header与报体$http_body
- 以分隔符 \\r\\n分割报头$http_header,得到数组$header_data
- $header_data[0]为请求行,再以" " 进行分割,得到请求方法、请求url、请求协议,并依次赋值给$_SERVER全局变量
- 然后unset($header_data[0]);再遍历$header_data,得到头部信息键值对,并保存在$_SERVER全局变量
- 头部信息CONTENT_TYPE的判断:
-- 为multipart/form-data,这说明此请求是上传资源
则执行parseUploadFiles($http_body, $http_post_boundary);
上传资源的报文格式
------WebKitFormBoundary9BTgq0bt94hMtm23\\r\\nContent-Disposition: form-data; name="data"; filename="blob"\\r\\nContent-Type: application/octet-stream\\r\\n\\r\\n资源\\r\\n------WebKitFormBoundary9BTgq0bt94hMtm23\\r\\nContent-Disposition: form-data; name="index"\\r\\n\\r\\n1\\r\\n------WebKitFormBoundary9BTgq0bt94hMtm23--\\r\\n
-- application/json
$_POST = json_decode($http_body, true);
-- application/x-www-form-urlencoded
parse_str($http_body, $_POST);
- encode静态方法
- 按照http报文的响应格式遍历HttpCache::$header数组
- 执行sessionWriteClose方法
判断是否启动session和全局变量$_SESSION是否非空
如都满足,则$_SESSION进行序列化处理,并把序列化结果后保存在本地
- header
设置响应报头信息。借用辅助类HttpCache的静态属性$header数组,存在报头key-value信息
4、如何启动session
由于workerman是基于cli模式,因此,原来php的session_start已无法使用。但通过Http类的sessionStart()静态方法,也能达到同样的效果。
4.1、触发
在onMessage回调方法中,进行调用Http::sessionStart()
4.2、sessionStart()执行流程
-- tryGcSessions()
根据session文件的创建时间进行判断,如果大于session的生存时间,则进行删除
-- 设置session启动标志位
-- 如果本次全局变量$_COOKIE中不存在sessionid的值和session文件不存在,则创建新的sessionid和session文件
-- 然后设置头部信息cookie值,此操作在返回信息给客户端时,encode方法中得到体现
-- 这样下次,再执行sessionStart()静态方法,就可以从cookie中获取本地session文件名,和获取里面的内容
閱讀更多 iamasb 的文章