【问题标题】:How to send WebSocket hybi-17 frame with php server如何使用 php 服务器发送 WebSocket hybi-17 帧
【发布时间】:2012-06-13 13:43:42
【问题描述】:

我尝试在 PHP 握手和接收数据中实现 WebSocket 服务器,但如果我尝试向客户端发送数据,Chrome 19 会说 “服务器不得屏蔽它发送给客户端的任何帧。” 但我不掩盖数据。 我的代码如下所示:

function wrap($msg=""){ 

$byte1 = 0x80 | (0x1 & 0x0f);

if(strlen($msg) <= 125){
    $header = pack('CC', $byte1, strlen($msg));
}
elseif(strlen($msg) >= 126 && strlen($msg) <= 65535){
    $header = pack('CCn', $byte1, 126, strlen($msg));
}
else{
    $header = pack('CCN', $byte1, 126, strlen($msg));
}
$this->log($header);
return $header.$msg;
}

我使用 socket_write() 将它发送给客户端

编辑: Firefox 13 也会关闭连接

【问题讨论】:

    标签: php phpwebsocket


    【解决方案1】:

    我遇到了同样的问题:对于从服务器发送的某些消息,浏​​览器中没有响应,对于某些消息,虽然我没有添加任何掩码,但会显示错误“A server must not mask any frames ...” . 原因在于发送的握手。 握手是:

    "HTTP/1.1 101 Web Socket Protocol Handshake\r\n" .
    ...
    "WebSocket-Location: ws://{$host}{$resource}\r\n\r\n" . chr(0)
    

    那个 chr(0) 是原因,在我删除它之后一切正常。

    消息编码函数:

    protected function hybi10Encode($payload, $type = 'text', $masked = true) {
            $frameHead = array();
            $frame = '';
            $payloadLength = strlen($payload);
    
            switch ($type) {
                case 'text':
                    // first byte indicates FIN, Text-Frame (10000001):
                    $frameHead[0] = 129;
                    break;
    
                case 'close':
                    // first byte indicates FIN, Close Frame(10001000):
                    $frameHead[0] = 136;
                    break;
    
                case 'ping':
                    // first byte indicates FIN, Ping frame (10001001):
                    $frameHead[0] = 137;
                    break;
    
                case 'pong':
                    // first byte indicates FIN, Pong frame (10001010):
                    $frameHead[0] = 138;
                    break;
            }
    
            // set mask and payload length (using 1, 3 or 9 bytes)
            if ($payloadLength > 65535) {
                $payloadLengthBin = str_split(sprintf('%064b', $payloadLength), 8);
                $frameHead[1] = ($masked === true) ? 255 : 127;
                for ($i = 0; $i < 8; $i++) {
                    $frameHead[$i + 2] = bindec($payloadLengthBin[$i]);
                }
    
                // most significant bit MUST be 0 (close connection if frame too big)
                if ($frameHead[2] > 127) {
                    $this->close(1004);
                    return false;
                }
            } elseif ($payloadLength > 125) {
                $payloadLengthBin = str_split(sprintf('%016b', $payloadLength), 8);
                $frameHead[1] = ($masked === true) ? 254 : 126;
                $frameHead[2] = bindec($payloadLengthBin[0]);
                $frameHead[3] = bindec($payloadLengthBin[1]);
            } else {
                $frameHead[1] = ($masked === true) ? $payloadLength + 128 : $payloadLength;
            }
    
            // convert frame-head to string:
            foreach (array_keys($frameHead) as $i) {
                $frameHead[$i] = chr($frameHead[$i]);
            }
    
            if ($masked === true) {
                // generate a random mask:
                $mask = array();
                for ($i = 0; $i < 4; $i++) {
                    $mask[$i] = chr(rand(0, 255));
                }
    
                $frameHead = array_merge($frameHead, $mask);
            }
            $frame = implode('', $frameHead);
            // append payload to frame:
            for ($i = 0; $i < $payloadLength; $i++) {
                $frame .= ($masked === true) ? $payload[$i] ^ $mask[$i % 4] : $payload[$i];
            }
    
            return $frame;
        }
    

    【讨论】:

    • 感谢分享您的代码。我使用它但不得不将$masked = true 更改为$masked = false
    猜你喜欢
    • 2011-12-15
    • 2011-12-18
    • 1970-01-01
    • 2011-10-26
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多