pogo / websocket
Pogo WebSocket Driver
Requires
- laravel/octane: ^2.0
- pusher/pusher-php-server: ^7.0
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.0
- mockery/mockery: ^1.0
- phpstan/phpstan: ^2.0
- phpunit/phpunit: ^13.0
- dev-main
- v0.0.9
- v0.0.8
- v0.0.7
- v0.0.6
- v0.0.5
- v0.0.4
- v0.0.3
- v0.0.2
- v0.0.1
- dev-renovate/laravel-octane-2.x
- dev-renovate/friendsofphp-php-cs-fixer-3.x
- dev-renovate/php-8.x
- dev-renovate/phpstan-packages
- dev-renovate/phpunit-phpunit-13.x
- dev-renovate/github.com-dunglas-frankenphp-caddy-1.x
- dev-renovate/github.com-caddyserver-caddy-v2-2.x
- dev-renovate/github.com-dunglas-frankenphp-1.x
- dev-renovate/go-1.x
- dev-renovate/golangci-golangci-lint-2.x
- dev-renovate/github.com-redis-go-redis-v9-9.x
This package is auto-updated.
Last update: 2026-06-09 18:06:36 UTC
README
An experimental native real-time module for FrankenPHP applications.
- A Caddy module that embeds a scalable, Pusher-compatible WebSocket server directly into the FrankenPHP binary
- CGO-exported functions
pogo_websocket_publishandpogo_websocket_broadcast_multiallow PHP to broadcast messages instantly and return native status codes for precise failures. - The Caddy module uses FrankenPHP's
ExtensionWorkerAPI to invoke a dedicated pool of PHP threads for authentication, avoiding network overhead.
Repository Layout
This repository is intentionally limited to the websocket extension and its package-level validation:
module/: the Go/Caddy/FrankenPHP websocket module.lib/: the Laravel broadcasting driver.module/tests/: module unit, integration, and low-level performance tests.
Full application showcases belong in pogoShowcase. Keep this repository focused on code that ships with, tests, or measures the websocket package.
Features
- Pusher-Compatible Protocol Subset: Supports public, private, and presence channels, client events, and user authentication for Echo/Pusher-style clients.
- Benchmark Harness: A reproducible benchmark setup is available in the
benchmarks/workspace. Current results are experimental and topology-specific, so this README intentionally does not quote headline performance numbers. - Prepared Broadcast Fanout: Optimizes CPU usage by encoding broadcast payloads once per channel fanout.
- DoS Protection: Built-in Token Bucket Rate Limiting, Handshake Throttling, and Circuit Breakers for PHP Auth.
- Horizontal Scaling: Redis Pub/Sub support for multi-node clusters with at-most-once delivery semantics.
Production status
Pogo WebSocket is experimental and its API may change. It is suitable for demos, local testing, and controlled evaluation of a FrankenPHP-native WebSocket runtime.
Do not yet present it as a production substitute for Laravel Reverb, Pusher, or other hosted realtime systems. Validate behavior, benchmarks, and failure modes for your topology before using it with production traffic.
Supported Pusher protocol behavior is intentionally scoped: connection
establishment, ping/pong, public/private/presence subscriptions, client events on
private and presence channels, and pusher:signin. Features such as durable
delivery, replay, encrypted channels, watchlists, and hosted Pusher management
APIs are not implemented.
Installation
Step 1: Docker
Build a FrankenPHP binary that includes Pogo WebSocket with xcaddy. See the official
FrankenPHP Docker documentation for the
base image details.
Example Dockerfile from this repository root:
FROM dunglas/frankenphp:builder AS builder COPY --from=caddy:builder /usr/bin/xcaddy /usr/bin/xcaddy COPY . /src/websocket RUN CGO_ENABLED=1 \ XCADDY_SETCAP=1 \ XCADDY_GO_BUILD_FLAGS="-ldflags='-w -s' -tags=nobadger,nomysql,nopgx" \ CGO_CFLAGS="$(php-config --includes)" \ CGO_LDFLAGS="$(php-config --ldflags) $(php-config --libs)" \ xcaddy build \ --output /usr/local/bin/frankenphp \ --with github.com/dunglas/frankenphp=./ \ --with github.com/dunglas/frankenphp/caddy=./caddy \ --with github.com/dunglas/caddy-cbrotli \ --with github.com/y-l-g/websocket/module=./src/websocket/module FROM dunglas/frankenphp AS runner COPY --from=builder /usr/local/bin/frankenphp /usr/local/bin/frankenphp
Then copy your app and Caddyfile into the runner image as usual.
Step 2: Install the Laravel Broadcast Driver
composer require pogo/websocket php artisan pogo:ws-install
Configuration
Configure the module within your Caddyfile at the root of your laravel project (this exemple is an adapted copy of the octane Caddyfile, it will work with php artisan octane:frankenphp --caddyfile=Caddyfile).
{
frankenphp {
worker {
file "public/frankenphp-worker.php"
}
}
order pogo_websocket before php_server
}
:8080 {
log {
level info
}
format filter {
wrap json
fields {
uri query {
replace authorization REDACTED
}
}
}
route /app/* {
pogo_websocket {
app_id pogo-app
app_secret {$WS_APP_SECRET}
auth_path /pogo/auth
auth_script public/websocket-worker.php
webhook_secret {$POGO_WEBHOOK_SECRET}
allowed_origins https://app.example.com https://admin.example.com
handshake_rate 100 # New connection attempts per second (Default: 100)
handshake_burst 50 # Burst allowance (Default: 50)
max_connections 10000 # Max concurrent clients
max_auth_body 16384 # Max PHP Auth response size (bytes)
max_concurrent_auth 100 # Max concurrent PHP Auth requests (DoS Protection)
broker_queue_size 1024 # Internal broker queue before publish fails fast
shard_queue_size 1024 # Per-shard control/broadcast queue
num_workers 2 # Number of PHP workers dedicated to Auth
num_shards 8 # Internal sharding (Default: 2 * CPU Cores)
ping_period 54s # Server Ping interval
pong_wait 60s # Client Pong timeout
write_wait 10s # Socket write timeout
shutdown_timeout 10s # Max graceful shutdown wait
# redis_host localhost:6379
}
}
route {
root * public
encode zstd br gzip
php_server {
index frankenphp-worker.php
try_files {path} frankenphp-worker.php
resolve_root_symlink
}
}
}
By default, WebSocket upgrades accept requests without an Origin header and browser
requests whose Origin host matches the request host. Configure allowed_origins
when your frontend connects from a different origin; entries must be exact
http:// or https:// origins, including the port when one is used.
Handshake throttling is applied per direct remote IP address. If FrankenPHP sits behind a reverse proxy or load balancer that hides client IPs, enforce per-client rate limits at that proxy layer as well.
Private and presence channel auth accepts standard Pusher signatures:
socket_id:channel for private channels and
socket_id:channel:channel_data for presence channels. If a client omits
auth, the module falls back to the configured FrankenPHP auth worker and
validates the worker's returned signature before subscribing the client.
Native publish functions return 0 on success. Nonzero status codes indicate:
1 hub missing, 2 channel too long, 3 event too long, 4 payload too large,
5 invalid payload JSON, 6 broker publish failed, and 7 invalid multi-channel
JSON, 8 broker queue full, and 9 shard queue full. Success means the message
was accepted by the broker and shard queue; delivery to every connected client is
at-most-once and may still fail for slow clients with full outbound queues. The
Laravel broadcaster turns native failures into BroadcastException.
Fill your .env
BROADCAST_CONNECTION=pogo WS_APP_ID=pogo-app WS_APP_SECRET=change-me-to-a-long-random-secret POGO_WEBHOOK_SECRET=change-me-to-a-different-random-secret VITE_POGO_APP_KEY="${WS_APP_ID}" VITE_POGO_HOST=localhost #your site adress VITE_POGO_PORT=80 #your site port VITE_POGO_WSS_PORT=443 #your site port
For multiple apps in one process, an explicit app_secret in the Caddyfile wins.
If it is omitted, the module checks WS_APP_SECRET_<APP_ID> where non
alphanumeric characters in the app id are converted to _, then falls back to
WS_APP_SECRET.
Start octane (frankenphp must be compiled with pogo_websocket).
php artisan octane:start --caddyfile=Caddyfile # or frankenphp run --caddyfile=Caddyfile # or frankenphp run --config Caddyfile
Metrics
Prometheus metrics are available at http://localhost:2019/metrics (Caddy Admin):
Set POGO_WS_HOT_PATH_METRICS=true to enable detailed per-message fanout, queue-depth, and write-duration histograms.
| Metric | Type | Description |
|---|---|---|
pogo_websocket_connections_active |
Gauge | Active TCP connections. |
pogo_websocket_messages_total |
Counter | Total messages broadcasted. |
pogo_websocket_auth_failures_total |
Counter | Failed auths (labels: concurrency_limit, worker_error). |
pogo_websocket_circuit_breaker_open_total |
Counter | Requests rejected by Circuit Breaker. |
pogo_websocket_broker_dropped_messages_total |
Counter | Messages dropped due to internal backpressure. |
pogo_websocket_subscriptions_total |
Counter | Total active subscriptions. |
pogo_websocket_auth_duration_seconds |
Histogram | Latency of the PHP Auth Worker. |
pogo_websocket_client_dropped_messages_total |
Counter | Messages dropped due to full client buffer. |
pogo_websocket_publish_failures_total |
Counter | Failed publish attempts by app and reason. |
pogo_websocket_webhook_dropped_total |
Counter | Webhook notifications dropped by reason. |
Reliability and security notes
- Redis clustering uses Redis Pub/Sub. Messages are not persisted, replayed, or acknowledged across nodes; messages can be lost during Redis outages, reconnects, or local overload.
/pogo/authand/pogo/user-authare CSRF-exempt because they are called by WebSocket clients. Protect them with normal Laravel authentication, same-site/session settings, and the WebSocketallowed_originspolicy.- Webhook notifications are best-effort and may be dropped when the webhook queue is full or the module is shutting down.
Troubleshooting
- 4100 Over Capacity: Increase
max_connectionsin Caddyfile. - 4009 Connection Unauthorized: Check
app_secretmatchesWS_APP_SECRET. - Too Many Requests: Tune
handshake_rateif legitimate traffic is being blocked.