Socket.io 3 和 PHP 集成

Mao*_*ari 1 php sockets websocket node.js

我使用 PHP SocketIO 类连接 NodeJS 应用程序并发送消息。Socket.io 2 一切都运行得很好,但升级到版本 3 后,PHP 集成停止工作。

当我发送请求时,我收到以下回复:

HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: hNcappwZIQEbMz7ZGWS71lNcROc=
Run Code Online (Sandbox Code Playgroud)

但即使我尝试使用“连接”事件记录与服务器的任何连接,我也没有在 NodeJS 端看到任何内容。

这是 PHP 类:

class SocketIO
{
    /**
     * @param null $host - $host of socket server
     * @param null $port - port of socket server
     * @param string $action - action to execute in sockt server
     * @param null $data - message to socket server
     * @param string $address - addres of socket.io on socket server
     * @param string $transport - transport type
     * @return bool
     */
    public function send($host = null, $port = null, $action= "message",  $data = null, $address = "/socket.io/?EIO=2", $transport = 'websocket')
    {
        $fd = fsockopen($host, $port, $errno, $errstr);
        
        if (!$fd) {
            return false;
        } //Can't connect tot server
        $key = $this->generateKey();
        $out = "GET $address&transport=$transport HTTP/1.1\r\n";
        $out.= "Host: https://$host:$port\r\n";
        $out.= "Upgrade: WebSocket\r\n";
        $out.= "Connection: Upgrade\r\n";
        $out.= "Sec-WebSocket-Key: $key\r\n";
        $out.= "Sec-WebSocket-Version: 13\r\n";
        $out.= "Origin: https://$host\r\n\r\n";

        fwrite($fd, $out);

        // 101 switching protocols, see if echoes key
        $result= fread($fd,10000);

        preg_match('#Sec-WebSocket-Accept:\s(.*)$#mU', $result, $matches);

        $keyAccept = trim($matches[1]);
        $expectedResonse = base64_encode(pack('H*', sha1($key . '258EAFA5-E914-47DA-95CA-C5AB0DC85B11')));
        $handshaked = ($keyAccept === $expectedResonse) ? true : false;

        if ($handshaked){
            fwrite($fd, $this->hybi10Encode('42["' . $action . '", "' . addslashes($data) . '"]'));
            fread($fd,1000000);
            return true;
        } else {return false;}
    }
    private function generateKey($length = 16)
    {
        $c = 0;
        $tmp = '';
        while ($c++ * 16 < $length) { $tmp .= md5(mt_rand(), true); }
        return base64_encode(substr($tmp, 0, $length));
    }
    private function hybi10Encode($payload, $type = 'text', $masked = true)
    {
        $frameHead = array();
        $payloadLength = strlen($payload);
        switch ($type) {
            case 'text':
                $frameHead[0] = 129;
                break;
            case 'close':
                $frameHead[0] = 136;
                break;
            case 'ping':
                $frameHead[0] = 137;
                break;
            case 'pong':
                $frameHead[0] = 138;
                break;
        }
        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]);
            }
            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;
        }
        foreach (array_keys($frameHead) as $i) {
            $frameHead[$i] = chr($frameHead[$i]);
        }
        if ($masked === true) {
            $mask = array();
            for ($i = 0; $i < 4; $i++) {
                $mask[$i] = chr(rand(0, 255));
            }
            $frameHead = array_merge($frameHead, $mask);
        }
        $frame = implode('', $frameHead);
        for ($i = 0; $i < $payloadLength; $i++) {
            $frame .= ($masked === true) ? $payload[$i] ^ $mask[$i % 4] : $payload[$i];
        }
        return $frame;
    }
}
Run Code Online (Sandbox Code Playgroud)

谢谢你的帮助!

小智 9

我对 github 上存在的所有库都遇到了同样的问题,问题是它们被放弃或没有更新到 socket.io V3。

\n

在 socket.io 文档中说:

\n
\n

TL;DR:由于几项重大更改,v2 客户端将无法连接到 v3 服务器(反之亦然)

\n
\n

要解决这个问题,您需要了解 socket.io 客户端的工作原理,这很容易,因为在协议文档的示例会话部分中。

\n

Socket.Io协议文档

\n

要解决这个问题,您需要忘记 fsockopen 和 fwrite 函数,您需要直接使用 CURL 执行协议文档中提到的请求。

\n

请求 n\xc2\xb01
\n GET
\n url : /socket.io/?EIO=4&transport=polling&t=N8hyd7H
\n打开数据包:打开 php 和 socket.io 服务器之间的连接。服务器将返回一个名为“sid”的“会话 ID”,您将其添加到后续查询的 url 查询中。

\n

请求 n\xc2\xb02
\n POST
\n url : /socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
\n post body : \'40\'
\n命名空间连接请求: 您需要发送body 数字40,作为字符串,这意味着您要连接到socket.io“消息”类型

\n

请求 n\xc2\xb03
\n GET
\n url : /socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
\n命名空间连接批准: 如果连接成功或出现错误,则会返回,此处如果您需要令牌,socket.io 服务器会授权您的连接。

\n

请求 n\xc2\xb04
\n POST
\n url : /socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
\n帖子正文: 42[event,data]
\n例如 42["notifications","Hi ,我是一个通知"],相当于socket.emit(event,data)\n向服务器发送消息:将消息发送到socket.io 服务器。

\n

这是使用 Symfony 5.2 和 HttpClientInterface 的 BASIC 示例:

\n
<?php\n\n// install dependencies before: composer require symfony/http-client\n\nuse Symfony\\Component\\HttpClient\\CurlHttpClient;\n\ninclude(\'vendor/autoload.php\');\n\n$client = new CurlHttpClient();\nsendToSocket($client);\n\nfunction sendToSocket(HttpClientInterface $client)\n{\n    $first = $client->request(\'GET\', \'http://localhost:3000/socket.io/?EIO=4&transport=polling&t=N8hyd6w\');\n    $res = ltrim($first->getContent(), \'0\');\n    $res = json_decode($res, true);\n    $sid = $res[\'sid\'];\n\n    $second = $client->request(\'POST\', \'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid=\'.$sid, [\n            \'body\' => \'40\'\n        ]);\n    $third = $client->request(\'GET\', \'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid=\'.$sid);\n\n    $fourth = $client->request(\'POST\', \'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid=\'.$sid, [\n        \'body\' => \'42["notifications","Hi, Im a notification"]\'\n    ]);\n\n}\n
Run Code Online (Sandbox Code Playgroud)\n

正如您所看到的,这非常简单,并且您不需要令人烦恼的“复制粘贴”库。我说“复制粘贴”是因为所有人都使用相同的代码来打开套接字并发送信息,但没有人与socket.io V3兼容。

\n

这是一张图片,证明给定的代码在 2021 年 1 月 4 日使用 php 7.4、symfony 5.2 和 socket.io V3 运行。

\n

Socket.io v3 PHP

\n

这是我在节点中的测试服务器

\n
// Install dependencies before: npm i express socket.io\n\nconst app = require(\'express\')();\nconst http = require(\'http\').createServer(app);\n\nconst io = require(\'socket.io\')(http, {\n   cors: {\n      origin: "*",\n      methods: ["GET", "POST"]\n   }\n});\n\nio.on(\'connection\', function (socket) {\n\n   console.log("New Connection with transport", socket.conn.transport.name);\n\n   socket.on(\'notifications\', function (data) {\n      console.log(data);\n   });\n});\n\nhttp.listen(3000, () => {\n   console.log(\'Server started port 3000\');\n});\n
Run Code Online (Sandbox Code Playgroud)\n

我需要说的是,如果您想向您的 socket.io 服务器发送“单向”消息(例如新通知或任何不需要永久连接的内容),则该解决方案非常有效,只是“一次”,什么也没有别的。

\n

来自墨西哥的快乐编码和问候。

\n

这是另一个示例:\nSocket.io v3 PHP

\n

第一列是 Postman 向 php 服务器发出请求,模拟服务器端事件,例如创建的新问题。响应中包含您需要发出的 4 个请求的响应正文转储。

\n

第二列是在端口3000上运行的socket.IO节点服务器

\n

最后一列是 chrome 控制台,模拟用户通过 websocket 连接到 socket.IO 服务器,在“questions”事件中查找通知。

\n