clrghouz/app/Classes/Sock/SocketServer.php

168 lines
4.1 KiB
PHP

<?php
namespace App\Classes\Sock;
use Illuminate\Support\Facades\Log;
use App\Classes\Sock\Exception\{HAproxyException,SocketException};
final class SocketServer {
private const LOGKEY = 'SS-';
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
private int $type; // Socket type
private array $handler; // The class and method that will handle our connection
public function __construct(int $port,string $bind='0.0.0.0',int $type=SOCK_STREAM)
{
$this->bind = $bind;
$this->port = $port;
$this->type = $type;
$this->createSocket();
if (socket_bind($this->server,$this->bind,$this->port) === FALSE)
throw new SocketException(SocketException::CANT_BIND_SOCKET,socket_strerror(socket_last_error($this->server)));
}
public function __get($key)
{
switch ($key) {
case 'handler':
return $this->handler;
default:
throw new \Exception('Unknown key: '.$key);
}
}
public function __set($key,$value)
{
switch ($key) {
case 'handler':
return $this->handler = $value;
default:
throw new \Exception('Unknown key: '.$key);
}
}
/**
* 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');
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);
}
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);
}
/**
* 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.');
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)));
Log::info(sprintf('%s:- Listening on [%s:%d]',self::LOGKEY,$this->bind,$this->port));
switch ($this->type) {
case SOCK_STREAM:
$this->loop_tcp();
break;
case SOCK_DGRAM:
$this->loop_udp();
break;
}
socket_close($this->server);
Log::info(sprintf('%s:= Closed [%s:%d]',self::LOGKEY,$this->bind,$this->port));
}
/**
* Manage and execute incoming connections
*
* @throws SocketException
*/
private function loop_tcp(): void
{
while (TRUE) {
if (($accept = socket_accept($this->server)) === FALSE)
throw new SocketException(SocketException::CANT_ACCEPT,socket_strerror(socket_last_error($this->server)));
Log::debug(sprintf('%s:* TCP Loop Start',self::LOGKEY));
try {
$r = new SocketClient($accept);
} catch (HAproxyException $e) {
Log::notice(sprintf('%s:! HAPROXY Exception [%s]',self::LOGKEY,$e->getMessage()));
socket_close($accept);
continue;
} catch (\Exception $e) {
Log::notice(sprintf('%s:! Creating Socket client failed? [%s]',self::LOGKEY,$e->getMessage()));
socket_close($accept);
continue;
}
// If the handler returns a value, then that is the main thread
if (! $this->handler[0]->{$this->handler[1]}($r)) {
$r->close();
exit(0);
}
}
}
private function loop_udp(): void
{
while (TRUE) {
$r = new SocketClient($this->server);
if ($r->hasData(30)) {
if (! ($this->handler[0]->{$this->handler[1]}($r)))
exit(0);
// Sleep so our thread has a chance to pick up the data from our connection
usleep(50000);
}
}
}
}