最近几天没有项目,闲的没事干,跟同事开始研究socket,在遇到websocket之前一直好好的,直到在使用websocket时因为没有仔细研究websocket协议,我们使用socket_recv()接收信息,传递过来的值一直是乱码,折腾了半天,也没有搞懂为啥,上网查了半天也不知道为啥,一直以为是编码问题,直到我们老大跟我提了一下,是不是websocket传输过来时数据被重新处理过了,我们一看,果然是这个问题,下面贴出协议内容与解码代码
websocket 协议中的 数据传输帧的格式,图片跟文字来自网络
FIN 1bit 表示信息的最后一帧,flag,也就是标记符
RSV 1-3 1bit each 以后备用的 默认都为 0
Opcode 4bit 帧类型,
Mask 1bit 掩码,是否加密数据,默认必须置为1
Payload len 7bit 数据的长度,当这个7 bit的数据 == 126 时,后面的2 个字节也是表示数 据长度,当它 == 127 时,后面的 8 个字节表示数据长度Masking-key 1 or 4 bit 掩码Payload data playload len bytes 数据
所以我们这里的代码,通过判断 Playload len的值,来用 substr 截取 Masking-key 和 PlayloadData。
根据掩码解析数据的方法是:
/**
* $buffer为从websocket接收到的数据
*/
function decode($buffer) {
$len = $masks = $data = $decoded = null;
$len = ord($buffer[1]) & 127;
if ($len === 126) {
$masks = substr($buffer, 4, 4);
$data = substr($buffer, 8);
}
else if ($len === 127) {
$masks = substr($buffer, 10, 4);
$data = substr($buffer, 14);
}
else {
$masks = substr($buffer, 2, 4);
$data = substr($buffer, 6);
}
//
for ($index = 0; $index < strlen($data); $index++) {
$decoded .= $data[$index] ^ $masks[$index % 4];
}
return $decoded;
}//解码其实我也没看太懂,但是使用起来是没有任何问题的
同理根据websocket协议,我们发送过去的数据也是要经过处理才可以发送
/**
* $s为要发送的数据,返回结果为组装完的数据
*/
function frame($s){
$a = str_split($s, 125);
//添加头文件信息,不然前台无法接受
if (count($a) == 1){
return "\x81" . chr(strlen($a[0])) . $a[0];
}
$ns = "";
foreach ($a as $o){
$ns .= "\x81" . chr(strlen($o)) . $o;
}
return $ns;
}
这样一来websocket服务端数据处理就没有问题了,下面贴出资料来源
http://www.51edu.com/it/bckf/76471.html
http://www.cnblogs.com/smark/archive/2012/11/26/2789812.html