Приветствую.
Возникла потребность осуществить рабочие запросы php->node js.
Нашел на гитхабе проект
проект для этих целей. Собственно последний комит был два года назад и проблема возможно в том, что модуль socket io потерпел некоторые изменения.
Выложу сразу класс для того что бы вы понимали как происходит отправка запроса на сервер:
Кликните здесь для просмотра всего текста
PHP |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
| /**
* Class SocketIO
* develope by psinetron (slybeaver)
* Git: https://github.com/psinetron
* web-site: http://slybeaver.ru
*
*/
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: http://$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: *\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;
}
} |
|
Он подключается, имитирует эмит и благополучно отключается (жаль не в моём случае).
Итак суть проблемы:
Есть Server.js :
Javascript |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
| const websocketport = 3000; //Our port
var io = require('socket.io').listen(websocketport); // Web-socket
io.sockets.on('connection', function (socket) {
console.log("New User connected");
socket.on('messagess', function (msg) {
console.log(msg);
});
socket.on('disconnect', function () {
console.log("User disconnected");
});
}); |
|
Есть Phptonode.php :
PHP |
1
2
3
4
5
6
7
8
9
10
| <?php
include_once 'socket.io.php';
$socketio = new SocketIO();
if ($socketio->send('x.x.x.x', 3000, 'messagess', 'Hello world!')){
echo 'we sent the message and disconnected';
} else {
echo 'Sorry, we have a mistake :\'(';
}
?> |
|
Есть проблема:
Задействовав phptonode.php, он подключается к серверу, отправляет сообщение и отключается (как и должно быть).
После того как я повторяю это действие 2-3 раза, страничка phptonode.php зависает на секунд так 15-20, да и другие странички тоже.. Приходится выключать node js сервер для того чтобы все снова заработало... Иногда зависает на более длительный срок.
Причем в логах нода (если мы берем момент когда зависает phptonode.php) пишет:
Код
User connected
"Сообщение с эмита"
Потом, когда phptonode.php отвисает, на ноду уже приходит
В чём может быть проблема?
В дебаг моде на ноде с сокетами вроде никаких проблем нету.. Если необходимо, дебаг лог нода могу выложить.
К справке: если отправлять данных сокетами с помощью JS'a - никаких проблем не возникает.