zoukankan      html  css  js  c++  java
  • WebSocket帧数据 解码/转码

    数据从浏览器通过websocket发送给服务器的数据,是原始的帧数据,默认是被掩码处理过的,所以需要对其利用掩码进行解码。

    从服务器发送给浏览器的数据是默认没有掩码处理的,只要符合一定结构就可以了。具体可以参考websocket的RFC文档

    http://www.rfcreader.com/#rfc6455

    The threat model being protected against is one in which the client sends data that appears to be an HTTP request. As such, the channel that needs to be masked is the data from the client to the server. The data from the server to the client can be made to look like a response, but to accomplish this request, the client must also be able to forge a request. As such, it was not deemed necessary to mask data in both directions (the data from the server to the client is not masked)

    解码函数

    function decode($received)
          {
              $len = $masks = $data = $decoded = null;
              $buffer = $received;
              $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;
          }
    

      

    编码函数:

     1 function encode($buffer)
     2       {
     3           $len = strlen($buffer);
     4 
     5           $first_byte = self::BINARY_TYPE_BLOB;
     6 
     7           if ($len <= 125) {
     8               $encode_buffer = $first_byte . chr($len) . $buffer;
     9           } else {
    10               if ($len <= 65535) {
    11                   $encode_buffer = $first_byte . chr(126) . pack("n", $len) . $buffer;
    12               } else {
    13                 //pack("xxxN", $len)pack函数只处理2的32次方大小的文件,实际上2的32次方已经4G了。 
    14                   $encode_buffer = $first_byte . chr(127) . pack("xxxxN", $len) . $buffer;
    15               }
    16           }
    17 
    18           return $encode_buffer;
    19       }
    20   }

    服务器端代码:

      1 <?php 
      2   
      3   class WebSocket
      4   {
      5       const BINARY_TYPE_BLOB = "x81";
      6       private $socket;
      7 
      8       public function __construct($port)
      9       {
     10           $socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
     11           socket_set_option($socket, SOL_SOCKET, SO_REUSEADDR, 1);
     12           socket_bind($socket, 0, $port);
     13           socket_listen($socket);
     14           return $this ->socket = $socket;
     15       }
     16 
     17       public function run()
     18       {
     19           $clients[] = $this -> socket;
     20           while(true)
     21           {
     22               $read = $clients;
     23               $write = null;
     24               $except = null;
     25               if(false === socket_select($read, $write, $except, null))
     26               {
     27                   continue;
     28               }
     29 
     30               if(in_array($this -> socket, $read))
     31               {
     32                   echo "new request ...
    ";
     33                   $clients[] = $newsock = socket_accept($this -> socket);
     34                   $request = socket_read($newsock, 321600*2);
     35                   if($this -> handshake($newsock, $request))
     36                   {
     37                       socket_getpeername($newsock, $ip);
     38                       echo "new client $ip conntected
    ";
     39                       $key = array_search($this -> socket, $read);
     40                       unset($read[$key]);
     41                   }
     42                   else
     43                   {
     44                       echo "handshake failed 
    ";
     45                   }
     46               }
     47 
     48               foreach ($read as $sock_read)
     49               {
     50                   $data = socket_read($sock_read, 32160*2);
     51                   if(false === $data)
     52                   {
     53                       $key = array_search($sock_read, $clients);
     54                       socket_getpeername($clients[$key], $ip);
     55                       unset($clients[$key]);
     56                       echo "client $ip disconnected
    ";
     57                       continue;
     58                   }
     59                   $data = $this -> decode($data);
     60                   $response = "recevied data len : ".strlen($data)." $data";
     61                   $response = $this -> encode($response);
     62                   socket_write($sock_read, $response, strlen($response));
     63               }
     64           }
     65           socket_close($this -> socket);
     66       }
     67 
     68       public function handshake($socket, $request)
     69       {
     70             if (preg_match("/Sec-WebSocket-Key: *(.*?)
    /i", $request, $match)) {
     71                 $Sec_WebSocket_Key = $match[1];
     72             } 
     73             // Calculation websocket key.
     74             $new_key = base64_encode(sha1($Sec_WebSocket_Key . "258EAFA5-E914-47DA-95CA-C5AB0DC85B11", true));
     75             // Handshake response data.
     76             $handshake_message = "HTTP/1.1 101 Switching Protocols
    ";
     77             $handshake_message .= "Upgrade: websocket
    ";
     78             $handshake_message .= "Sec-WebSocket-Version: 13
    ";
     79             $handshake_message .= "Connection: Upgrade
    ";
     80             $handshake_message .= "Sec-WebSocket-Accept: " . $new_key . "
    
    ";
     81 
     82             socket_write($socket, $handshake_message);
     83 
     84             return true;
     85       }
     86 
     87       public function decode($received)
     88       {
     89           $len = $masks = $data = $decoded = null;
     90           $buffer = $received;
     91           $len = ord($buffer[1]) & 127;
     92           if ($len === 126) {
     93               $masks = substr($buffer, 4, 4);
     94               $data  = substr($buffer, 8);
     95           } else {
     96               if ($len === 127) {
     97                   $masks = substr($buffer, 10, 4);
     98                   $data  = substr($buffer, 14);
     99               } else {
    100                   $masks = substr($buffer, 2, 4);
    101                   $data  = substr($buffer, 6);
    102               }
    103           }
    104           for ($index = 0; $index < strlen($data); $index++) {
    105               $decoded .= $data[$index] ^ $masks[$index % 4];
    106           }
    107 
    108           return $decoded;
    109       }
    110 
    111       public function encode($buffer)
    112       {
    113           $len = strlen($buffer);
    114 
    115           $first_byte = self::BINARY_TYPE_BLOB;
    116 
    117           if ($len <= 125) {
    118               $encode_buffer = $first_byte . chr($len) . $buffer;
    119           } else {
    120               if ($len <= 65535) {
    121                   $encode_buffer = $first_byte . chr(126) . pack("n", $len) . $buffer;
    122               } else {
    123                 //pack("xxxN", $len)pack函数只处理2的32次方大小的文件,实际上2的32次方已经4G了。 
    124                   $encode_buffer = $first_byte . chr(127) . pack("xxxxN", $len) . $buffer;
    125               }
    126           }
    127 
    128           return $encode_buffer;
    129       }
    130   }
    131 
    132   $ws = new WebSocket(1313);
    133   $ws -> run();

    客户端代码:

     1 $(function(){
     2     url = "ws://localhost:1313/server.php";
     3     socket = new WebSocket(url);
     4     cnt = 1;
     5 
     6     socket.onopen = function()
     7     {
     8         socket.send("this is chrome broswer");
     9     }
    10 
    11     socket.onmessage = function(msg)
    12     {
    13         $('#info').append(msg.data);
    14     }
    15 });

    效果

  • 相关阅读:
    深入理解JavaScript定时器(续)
    也谈前端基础设施建设
    Reporting Services在指定计算机上找不到报表服务器
    优化tempdb提高SQL Server的性能
    SQL 代理服务未运行。此操作需要 SQL 代理服务。 (rsSchedulerNotResponding) 获取联机帮助
    报表服务器上出现内部错误。有关详细信息,请参阅错误日志。 (rsInternalError) 获取联机帮助.找不到存储过程 'GetOneConfigurationInfo'。
    表中包含有外键时无法进行导入数据,
    SQLSTATE ODBC API(驱动程序管理器)错误
    数据库只能用机器名连接,不能用ip地址连接
    请教:不能访问通过IP访问,却可以通过机器名访问
  • 原文地址:https://www.cnblogs.com/yangxunwu1992/p/5568856.html
Copyright © 2011-2022 走看看