php websocket编程之握手

php websocket编程之握手

 这两天在研究websocekt技术,看到了一些很棒的类库。原本打算在直接研究workerman的,后来想想,websocket的基础还没怎么去理解呢,直接搞那个不太好,先研究一下自己怎么去写一个简单的socket服务器。

  WebSocket protocol 是HTML5一种新的协议。它实现了浏览器与服务器全双工通信(full-duplex)。在这个协议提出之前,我想大家去做一些类似于IM这样的应用的时候,由于http是短连接性质的,大家肯定会用ajax轮询的方式去定时的请求服务器,以获得数据。当然这样的缺点也是很明显的。HTTP request 的header是非常长的,里面包含的有用数据可能只是一个很小的值,这样会占用很多的带宽。那怎么去解决这种应用呢?看看websocket的做法。

  握手协议

     在实现websocket连线过程中,需要通过浏览器发出websocket连线请求,然后服务器发出回应,这个过程通常称为“握手” (handshaking)。

     本文只介绍一种版本的websocket握手方式。

  握手的原理大致如下:

     客户端请求web socket连接时,会向服务器端发送握手请求

var ws = new WebSocket('ws://127.0.0.1:8090');

     请求内容大致如下:

GET / HTTP/1.1
Host: 192.168.0.10:8080
Connection: Upgrade
Pragma: no-cache
Cache-Control: no-cache
Upgrade: websocket
Origin: null
Sec-WebSocket-Version: 13
User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/39.0.2171.95 Safari/537.36
Accept-Encoding: gzip, deflate, sdch
Accept-Language: zh-CN,zh;q=0.8
Sec-WebSocket-Key: VR+OReqwhymoQ21dBtoIMQ==
Sec-WebSocket-Extensions: permessage-deflate; client_max_window_bits

  请求包说明:
* 必须是有效的http request 格式;
* HTTP request method 必须是GET,协议应不小于1.1 如: Get / HTTP/1.1;
* 必须包括Upgrade头域,并且其值为”websocket”;
* 必须包括”Connection” 头域,并且其值为”Upgrade”;
* 必须包括”Sec-WebSocket-Key”头域,其值采用base64编码的随机16字节长的字符序列;
* 如果请求来自浏览器客户端,还必须包括Origin头域 。 该头域用于防止未授权的跨域脚本攻击,服务器可以从Origin决定是否接受该WebSocket连接;
* 必须包括”Sec-webSocket-Version” 头域,当前值必须是13;
* 可能包括”Sec-WebSocket-Protocol”,表示client(应用程序)支持的协议列表,server选择一个或者没有可接受的协议响应之;
* 可能包括”Sec-WebSocket-Extensions”, 协议扩展, 某类协议可能支持多个扩展,通过它可以实现协议增强;
* 可能包括任意其他域,如cookie.

服务器端响应如下:

HTTP/1.1 101 Web Socket Protocol

HandshakeUpgrade: websocket

Connection: Upgrade

Sec-WebSocket-Accept: Y+Te7S7wQJC0FwXumEdGbv9/Mek=

 应答包说明:
*必须包括Upgrade头域,并且其值为”websocket”;
*必须包括Connection头域,并且其值为”Upgrade”;
*必须包括Sec-WebSocket-Accept头域,其值是将请求包“Sec-WebSocket-Key”的值,与”258EAFA5- E914-47DA-95CA-C5AB0DC85B11″这个字符串进行拼接,然后对拼接后的字符串进行sha-1运算,再进行base64编码,就是 “Sec-WebSocket-Accept”的值;
*应答包中冒号后面有一个空格;
*最后需要两个空行作为应答包结束。

请注意:258EAFA5- E914-47DA-95CA-C5AB0DC85B11 这一串加密的字串是固定的,不可更改,否则会握手失败。客户端的请求头实际上是一个http请求,我们只要从头部匹配出 Sec-WebSocket-Key 并且按照固定加密返回即可。

获取 Sec-WebSocket-Key 方法:

1 public function getKey($req) {
2         $key = null;
3         if (preg_match("/Sec-WebSocket-Key: (.*)\r\n/", $req, $match)) {
4             $key = $match[1];
5         }
6         return $key;
7     }

 加密返回方法:

1 public function encry($req){
2         $key = $this->getKey($req);
3         $mask = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
4 
5         return base64_encode(sha1($key . $mask, true));
6     }

拼装握手方法:

 1 public function dohandshake($socket, $req){
 2         // 获取加密key
 3         $acceptKey = $this->encry($req);
 4         $upgrade = "HTTP/1.1 101 Switching Protocols\r\n" .
 5                    "Upgrade: websocket\r\n" .
 6                    "Connection: Upgrade\r\n" .
 7                    "Sec-WebSocket-Accept: " . $acceptKey . "\r\n" .
 8                    "\r\n";
 9 
10         // 写入socket
11         socket_write($socket, $upgrade.chr(0), strlen($upgrade.chr(0)));
12     }

也可以看看GoEasy文库的其他资料。

发表评论

邮箱地址不会被公开。

此站点使用Akismet来减少垃圾评论。了解我们如何处理您的评论数据