2021-05-07 22:07:26 +10:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Classes\Sock;
|
|
|
|
|
|
|
|
use Illuminate\Support\Facades\Log;
|
|
|
|
|
|
|
|
final class SocketServer {
|
2021-08-16 00:41:43 +10:00
|
|
|
private const LOGKEY = 'SS-';
|
|
|
|
|
2021-05-07 22:07:26 +10:00
|
|
|
private $server; // Our server resource
|
|
|
|
private string $bind; // The IP address to bind to
|
|
|
|
private int $port; // The Port to bind to
|
|
|
|
private int $backlog = 5; // Number of incoming connections queued
|
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
private int $type; // Socket type
|
|
|
|
|
2021-05-07 22:07:26 +10:00
|
|
|
private array $handler; // The class and method that will handle our connection
|
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
public function __construct(int $port,string $bind='0.0.0.0',int $type=SOCK_STREAM)
|
2021-05-07 22:07:26 +10:00
|
|
|
{
|
|
|
|
$this->bind = $bind;
|
|
|
|
$this->port = $port;
|
2023-04-22 21:30:30 +10:00
|
|
|
$this->type = $type;
|
2021-05-07 22:07:26 +10:00
|
|
|
|
|
|
|
$this->_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bind to our Socket
|
|
|
|
*
|
|
|
|
* @throws SocketException
|
|
|
|
*/
|
|
|
|
private function _bindSocket(): void
|
|
|
|
{
|
|
|
|
if (socket_bind($this->server,$this->bind,$this->port) === FALSE)
|
|
|
|
throw new SocketException(SocketException::CANT_BIND_SOCKET,socket_strerror(socket_last_error($this->server)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create our Socket
|
|
|
|
*
|
|
|
|
* @throws SocketException
|
|
|
|
*/
|
|
|
|
private function _createSocket(): void
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Check dependencies
|
|
|
|
*/
|
|
|
|
if (! extension_loaded('sockets'))
|
|
|
|
throw new SocketException(SocketException::CANT_ACCEPT,'Missing sockets extension');
|
|
|
|
|
|
|
|
if (! extension_loaded('pcntl'))
|
|
|
|
throw new SocketException(SocketException::CANT_ACCEPT,'Missing pcntl extension');
|
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
switch ($this->type) {
|
|
|
|
case SOCK_STREAM:
|
|
|
|
$this->server = socket_create(AF_INET|AF_INET6,$this->type,SOL_TCP);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SOCK_DGRAM:
|
|
|
|
$this->server = socket_create(AF_INET|AF_INET6,$this->type,SOL_UDP);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw new \Exception('Unknown socket_type:'.$this->type);
|
|
|
|
}
|
2021-05-07 22:07:26 +10:00
|
|
|
|
|
|
|
if ($this->server === FALSE)
|
|
|
|
throw new SocketException(SocketException::CANT_CREATE_SOCKET,socket_strerror(socket_last_error()));
|
|
|
|
|
|
|
|
socket_set_option($this->server,SOL_SOCKET,SO_REUSEADDR,1);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Setup Socket and Bind
|
|
|
|
*
|
|
|
|
* @throws SocketException
|
|
|
|
*/
|
|
|
|
private function _init(): void
|
|
|
|
{
|
|
|
|
$this->_createSocket();
|
|
|
|
$this->_bindSocket();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Our main loop where we listen for connections
|
|
|
|
*
|
|
|
|
* @throws SocketException
|
|
|
|
*/
|
|
|
|
public function listen()
|
|
|
|
{
|
|
|
|
if (! $this->handler)
|
|
|
|
throw new SocketException(SocketException::CANT_LISTEN,'Handler not set.');
|
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
if (in_array($this->type,[SOCK_STREAM,SOCK_SEQPACKET]))
|
|
|
|
if (socket_listen($this->server,$this->backlog) === FALSE)
|
|
|
|
throw new SocketException(SocketException::CANT_LISTEN,socket_strerror(socket_last_error($this->server)));
|
2021-05-07 22:07:26 +10:00
|
|
|
|
2021-08-16 00:41:43 +10:00
|
|
|
Log::info(sprintf('%s:- Listening on [%s:%d]',self::LOGKEY,$this->bind,$this->port));
|
2021-05-07 22:07:26 +10:00
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
switch ($this->type) {
|
|
|
|
case SOCK_STREAM:
|
|
|
|
$this->loop_tcp();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case SOCK_DGRAM:
|
|
|
|
$this->loop_udp();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-05-07 22:07:26 +10:00
|
|
|
socket_close($this->server);
|
|
|
|
|
2021-08-16 00:41:43 +10:00
|
|
|
Log::info(sprintf('%s:= Closed [%s:%d]',self::LOGKEY,$this->bind,$this->port));
|
2021-05-07 22:07:26 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-20 00:23:41 +10:00
|
|
|
* Manage and execute incoming connections
|
2021-05-07 22:07:26 +10:00
|
|
|
*
|
|
|
|
* @throws SocketException
|
|
|
|
*/
|
2023-04-22 21:30:30 +10:00
|
|
|
private function loop_tcp()
|
2021-05-07 22:07:26 +10:00
|
|
|
{
|
|
|
|
while (TRUE) {
|
|
|
|
if (($accept = socket_accept($this->server)) === FALSE)
|
|
|
|
throw new SocketException(SocketException::CANT_ACCEPT,socket_strerror(socket_last_error($this->server)));
|
|
|
|
|
2021-09-11 11:47:26 +10:00
|
|
|
try {
|
|
|
|
$r = new SocketClient($accept);
|
|
|
|
|
|
|
|
} catch (\ErrorException $e) {
|
|
|
|
Log::error(sprintf('%s:! Creating Socket client failed? [%s]',self::LOGKEY,$e->getMessage()));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->handler[0]->{$this->handler[1]}($r);
|
2021-05-07 22:07:26 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-22 21:30:30 +10:00
|
|
|
private function loop_udp()
|
|
|
|
{
|
|
|
|
$buf = '';
|
|
|
|
$remote = [];
|
|
|
|
$remote['ip'] = NULL;
|
|
|
|
$remote['port'] = NULL;
|
|
|
|
|
|
|
|
while (TRUE) {
|
|
|
|
if (socket_recvfrom($this->server,$buf,512,MSG_WAITALL,$remote['ip'],$remote['port']))
|
|
|
|
$this->handler[0]->{$this->handler[1]}($remote,$buf,$this->server);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-07 22:07:26 +10:00
|
|
|
/**
|
|
|
|
* Set our connection handler Class and Method
|
|
|
|
*
|
|
|
|
* @param array $handler
|
|
|
|
*/
|
|
|
|
public function setConnectionHandler(array $handler): void
|
|
|
|
{
|
|
|
|
$this->handler = $handler;
|
|
|
|
}
|
|
|
|
}
|