我使用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;
}
}
感谢您的帮助!
1条答案
按热度按时间vlju58qv1#
我对github上的所有库都有同样的问题,问题是它们被放弃了或者没有更新到socket.io V3。
在socket.io文档中指出:
TL;DR:由于几项重大更改,v2客户端将无法连接到v3服务器(反之亦然)
要解决这个问题,您需要了解socket.io客户端是如何工作的,这很容易,因为在协议文档的sample-session部分中有介绍。
Socket.Io协议文档
要解决这个问题,您需要忘记fsockopen和fwrite函数,您需要直接使用CURL来执行协议文档中提到的请求。
请求编号1
请求编号2
请求编号3
请求编号4
例如42[“notifications”,“Hi,Im a notification”],它相当于socket.emit(event,data)* 向服务器发出消息 *:将您的邮件发送到socket.io服务器。
下面是一个使用Symfony 5.2和HttpClientInterface的BASIC示例:
正如你所看到的,非常简单,你不需要麻烦的“复制粘贴”库。我说“复制粘贴”是因为所有的库都使用相同的代码打开de socket并发送信息,但没有一个库与socket.io V3兼容。
这里有一个图片,证明了给定的代码在2021年1月4日可以在php 7.4,symfony 5.2和www.example.com V3上正常工作socket.io。
这是我在节点中测试服务器
我需要说的是,如果你想向你的www.example.com服务器发送“单向”消息socket.io,比如一个新的通知或任何不需要永久连接的东西,这个解决方案非常好用,只是“一次性”,没有别的。
编码快乐,来自墨西哥的问候。
下面是另一个例子:
第一列是Postman向php服务器发出请求,模拟一个服务器端事件,比如创建一个新的问题,在响应中是您需要发出的4个请求的响应正文的转储。
第二列是socket.IO节点服务器,运行在端口3000上
而最后一列是chrome控制台,模拟一个用户通过WebSocket连接到socket.io服务器上寻找'questions'事件中的通知。