laravel-websockets/src/Websocket/Handler.php

1063 lines
33 KiB
PHP
Raw Normal View History

2025-01-16 07:54:02 +00:00
<?php
declare(strict_types=1);
namespace BlaxSoftware\LaravelWebSockets\Websocket;
use BlaxSoftware\LaravelWebSockets\Apps\App;
use BlaxSoftware\LaravelWebSockets\Cache\IpcCache;
2025-01-16 07:54:02 +00:00
use BlaxSoftware\LaravelWebSockets\Channels\Channel;
use BlaxSoftware\LaravelWebSockets\Channels\PresenceChannel;
use BlaxSoftware\LaravelWebSockets\Channels\PrivateChannel;
use BlaxSoftware\LaravelWebSockets\Contracts\ChannelManager;
use BlaxSoftware\LaravelWebSockets\Events\ConnectionClosed;
use BlaxSoftware\LaravelWebSockets\Events\NewConnection;
use BlaxSoftware\LaravelWebSockets\Exceptions\WebSocketException;
use BlaxSoftware\LaravelWebSockets\Server\Exceptions\ConnectionsOverCapacity;
use BlaxSoftware\LaravelWebSockets\Server\Exceptions\OriginNotAllowed;
use BlaxSoftware\LaravelWebSockets\Server\Exceptions\UnknownAppKey;
use BlaxSoftware\LaravelWebSockets\Server\Exceptions\WebSocketException as ExceptionsWebSocketException;
use BlaxSoftware\LaravelWebSockets\Server\Messages\PusherMessageFactory;
use BlaxSoftware\LaravelWebSockets\Server\QueryParameters;
use Exception;
2025-05-08 08:54:11 +00:00
use Illuminate\Support\Facades\Auth;
2025-01-17 09:45:53 +00:00
use Illuminate\Support\Facades\DB;
use Illuminate\Support\Facades\Log;
2025-01-16 07:54:02 +00:00
use Ratchet\ConnectionInterface;
use Ratchet\RFC6455\Messaging\MessageInterface;
use Ratchet\WebSocket\MessageComponentInterface;
class Handler implements MessageComponentInterface
{
2025-12-05 19:53:52 +00:00
/**
* Track channel connections using associative arrays for O(1) lookup
* Structure: [channel_name => [socket_id => true]]
*/
protected array $channel_connections = [];
2025-01-16 07:54:02 +00:00
2025-12-05 20:48:23 +00:00
/**
* Cache write buffer for batching operations
* Reduces file I/O when multiple rapid requests occur
*/
protected array $cacheWriteBuffer = [];
protected array $cacheDeleteBuffer = [];
protected bool $cacheBufferScheduled = false;
/**
* Pre-encoded static JSON responses for performance
* Encoding once at startup is faster than encoding every time
*/
private static string $PONG_RESPONSE = '{"event":"pusher.pong"}';
/**
* GC collection counter - only collect every N pings
*/
private int $gcCounter = 0;
private const GC_INTERVAL = 100;
2025-12-05 20:48:23 +00:00
2025-01-16 07:54:02 +00:00
/**
* Initialize a new handler.
*/
public function __construct(
protected ChannelManager $channelManager
) {}
/**
* Handle incoming WebSocket message with optimized fast path for ping/pong
*/
public function onMessage(
ConnectionInterface $connection,
MessageInterface $message
): void {
if (!isset($connection->app)) {
return;
2025-12-05 19:53:52 +00:00
}
2025-06-12 14:16:07 +00:00
// FAST PATH: Check for ping before any heavy processing
// Use raw string comparison on payload to avoid JSON decode overhead
$payload = $message->getPayload();
// Quick ping detection using strpos (faster than json_decode + array access)
if ($this->tryHandlePingFast($payload, $connection)) {
return;
}
// SLOW PATH: Full message processing
2025-12-05 19:53:52 +00:00
try {
$this->processFullMessage($connection, $message, $payload);
} catch (\Throwable $e) {
$this->handleMessageError($e);
2025-01-16 07:54:02 +00:00
}
}
/**
* Fast path for ping/pong - avoids JSON decode, object creation, promises
* Target: < 1ms processing time
*/
private function tryHandlePingFast(string $payload, ConnectionInterface $connection): bool
{
// Quick string check - if doesn't contain "ping", skip fast path
// strpos is O(n) but very fast for short strings
if (strpos($payload, 'ping') === false) {
return false;
}
// Now do minimal JSON decode to confirm it's a ping
$data = json_decode($payload, true);
if ($data === null) {
return false;
}
$event = $data['event'] ?? '';
// Direct string comparison (faster than strtolower + comparison)
if ($event !== 'pusher:ping' && $event !== 'pusher.ping') {
return false;
}
// Update connection timestamp directly on connection object (no promise chain)
$connection->lastPongedAt = time();
// Send pre-encoded pong response immediately
$connection->send(self::$PONG_RESPONSE);
// Periodic GC instead of every ping
if (++$this->gcCounter >= self::GC_INTERVAL) {
$this->gcCounter = 0;
gc_collect_cycles();
}
return true;
}
/**
* Full message processing for non-ping messages
*/
private function processFullMessage(
2025-01-19 08:01:22 +00:00
ConnectionInterface $connection,
MessageInterface $message,
string $payload
): void {
// Set remote address once (moved from per-message to reduce overhead)
if (isset($connection->remoteAddress)) {
request()->server->set('REMOTE_ADDR', $connection->remoteAddress);
}
// Decode message (we already have payload string)
$messageArray = json_decode($payload, true, 512, JSON_THROW_ON_ERROR);
// Handle pusher protocol messages (subscribe, unsubscribe, etc.)
$this->handlePusherProtocolMessage($message, $connection, $messageArray);
$channel = $this->handleChannelSubscriptions($messageArray, $connection);
if ($this->shouldRejectMessage($channel, $connection, $messageArray)) {
2025-12-05 19:53:52 +00:00
return;
}
2025-01-16 07:54:02 +00:00
$this->authenticateConnection($connection, $channel, $messageArray);
2025-01-16 07:54:02 +00:00
// Only log in debug mode to reduce I/O
if (config('app.debug')) {
Log::channel('websocket')->debug('[' . $connection->socketId . ']@' . $channel->getName() . ' | ' . $payload);
}
2025-01-16 07:54:02 +00:00
if ($this->handlePusherEvent($messageArray, $connection)) {
return;
}
2025-01-16 07:54:02 +00:00
$this->forkAndProcessMessage($connection, $channel, $messageArray);
}
/**
* Handle pusher protocol messages (formerly in PusherMessageFactory)
* Inlined for performance - avoids object creation
*/
private function handlePusherProtocolMessage(
MessageInterface $message,
ConnectionInterface $connection,
array $messageArray
): void {
$event = $messageArray['event'] ?? '';
2025-01-16 07:54:02 +00:00
// Fast check - most messages don't start with 'pusher' or 'client-'
$firstChar = $event[0] ?? '';
if ($firstChar !== 'p' && $firstChar !== 'c') {
return;
}
2025-01-16 07:54:02 +00:00
// Check for client- messages
if (strpos($event, 'client-') === 0) {
if (!$connection->app->clientMessagesEnabled) {
2025-12-05 19:53:52 +00:00
return;
2025-09-18 16:07:15 +00:00
}
$channelName = $messageArray['channel'] ?? null;
if (!$channelName) {
2025-12-05 19:53:52 +00:00
return;
2025-01-16 07:54:02 +00:00
}
$channel = $this->channelManager->find($connection->app->id, $channelName);
if ($channel) {
$channel->broadcastToEveryoneExcept(
(object) $messageArray,
$connection->socketId,
$connection->app->id
);
}
return;
}
// Check for pusher: or pusher. messages (subscribe/unsubscribe handled elsewhere)
// This is handled by handleChannelSubscriptions for subscribe/unsubscribe
}
public function onOpen(ConnectionInterface $connection): void
{
if (!$this->connectionCanBeMade($connection)) {
$connection->close();
return;
}
try {
$this->setupConnectionAddress($connection);
$this->verifyAppKey($connection);
$this->verifyOrigin($connection);
$this->limitConcurrentConnections($connection);
$this->generateSocketId($connection);
$this->establishConnection($connection);
$this->initializeAppConnection($connection);
} catch (UnknownAppKey $e) {
Log::channel('websocket')->error('Root level error: ' . $e->getMessage(), [
'file' => $e->getFile(),
'line' => $e->getLine(),
'trace' => $e->getTraceAsString(),
]);
2025-12-02 17:33:52 +00:00
}
2025-01-16 07:54:02 +00:00
}
/**
* Handle the websocket close.
*/
2025-01-17 09:45:53 +00:00
public function onClose(ConnectionInterface $connection): void
2025-01-16 07:54:02 +00:00
{
2025-10-15 07:27:37 +00:00
$this->authenticateConnection($connection, null);
2025-12-05 19:53:52 +00:00
if (isset($connection->remoteAddress)) {
request()->server->set('REMOTE_ADDR', $connection->remoteAddress);
}
2025-12-05 19:53:52 +00:00
$this->cleanupChannelConnections($connection);
$this->finalizeConnectionClose($connection);
}
protected function setupConnectionAddress(ConnectionInterface $connection): void
{
$connection->remoteAddress = trim(
explode(
',',
$connection->httpRequest->getHeaderLine('X-Forwarded-For')
)[0] ?? $connection->remoteAddress
);
request()->server->set('REMOTE_ADDR', $connection->remoteAddress);
Log::channel('websocket')->info('WS onOpen IP: ' . $connection->remoteAddress);
}
protected function initializeAppConnection(ConnectionInterface $connection): void
{
if (!isset($connection->app)) {
return;
}
// Initialize lastPongedAt with unix timestamp (faster than Carbon)
$connection->lastPongedAt = time();
2025-12-05 19:53:52 +00:00
$this->channelManager->subscribeToApp($connection->app->id);
NewConnection::dispatch(
$connection->app->id,
$connection->socketId
);
}
protected function shouldRejectMessage(?Channel $channel, ConnectionInterface $connection, array $message): bool
{
$event = $message['event'] ?? '';
$isUnsubscribe = $event === 'pusher:unsubscribe' || $event === 'pusher.unsubscribe';
2025-12-05 19:53:52 +00:00
if (!$channel?->hasConnection($connection) && !$isUnsubscribe) {
$connection->send(json_encode([
'event' => $event . ':error',
2025-12-05 19:53:52 +00:00
'data' => [
'message' => 'Subscription not established',
'meta' => $message,
],
]));
return true;
}
if (!$channel) {
$connection->send(json_encode([
'event' => $message['event'] . ':error',
'data' => [
'message' => 'Channel not found',
'meta' => $message,
],
]));
return true;
}
return false;
}
protected function handlePusherEvent(array $message, ConnectionInterface $connection): bool
{
if (!str_contains($message['event'], 'pusher')) {
return false;
}
$connection->send(json_encode([
'event' => $message['event'] . ':response',
'data' => [
'message' => 'Success',
],
]));
return true;
}
protected function forkAndProcessMessage(
ConnectionInterface $connection,
Channel $channel,
array $message
): void {
// Generate unique request ID BEFORE forking to avoid race conditions
// Using uniqid with more_entropy + random_bytes for guaranteed uniqueness
$requestId = uniqid('req_', true) . '_' . bin2hex(random_bytes(4));
2025-12-05 19:53:52 +00:00
$pid = pcntl_fork();
if ($pid === -1) {
Log::error('Fork error');
return;
}
if ($pid === 0) {
$this->processMessageInChild($connection, $channel, $message, $requestId);
2025-12-05 19:53:52 +00:00
exit(0);
}
$this->addDataCheckLoop($connection, $message, $requestId);
2025-12-05 19:53:52 +00:00
}
protected function processMessageInChild(
ConnectionInterface $connection,
Channel $channel,
array $message,
string $requestId
2025-12-05 19:53:52 +00:00
): void {
try {
DB::disconnect();
DB::reconnect();
$this->setRequest($message, $connection);
$mock = new MockConnection($connection, $requestId);
2025-12-05 19:53:52 +00:00
Controller::controll_message(
$mock,
$channel,
$message,
$this->channelManager
);
\Illuminate\Container\Container::getInstance()
->make(\Illuminate\Support\Defer\DeferredCallbackCollection::class)
->invokeWhen(fn($callback) => true);
} catch (Exception $e) {
$mock->send(json_encode([
'event' => $message['event'] . ':error',
'data' => [
'message' => $e->getMessage(),
],
]));
if (app()->bound('sentry')) {
app('sentry')->captureException($e);
}
}
}
protected function handleMessageError(\Throwable $e): void
{
Log::channel('websocket')->error('onMessage unhandled error: ' . $e->getMessage(), [
'file' => $e->getFile(),
'line' => $e->getLine(),
'trace' => $e->getTraceAsString(),
]);
if (app()->bound('sentry')) {
app('sentry')->captureException($e);
}
}
protected function cleanupChannelConnections(ConnectionInterface $connection): void
{
$cacheUpdates = [];
$cacheDeletes = ['ws_socket_auth_' . $connection->socketId];
2025-12-05 20:48:23 +00:00
$socketId = $connection->socketId;
2025-12-05 19:53:52 +00:00
2025-01-16 07:54:02 +00:00
foreach ($this->channel_connections as $channel => $connections) {
2025-12-05 20:48:23 +00:00
if (!isset($connections[$socketId])) {
2025-12-05 19:53:52 +00:00
continue;
2025-01-16 07:54:02 +00:00
}
2025-12-05 20:48:23 +00:00
unset($this->channel_connections[$channel][$socketId]);
2025-12-05 19:53:52 +00:00
if (empty($this->channel_connections[$channel])) {
2025-01-16 07:54:02 +00:00
unset($this->channel_connections[$channel]);
2025-12-05 19:53:52 +00:00
$cacheDeletes[] = 'ws_channel_connections_' . $channel;
continue;
2025-01-16 07:54:02 +00:00
}
2025-12-05 20:48:23 +00:00
// Pre-compute array_keys once per channel
2025-12-05 19:53:52 +00:00
$cacheUpdates['ws_channel_connections_' . $channel] = array_keys($this->channel_connections[$channel]);
}
2025-01-16 07:54:02 +00:00
2025-12-05 20:48:23 +00:00
// Pre-compute active channels once
$activeChannels = array_keys($this->channel_connections);
$cacheUpdates['ws_active_channels'] = $activeChannels;
2025-01-16 07:54:02 +00:00
2025-12-05 20:48:23 +00:00
// Batch read authed_users - we'll update it in the same batch
2025-12-05 19:53:52 +00:00
$authed_users = cache()->get('ws_socket_authed_users') ?? [];
2025-12-05 20:48:23 +00:00
unset($authed_users[$socketId]);
2025-12-05 19:53:52 +00:00
$cacheUpdates['ws_socket_authed_users'] = $authed_users;
2025-09-15 08:20:13 +00:00
2025-12-05 20:48:23 +00:00
// Single batched write and delete operation - MASSIVE latency improvement
if (!empty($cacheUpdates)) {
cache()->setMultiple($cacheUpdates);
}
if (!empty($cacheDeletes)) {
cache()->deleteMultiple($cacheDeletes);
}
2025-10-15 07:27:37 +00:00
2025-12-05 20:48:23 +00:00
// Note: Removed redundant WebsocketService::clearUserAuthed() call
// as we already handle all cache operations above in a single batch
2025-12-05 19:53:52 +00:00
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
protected function finalizeConnectionClose(ConnectionInterface $connection): void
{
2025-01-16 07:54:02 +00:00
$this->channelManager
->unsubscribeFromAllChannels($connection)
2025-01-17 09:45:53 +00:00
->then(function (bool $unsubscribed) use ($connection): void {
2025-12-05 19:53:52 +00:00
if (!isset($connection->app)) {
return;
2025-01-16 07:54:02 +00:00
}
2025-12-05 19:53:52 +00:00
$this->channelManager->unsubscribeFromApp($connection->app->id);
ConnectionClosed::dispatch($connection->app->id, $connection->socketId);
cache()->forget('ws_connection_' . $connection->socketId);
2025-01-16 07:54:02 +00:00
});
}
/**
* Handle the websocket errors.
*
* @param WebSocketException $exception
*/
2025-01-17 09:45:53 +00:00
public function onError(ConnectionInterface $connection, Exception $exception): void
2025-01-16 07:54:02 +00:00
{
if ($exception instanceof ExceptionsWebSocketException) {
$connection->send(json_encode(
$exception->getPayload()
));
}
}
/**
* Check if the connection can be made for the
* current server instance.
*/
2025-01-17 09:45:53 +00:00
protected function connectionCanBeMade(ConnectionInterface $connection): bool
2025-01-16 07:54:02 +00:00
{
return $this->channelManager->acceptsNewConnections();
}
/**
* Verify the app key validity.
*
* @return $this
*/
protected function verifyAppKey(ConnectionInterface $connection)
{
$query = QueryParameters::create($connection->httpRequest);
$appKey = $query->get('appKey');
if (! $app = App::findByKey($appKey)) {
throw new UnknownAppKey($appKey);
}
2025-01-17 09:45:53 +00:00
$app->then(function ($app) use ($connection) {
$connection->app = $app;
});
2025-01-16 07:54:02 +00:00
return $this;
}
/**
* Verify the origin.
*
* @return $this
*/
protected function verifyOrigin(ConnectionInterface $connection)
{
if (! $connection->app->allowedOrigins) {
return $this;
}
$header = (string) ($connection->httpRequest->getHeader('Origin')[0] ?? null);
$origin = parse_url($header, PHP_URL_HOST) ?: $header;
if (! $header || ! in_array($origin, $connection->app->allowedOrigins)) {
throw new OriginNotAllowed($connection->app->key);
}
return $this;
}
/**
* Limit the connections count by the app.
*
* @return $this
*/
protected function limitConcurrentConnections(ConnectionInterface $connection)
{
if (! is_null($capacity = $connection->app->capacity)) {
$this->channelManager
->getGlobalConnectionsCount($connection->app->id)
2025-01-17 09:45:53 +00:00
->then(function ($connectionsCount) use ($capacity, $connection): void {
2025-01-16 07:54:02 +00:00
if ($connectionsCount >= $capacity) {
$exception = new ConnectionsOverCapacity;
$payload = json_encode($exception->getPayload());
tap($connection)->send($payload)->close();
}
});
}
return $this;
}
/**
* Create a socket id.
*
* @return $this
*/
protected function generateSocketId(ConnectionInterface $connection)
{
$socketId = sprintf('%d.%d', random_int(1, 1000000000), random_int(1, 1000000000));
$connection->socketId = $socketId;
return $this;
}
/**
* Establish connection with the client.
*
* @return $this
*/
protected function establishConnection(ConnectionInterface $connection)
{
$connection->send(json_encode([
2025-01-18 16:06:52 +00:00
'event' => 'pusher.connection_established',
2025-01-16 07:54:02 +00:00
'data' => json_encode([
'socket_id' => $connection->socketId,
'activity_timeout' => 30,
]),
]));
return $this;
}
2025-09-14 13:00:27 +00:00
protected function get_connection_channel(&$connection, &$message): ?Channel
2025-01-16 07:54:02 +00:00
{
// Put channel on its place
2025-12-05 19:53:52 +00:00
if (! isset($message['channel']) && isset($message['data']['channel'])) {
2025-01-16 07:54:02 +00:00
$message['channel'] = $message['data']['channel'];
unset($message['data']['channel']);
}
2025-09-16 06:54:13 +00:00
$this->channelManager->findOrCreate(
2025-01-16 07:54:02 +00:00
$connection->app->id,
$message['channel']
);
return $this->channelManager->find(
$connection->app->id,
$message['channel']
);
}
2025-09-18 13:56:13 +00:00
protected function handleChannelSubscriptions($message, $connection): ?Channel
2025-01-16 07:54:02 +00:00
{
2025-09-15 14:22:59 +00:00
$channel = $this->get_connection_channel($connection, $message);
2025-12-05 19:53:52 +00:00
$channel_name = $channel?->getName();
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
if (!$channel_name || !$channel) {
2025-09-15 14:22:59 +00:00
return null;
2025-09-15 12:29:07 +00:00
}
2025-12-05 19:53:52 +00:00
$eventLower = strtolower($message['event']);
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
if ($eventLower === 'pusher.subscribe' || $eventLower === 'pusher:subscribe') {
$this->handleSubscription($channel, $channel_name, $connection, $message);
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
if (str_contains($message['event'], '.unsubscribe')) {
$this->handleUnsubscription($channel, $channel_name, $connection);
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
return $channel;
}
2025-09-15 12:29:07 +00:00
2025-12-05 19:53:52 +00:00
protected function handleSubscription(
Channel $channel,
string $channel_name,
ConnectionInterface $connection,
array $message
): void {
2025-12-05 20:48:23 +00:00
$socketId = $connection->socketId;
2025-12-05 19:53:52 +00:00
if (!isset($this->channel_connections[$channel_name])) {
$this->channel_connections[$channel_name] = [];
2025-01-16 07:54:02 +00:00
}
2025-12-05 20:48:23 +00:00
if (!isset($this->channel_connections[$channel_name][$socketId])) {
$this->channel_connections[$channel_name][$socketId] = true;
2025-01-16 07:54:02 +00:00
2025-12-05 20:48:23 +00:00
// Only update cache if connection was actually added (avoid redundant writes)
// Pre-compute array_keys once for both updates
$channelSockets = array_keys($this->channel_connections[$channel_name]);
$activeChannels = array_keys($this->channel_connections);
// Buffer these writes - they can be batched with other subscriptions
$this->bufferCacheWrite('ws_channel_connections_' . $channel_name, $channelSockets);
$this->bufferCacheWrite('ws_active_channels', $activeChannels);
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
if ($channel->hasConnection($connection)) {
return;
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
try {
$channel->subscribe($connection, (object) $message);
} catch (\Throwable $e) {
// Silently handle subscription errors
}
}
2025-09-16 09:20:48 +00:00
2025-12-05 19:53:52 +00:00
protected function handleUnsubscription(
Channel $channel,
string $channel_name,
ConnectionInterface $connection
): void {
2025-12-05 20:48:23 +00:00
$socketId = $connection->socketId;
if (isset($this->channel_connections[$channel_name][$socketId])) {
unset($this->channel_connections[$channel_name][$socketId]);
// Pre-compute active channels once
$activeChannels = array_keys($this->channel_connections);
if (empty($this->channel_connections[$channel_name])) {
unset($this->channel_connections[$channel_name]);
// Buffer delete and update - can be batched
$this->bufferCacheDelete('ws_channel_connections_' . $channel_name);
$this->bufferCacheWrite('ws_active_channels', $activeChannels);
} else {
// Pre-compute channel sockets once
$channelSockets = array_keys($this->channel_connections[$channel_name]);
// Buffer these writes
$this->bufferCacheWrite('ws_channel_connections_' . $channel_name, $channelSockets);
$this->bufferCacheWrite('ws_active_channels', $activeChannels);
}
2025-12-05 19:53:52 +00:00
}
$channel->unsubscribe($connection);
2025-01-16 07:54:02 +00:00
}
protected function setRequest($message, $connection)
{
foreach (request()->keys() as $key) {
request()->offsetUnset($key);
}
2025-12-05 19:53:52 +00:00
request()->merge($message['data'] ?? []);
2025-01-16 07:54:02 +00:00
}
protected function authenticateConnection(
ConnectionInterface $connection,
PrivateChannel|Channel|PresenceChannel|null $channel,
2025-10-15 07:45:04 +00:00
$message = []
2025-01-16 07:54:02 +00:00
) {
2025-12-05 19:53:52 +00:00
$this->loadCachedAuth($connection, $channel);
$this->ensureUserIsSet($connection, $channel);
$this->updateAuthState($connection);
$this->cacheAuthenticatedUser($connection);
$this->scheduleLogout();
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
protected function loadCachedAuth(ConnectionInterface $connection, $channel): void
{
if (isset($connection->auth)) {
return;
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
if (!$connection->socketId) {
return;
2025-01-16 07:54:02 +00:00
}
2025-12-05 19:53:52 +00:00
$cached_auth = cache()->get('socket_' . $connection->socketId);
if (!$cached_auth || !isset($cached_auth['type'])) {
return;
2025-01-16 07:54:02 +00:00
}
2025-12-05 19:53:52 +00:00
$connection->user = $cached_auth['type']::find($cached_auth['id']);
if ($channel) {
$channel->saveConnection($connection);
}
}
protected function ensureUserIsSet(ConnectionInterface $connection, $channel): void
{
if (isset($connection->user) && $connection->user) {
return;
}
$connection->user = false;
if ($channel) {
$channel->saveConnection($connection);
}
}
protected function updateAuthState(ConnectionInterface $connection): void
{
$connection->user
2025-05-08 08:54:11 +00:00
? Auth::login($connection->user)
: Auth::logout();
2025-12-05 19:53:52 +00:00
}
2025-05-08 08:54:11 +00:00
2025-12-05 19:53:52 +00:00
protected function cacheAuthenticatedUser(ConnectionInterface $connection): void
{
if (!Auth::user()) {
return;
}
2025-09-15 08:20:13 +00:00
2025-12-05 19:53:52 +00:00
/** @var \App\Models\User */
$user = Auth::user();
$user->refresh();
2025-09-15 08:20:13 +00:00
2025-12-05 20:48:23 +00:00
$socketId = $connection->socketId;
2025-09-15 08:20:13 +00:00
2025-12-05 20:48:23 +00:00
// Batch all auth cache operations into a single read + single write
2025-12-05 19:53:52 +00:00
$authed_users = cache()->get('ws_socket_authed_users') ?? [];
2025-12-05 20:48:23 +00:00
$authed_users[$socketId] = $user->id;
2025-10-15 07:35:07 +00:00
2025-12-05 20:48:23 +00:00
// Single batched cache write - reduces 3 operations to 1
cache()->setMultiple([
'ws_socket_auth_' . $socketId => $user,
'ws_socket_authed_users' => $authed_users
]);
// Note: Removed redundant WebsocketService::setUserAuthed() call
// as we already handle all cache operations above in a single batch
2025-12-05 19:53:52 +00:00
}
2025-10-15 07:27:37 +00:00
2025-12-05 19:53:52 +00:00
protected function scheduleLogout(): void
{
2025-10-15 07:27:37 +00:00
$this->channelManager->loop->futureTick(function () {
Auth::logout();
});
2025-01-16 07:54:02 +00:00
}
2025-12-05 20:48:23 +00:00
/**
* Add cache operation to write buffer for batching
*/
protected function bufferCacheWrite(string $key, $value): void
{
$this->cacheWriteBuffer[$key] = $value;
$this->scheduleCacheFlush();
}
/**
* Add cache deletion to buffer for batching
*/
protected function bufferCacheDelete(string $key): void
{
$this->cacheDeleteBuffer[] = $key;
unset($this->cacheWriteBuffer[$key]); // Remove from write buffer if exists
$this->scheduleCacheFlush();
}
/**
* Schedule cache flush on next event loop tick
* Multiple rapid requests will be batched into single I/O operation
*/
protected function scheduleCacheFlush(): void
{
if ($this->cacheBufferScheduled) {
return;
}
$this->cacheBufferScheduled = true;
$this->channelManager->loop->futureTick(function () {
$this->flushCacheBuffer();
});
}
/**
* Flush cache buffer - performs all pending operations in single batch
* This is the key optimization: N operations -> 2 I/O calls (1 write, 1 delete)
*/
protected function flushCacheBuffer(): void
{
if (!empty($this->cacheWriteBuffer)) {
cache()->setMultiple($this->cacheWriteBuffer);
$this->cacheWriteBuffer = [];
}
if (!empty($this->cacheDeleteBuffer)) {
cache()->deleteMultiple(array_unique($this->cacheDeleteBuffer));
$this->cacheDeleteBuffer = [];
}
$this->cacheBufferScheduled = false;
}
/**
* Force immediate cache flush (use for critical operations)
*/
protected function flushCacheBufferImmediate(): void
{
$this->flushCacheBuffer();
$this->cacheBufferScheduled = false;
}
2025-01-16 07:54:02 +00:00
private function addDataCheckLoop(
$connection,
$message,
string $requestId,
2025-01-16 07:54:02 +00:00
$optional = false,
int $iteration = 0
2025-01-16 07:54:02 +00:00
) {
$iterationKey = $requestId . ($iteration > 0 ? '_' . $iteration : '');
$cacheKeyStart = 'dedicated_start_' . $iterationKey;
IpcCache::put($cacheKeyStart, microtime(true), 100);
2025-01-16 07:54:02 +00:00
$this->channelManager->loop->addPeriodicTimer(0.01, function ($timer) use (
$cacheKeyStart,
$iterationKey,
2025-01-16 07:54:02 +00:00
$message,
$requestId,
2025-01-16 07:54:02 +00:00
$connection,
$optional,
$iteration
) {
2025-12-05 19:53:52 +00:00
$this->checkDataLoopIteration(
$timer,
$cacheKeyStart,
2025-12-05 19:53:52 +00:00
$message,
$iterationKey,
$requestId,
2025-12-05 19:53:52 +00:00
$connection,
$optional,
$iteration
);
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
pcntl_waitpid(-1, $status, WNOHANG);
});
}
2025-01-16 07:54:02 +00:00
2025-12-05 19:53:52 +00:00
protected function checkDataLoopIteration(
$timer,
string $cacheKeyStart,
2025-12-05 19:53:52 +00:00
array $message,
string $iterationKey,
string $requestId,
2025-12-05 19:53:52 +00:00
$connection,
bool $optional,
int $iteration
2025-12-05 19:53:52 +00:00
): void {
$cacheKeyData = 'dedicated_data_' . $iterationKey;
$cacheKeyDone = 'dedicated_data_' . $iterationKey . '_done';
$cacheKeyComplete = 'dedicated_data_' . $iterationKey . '_complete';
2025-12-05 19:53:52 +00:00
if ($this->handleTimeout($timer, $cacheKeyStart, $cacheKeyComplete, $message, $connection, $optional)) {
2025-12-05 19:53:52 +00:00
return;
}
if (!IpcCache::has($cacheKeyDone)) {
2025-12-05 19:53:52 +00:00
return;
}
// Clean up cache entries for this iteration before processing
// This prevents memory leaks and stale data issues
$this->cleanupIterationCache($iterationKey);
$this->scheduleNextIteration($connection, $message, $requestId, $iteration);
$this->processAndSendData($connection, $cacheKeyData);
2025-12-05 19:53:52 +00:00
$this->channelManager->loop->cancelTimer($timer);
}
/**
* Clean up cache entries for a completed iteration
*/
protected function cleanupIterationCache(string $iterationKey): void
{
$keysToDelete = [
'dedicated_start_' . $iterationKey,
'dedicated_data_' . $iterationKey . '_done',
// Note: We don't delete 'dedicated_data_' here as we need it for processAndSendData
// It will expire naturally after 60 seconds
];
IpcCache::forgetMultiple($keysToDelete);
}
2025-12-05 19:53:52 +00:00
protected function handleTimeout(
$timer,
string $cacheKeyStart,
string $cacheKeyComplete,
2025-12-05 19:53:52 +00:00
array $message,
$connection,
bool $optional
): bool {
$startTime = IpcCache::get($cacheKeyStart);
if ($startTime === null) {
2025-12-05 19:53:52 +00:00
return false;
}
$diff = microtime(true) - ((float) $startTime);
2025-12-05 19:53:52 +00:00
if ($diff <= 60) {
return false;
}
if (!$optional) {
$connection->send(json_encode([
'event' => $message['event'] . ':error',
'data' => [
'message' => $message['event'] . ' timeout',
'diff' => $diff,
],
]));
}
$this->channelManager->loop->cancelTimer($timer);
IpcCache::put($cacheKeyComplete, true, 360);
2025-12-05 19:53:52 +00:00
return true;
}
protected function scheduleNextIteration($connection, array $message, string $requestId, int $iteration): void
2025-12-05 19:53:52 +00:00
{
$nextIteration = $iteration + 1;
$this->addDataCheckLoop($connection, $message, $requestId, true, $nextIteration);
2025-12-05 19:53:52 +00:00
}
protected function processAndSendData($connection, string $cacheKeyData): void
2025-12-05 19:53:52 +00:00
{
$sending = IpcCache::get($cacheKeyData);
// Clean up the data cache key immediately after reading
IpcCache::forget($cacheKeyData);
if (!$sending) {
return;
}
2025-12-05 19:53:52 +00:00
$bm = json_decode($sending, true);
if (isset($bm['broadcast']) && $bm['broadcast']) {
$this->broadcast(
$connection->app->id,
$bm['data'] ?? null,
$bm['event'] ?? null,
$bm['channel'] ?? null,
$bm['including_self'] ?? false,
$connection
);
return;
}
if (isset($bm['whisper']) && $bm['whisper']) {
$this->whisper(
$connection->app->id,
$bm['data'] ?? null,
$bm['event'] ?? null,
$bm['socket_ids'] ?? [],
$bm['channel'] ?? null,
);
return;
}
$connection->send($sending);
2025-01-16 07:54:02 +00:00
}
2025-09-13 17:33:29 +00:00
public function broadcast(
string $appId,
mixed $payload,
?string $event = null,
?string $channel = null,
bool $including_self = false,
$connection = null
2025-09-14 13:00:27 +00:00
): void {
2025-09-13 17:33:29 +00:00
2025-09-14 13:00:27 +00:00
$channel = $this->channelManager->findOrCreate($appId, $channel);
$p = [
'event' => ($event ?? $event),
'data' => $payload,
'channel' => $channel->getName(),
];
2025-09-13 17:33:29 +00:00
foreach ($channel->getConnections() as $channel_conection) {
2025-09-16 08:58:54 +00:00
if ($channel_conection->socketId !== $connection->socketId) {
2025-09-14 13:00:27 +00:00
$channel_conection->send(json_encode($p));
2025-09-13 17:33:29 +00:00
}
if ($including_self) {
2025-09-14 13:00:27 +00:00
$connection->send(json_encode($p));
2025-09-13 17:33:29 +00:00
}
}
}
2025-09-15 12:29:07 +00:00
public function whisper(
string $appId,
mixed $payload,
?string $event = null,
array $socketIds = [],
?string $channel = null
): void {
$channel = $this->channelManager->findOrCreate($appId, $channel);
$p = [
'event' => ($event ?? $event),
'data' => $payload,
'channel' => $channel->getName(),
];
2025-12-05 19:53:52 +00:00
$socketIdLookup = array_flip($socketIds);
2025-09-15 12:29:07 +00:00
foreach ($channel->getConnections() as $channel_conection) {
2025-12-05 19:53:52 +00:00
if (isset($socketIdLookup[$channel_conection->socketId])) {
2025-09-15 12:29:07 +00:00
$channel_conection->send(json_encode($p));
}
}
}
2025-01-16 07:54:02 +00:00
}