From 013719bb6fb3fb3aa72b8e45ff79c348d269dc5a Mon Sep 17 00:00:00 2001 From: roxblnfk Date: Mon, 23 Sep 2024 22:58:23 +0400 Subject: [PATCH 1/3] fix(server): Add throttling to `socket_accept()` calls --- src/Application.php | 1 + src/Socket/Server.php | 17 ++++++++++++++--- 2 files changed, 15 insertions(+), 3 deletions(-) diff --git a/src/Application.php b/src/Application.php index 92a9dc92..71af3e14 100644 --- a/src/Application.php +++ b/src/Application.php @@ -256,6 +256,7 @@ private function createServer(SocketServer $config, Inspector $inspector): Serve return Server::init( $config->port, payloadSize: 524_288, + acceptPeriod: .001, clientInflector: $clientInflector, logger: $this->logger, ); diff --git a/src/Socket/Server.php b/src/Socket/Server.php index b4b79270..15002911 100644 --- a/src/Socket/Server.php +++ b/src/Socket/Server.php @@ -10,7 +10,6 @@ use Buggregator\Trap\Processable; use Buggregator\Trap\Socket\Exception\ClientDisconnected; use Buggregator\Trap\Socket\Exception\ServerStopped; -use Socket; /** * @internal @@ -27,13 +26,18 @@ final class Server implements Processable, Cancellable, Destroyable private bool $cancelled = false; + /** Timestamp with microseconds when last socket_accept() was called */ + private float $lastAccept = 0; + /** * @param null|\Closure(Client, int $id): void $clientInflector * @param positive-int $payloadSize Max payload size. + * @param float $acceptPeriod Time to wait between socket_accept() calls in seconds. */ private function __construct( int $port, private readonly int $payloadSize, + private readonly float $acceptPeriod, private readonly ?\Closure $clientInflector, private readonly Logger $logger, ) { @@ -50,15 +54,17 @@ private function __construct( /** * @param int<1, 65535> $port * @param positive-int $payloadSize Max payload size. + * @param float $acceptPeriod Time to wait between socket_accept() calls in seconds. * @param null|\Closure(Client, int $id): void $clientInflector */ public static function init( int $port = 9912, int $payloadSize = 10485760, + float $acceptPeriod = .001, ?\Closure $clientInflector = null, Logger $logger = new Logger(), ): self { - return new self($port, $payloadSize, $clientInflector, $logger); + return new self($port, $payloadSize, $acceptPeriod, $clientInflector, $logger); } public function destroy(): void @@ -82,7 +88,12 @@ public function destroy(): void public function process(): void { // /** @psalm-suppress PossiblyInvalidArgument */ - while (!$this->cancelled and false !== ($socket = \socket_accept($this->socket))) { + while (match(true) { + $this->cancelled, + \microtime(true) - $this->lastAccept <= $this->acceptPeriod => false, + default => false !== ($socket = \socket_accept($this->socket)), + }) { + $this->lastAccept = \microtime(true); $client = null; try { /** @psalm-suppress MixedArgument */ From 6997e47b0a948b033ca07c822d09cddd58f4c09a Mon Sep 17 00:00:00 2001 From: roxblnfk Date: Mon, 23 Sep 2024 23:28:18 +0400 Subject: [PATCH 2/3] fix(server): Add throttling to `socket_select()` calls --- src/Socket/Client.php | 13 ++++++++++++- src/Socket/Server.php | 2 +- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/src/Socket/Client.php b/src/Socket/Client.php index 61d0ec55..bd9a63ef 100644 --- a/src/Socket/Client.php +++ b/src/Socket/Client.php @@ -27,13 +27,17 @@ final class Client implements Destroyable private \Closure $onClose; + private Timer $selectTimer; + /** * @param positive-int $payloadSize */ private function __construct( private readonly \Socket $socket, private readonly int $payloadSize, + float $selectPeriod, ) { + $this->selectTimer = new Timer($selectPeriod); \socket_set_nonblock($this->socket); $this->setOnPayload(static fn(string $payload) => null); $this->setOnClose(static fn() => null); @@ -41,12 +45,18 @@ private function __construct( /** * @param positive-int $payloadSize Max payload size. + * @param float $selectPeriod Time to wait between socket_select() calls in seconds. */ public static function init( \Socket $socket, int $payloadSize = 10485760, + float $selectPeriod = .001, ): self { - return new self($socket, $payloadSize); + return new self( + socket: $socket, + payloadSize: $payloadSize, + selectPeriod: $selectPeriod, + ); } public function destroy(): void @@ -103,6 +113,7 @@ public function process(): void throw new ClientDisconnected(); } \Fiber::suspend(); + $this->selectTimer->reset()->wait(); } while (true); } diff --git a/src/Socket/Server.php b/src/Socket/Server.php index 15002911..46a63153 100644 --- a/src/Socket/Server.php +++ b/src/Socket/Server.php @@ -97,7 +97,7 @@ public function process(): void $client = null; try { /** @psalm-suppress MixedArgument */ - $client = Client::init($socket, $this->payloadSize); + $client = Client::init($socket, $this->payloadSize, $this->acceptPeriod); $key = (int) \array_key_last($this->clients) + 1; $this->clients[$key] = $client; $this->clientInflector !== null and ($this->clientInflector)($client, $key); From 6016c18c07345659cc642bf11a81df56f1525679 Mon Sep 17 00:00:00 2001 From: roxblnfk Date: Mon, 23 Sep 2024 23:44:37 +0400 Subject: [PATCH 3/3] ci(psalm): Fix psalm issues --- psalm.xml | 1 + .../Console/Renderer/Sentry/Exceptions.php | 23 ++++++++++--------- src/Socket/Server.php | 4 ++-- 3 files changed, 15 insertions(+), 13 deletions(-) diff --git a/psalm.xml b/psalm.xml index 17dc32fc..add49247 100644 --- a/psalm.xml +++ b/psalm.xml @@ -6,6 +6,7 @@ findUnusedBaselineEntry="false" findUnusedCode="false" errorBaseline="psalm-baseline.xml" + phpVersion="8.1" > diff --git a/src/Sender/Console/Renderer/Sentry/Exceptions.php b/src/Sender/Console/Renderer/Sentry/Exceptions.php index 66b467f6..aff68504 100644 --- a/src/Sender/Console/Renderer/Sentry/Exceptions.php +++ b/src/Sender/Console/Renderer/Sentry/Exceptions.php @@ -87,26 +87,27 @@ private static function renderTrace(OutputInterface $output, array $frames, bool $class = empty($class) ? '' : $class . '::'; $function = $getValue($frame, 'function'); - $renderer = static fn() => $output->writeln( - \sprintf( - "%s%s%s\n%s%s%s()", - \str_pad("#$i", $numPad, ' '), - (string) $file, - $line !== '' ? ":$line" : '', - \str_repeat(' ', $numPad), - $class, - (string) $function, - ), + $renderedLine = \sprintf( + "%s%s%s\n%s%s%s()", + \str_pad("#$i", $numPad, ' '), + (string) $file, + $line !== '' ? ":$line" : '', + \str_repeat(' ', $numPad), + $class, + (string) $function, ); if ($isFirst) { $isFirst = false; $output->writeln('Stacktrace:'); - $renderer(); + $output->writeln($renderedLine); self::renderCodeSnippet($output, $frame, padding: $numPad); continue; } + $renderer = static function () use ($output, $renderedLine): void { + $output->writeln($renderedLine); + }; if (!$verbose && \str_starts_with(\ltrim(\str_replace('\\', '/', $file), './'), 'vendor/')) { $vendorLines[] = $renderer; continue; diff --git a/src/Socket/Server.php b/src/Socket/Server.php index 46a63153..b518330e 100644 --- a/src/Socket/Server.php +++ b/src/Socket/Server.php @@ -48,7 +48,7 @@ private function __construct( \socket_set_nonblock($this->socket); - $logger->status('Application', 'Server started on 127.0.0.1:%s', $port); + $logger->status('App', 'Server started on 127.0.0.1:%s', $port); } /** @@ -96,7 +96,7 @@ public function process(): void $this->lastAccept = \microtime(true); $client = null; try { - /** @psalm-suppress MixedArgument */ + /** @var \Socket $socket */ $client = Client::init($socket, $this->payloadSize, $this->acceptPeriod); $key = (int) \array_key_last($this->clients) + 1; $this->clients[$key] = $client;