똥서버에 웹소켓 100개 연결하기 (php openswoole benchmark)
ReactPHP와 같이 최근 php 진영에서도 node나 golang과 같이
논블록킹 I/O를 지원할 수 있도록 하는 서드파티 익스텐션이 나오고 있다.
그 중에서도 C++로 작성되어 php로 작성된 ReactPHP보다 성능이 좋다고 하는 swoole extension을 테스트 해봤다.
굉장히 단편적고 단순한 테스트임으로 참고만 하길...
일단 테스트서버는 구름 IDE에서 제공하는 가장 낮은 사양의 서버를 준비했다.
0.5vCPU / 1GB Memory를 갖는 빈 컨테이너를 준비했고,
우분투 24.04, PHP 8.3.14에 openSwoole은 22.1.2버전을 사용했다.
(참고로 Swoole과 Openswoole 은 엄연히 다르고 사용하는 방식도 다름!)
cli로 구동되는 php openswoole의 경우, node.js나 golang처럼
프로그램 동작중에 급작스러운 Fetal 에러가 발생되면 프로세스가 종료되면서 서비스전체가 다운될 수 있음으로
pm2를 사용하는 node.js 처럼 Supervisor를 설치하여 php를 감시할 수 있도록 하였다.
다음은 웹소켓 서버를 열어주는 간단한 예제이다.
<?php
use OpenSwoole\WebSocket\Server;
use OpenSwoole\Http\Request;
use OpenSwoole\WebSocket\Frame;
$server = new Server("0.0.0.0", 9502);
$server->on("Start", function(Server $server){
echo "OpenSwoole WebSocket Server is started at http://127.0.0.1:9502\n";
});
$server->on('Open', function(Server $server, OpenSwoole\Http\Request $request){
echo "connection open: {$request->fd}\n";
$server->tick(1000, function() use ($server, $request){
$server->push($request->fd, json_encode(["hello", time()]));
});
});
$server->on('Message', function(Server $server, Frame $frame){
echo "received message: {$frame->data}\n";
$server->push($frame->fd, json_encode(["hello", time()]));
});
$server->on('Close', function(Server $server, int $fd){
echo "connection close: {$fd}\n";
});
$server->on('Disconnect', function(Server $server, int $fd){
echo "connection disconnect: {$fd}\n";
});
$server->start();
?>
/etc/supervisor/conf.d/blabla.conf 에 설정한 supervisor설정은 다음과 같다.
[program:swoole_websocket]
command=php /경로/swoole_websocket.php queue:work
directory=/경로
autostart=true
autorestart=true
user=www-data
stdout_logfile=/var/log/swoole_websocket.log
sudo supervisorctl shutdown 명령후
sudo supervisord -c /etc/supervisor/supervisord.conf 으로 강제 실행하여 openswoole 웹소켓 서버를 실행했고,
원래는 Nginx에서 ReverseProxy를 통해 SSL을 붙여주려하였으나,
귀찮아서 구름에서 제공하는 URL을 그대로 사용했다.
(잘 작동되는것을 보아하니 이 친구도 Reverseproxy임에 틀림이 없다 ㅋㅋ)
웹소켓 클라이언트는 귀찮으니 Piehost에서 제공해주는 웹소켓 테스터를 활용했다.
그냥 무식하게 100개 탭을 열어 웹소켓 연결을 진행했다.
https://piehost.com/websocket-tester
시스템 프로세스별 점유량은 위와 같고...
구름 IDE에서 제공하는 컨테이너 사용량 상태를 봤을때는 아주 아주 준수했다.
생각보다 효율적으로 작동되어 단순한 기능을 작동시키는 마이크로서비스를 구성하기에
아주 좋은 방법일 듯 하다.
해외 커뮤에서 직접적으로 벤치를 진행해서 비교한 결과를 보면
node보다는 성능이 우수하고 golang에 비해서는 약간 못미치는 성능을 보여주는것 같다.