websocket Socket.io 3与PHP的集成

eulz3vhy  于 2022-11-11  发布在  PHP
关注(0)|答案(1)|浏览(245)

我使用PHP SocketIO类来连接NodeJS应用程序并发送消息。在www.example.com2中一切都运行得很好Socket.io,但在升级到版本3后,PHP集成停止了工作。
当我发送请求时,我得到以下响应:

HTTP/1.1 101 Switching Protocols
Upgrade: websocket
Connection: Upgrade
Sec-WebSocket-Accept: hNcappwZIQEbMz7ZGWS71lNcROc=

但是我在NodeJS端看不到任何东西,即使我尝试使用“connection”事件记录任何到服务器的连接。
下面是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;
    }
}

感谢您的帮助!

vlju58qv

vlju58qv1#

我对github上的所有库都有同样的问题,问题是它们被放弃了或者没有更新到socket.io V3。
在socket.io文档中指出:
TL;DR:由于几项重大更改,v2客户端将无法连接到v3服务器(反之亦然)
要解决这个问题,您需要了解socket.io客户端是如何工作的,这很容易,因为在协议文档的sample-session部分中有介绍。
Socket.Io协议文档
要解决这个问题,您需要忘记fsockopen和fwrite函数,您需要直接使用CURL来执行协议文档中提到的请求。

请求编号1

  • 获取 *
  • 网址 *:/socket.io/?EIO=4&transport=polling&t=N8hyd7H
  • 打开数据包:* 打开php和www.example.com服务器之间的连接socket.io。服务器将返回一个名为“sid”的“session id”,您将把它添加到后续查询的url查询中。
    请求编号2
  • 开机自检 *
  • 网址 *:/socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
  • 发文正文 *:'40'
  • 命名空间连接请求 *:你需要在正文中发送数字40,作为一个字符串,这意味着你要连接到socket.io的“消息”类型
    请求编号3
  • 获取 *
  • 网址 *:/socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
  • 命名空间连接批准 *:如果连接成功或出现错误,则会返回此消息,此处是socket.io服务器在您需要令牌时对您的连接进行授权的情况。
    请求编号4
  • 开机自检 *
  • 网址 *:/socket.io/?EIO=4&transport=polling&t=N8hyd7H&sid=sessionIdFromRequest1
  • 发文正文 *:42[事件,数据]

例如42[“notifications”,“Hi,Im a notification”],它相当于socket.emit(event,data)* 向服务器发出消息 *:将您的邮件发送到socket.io服务器。
下面是一个使用Symfony 5.2和HttpClientInterface的BASIC示例:

<?php

// install dependencies before: composer require symfony/http-client

use Symfony\Component\HttpClient\CurlHttpClient;

include('vendor/autoload.php');

$client = new CurlHttpClient();
sendToSocket($client);

function sendToSocket(HttpClientInterface $client)
{
    $first = $client->request('GET', 'http://localhost:3000/socket.io/?EIO=4&transport=polling&t=N8hyd6w');
    $res = ltrim($first->getContent(), '0');
    $res = json_decode($res, true);
    $sid = $res['sid'];

    $second = $client->request('POST', 'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid='.$sid, [
            'body' => '40'
        ]);
    $third = $client->request('GET', 'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid='.$sid);

    $fourth = $client->request('POST', 'http://localhost:3000/socket.io/?EIO=4&transport=polling&sid='.$sid, [
        'body' => '42["notifications","Hi, Im a notification"]'
    ]);

}

正如你所看到的,非常简单,你不需要麻烦的“复制粘贴”库。我说“复制粘贴”是因为所有的库都使用相同的代码打开de socket并发送信息,但没有一个库与socket.io V3兼容。
这里有一个图片,证明了给定的代码在2021年1月4日可以在php 7.4,symfony 5.2和www.example.com V3上正常工作socket.io。

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

// Install dependencies before: npm i express socket.io

const app = require('express')();
const http = require('http').createServer(app);

const io = require('socket.io')(http, {
   cors: {
      origin: "*",
      methods: ["GET", "POST"]
   }
});

io.on('connection', function (socket) {

   console.log("New Connection with transport", socket.conn.transport.name);

   socket.on('notifications', function (data) {
      console.log(data);
   });
});

http.listen(3000, () => {
   console.log('Server started port 3000');
});

我需要说的是,如果你想向你的www.example.com服务器发送“单向”消息socket.io,比如一个新的通知或任何不需要永久连接的东西,这个解决方案非常好用,只是“一次性”,没有别的。
编码快乐,来自墨西哥的问候。
下面是另一个例子:

第一列是Postman向php服务器发出请求,模拟一个服务器端事件,比如创建一个新的问题,在响应中是您需要发出的4个请求的响应正文的转储。
第二列是socket.IO节点服务器,运行在端口3000上
而最后一列是chrome控制台,模拟一个用户通过WebSocket连接到socket.io服务器上寻找'questions'事件中的通知。

相关问题