A lightweight TCP, UDP, TLS and SSL socket toolkit for PHP 8.1+. Both server
and client sides share a clean, typed API built around enums and a small
Channel strategy so each transport plugs in without switch ladders.
use InitPHP\Socket\Socket;
use InitPHP\Socket\Enum\Transport;
$server = Socket::server(Transport::TCP, '127.0.0.1', 8080);
$server->listen();
$server->live(function ($srv, $conn) {
$message = $conn->read(1024);
$conn->write("echo: {$message}");
});- PHP 8.1+
- ext-sockets
- ext-openssl (TLS / SSL)
- ext-pcntl (only for the integration test suite)
composer require initphp/socket- TCP, UDP, TLS, SSL — one factory, one interface per side.
- Non-blocking, select-driven server loop —
live()runs forever, or drive the loop yourself one iteration at a time withtick()(great for embedding into your own event loop or for deterministic tests). - First-class enums —
Transport,DomainandCryptoMethodreplace magic strings and integer flags. - Strategy-based channels —
TcpChannel,UdpChannelandStreamChannelisolate the transport-specific I/O. No static state shared between server instances. - Coherent exception hierarchy — every exception implements
SocketExceptionInterface, so a single catch covers the package. - Typed everywhere — PHP 8.1 enums, readonly promoted properties, full
declare(strict_types=1)coverage. - No silent data loss — liveness checks never consume data from the wire.
use InitPHP\Socket\Socket;
use InitPHP\Socket\Enum\Transport;
use InitPHP\Socket\Interfaces\{SocketServerInterface, SocketConnectionInterface};
$server = Socket::server(Transport::TCP, '127.0.0.1', 8080);
$server->listen();
$server->live(function (SocketServerInterface $srv, SocketConnectionInterface $conn) {
$message = $conn->read(1024);
if ($message === null) {
return;
}
if ($message === 'quit') {
$conn->write("bye\n");
$conn->close();
return;
}
$conn->write("echo: {$message}");
});use InitPHP\Socket\Socket;
use InitPHP\Socket\Enum\Transport;
$client = Socket::client(Transport::TCP, '127.0.0.1', 8080);
$client->connect();
$client->write("hello\n");
echo $client->read(1024);
$client->disconnect();use InitPHP\Socket\Socket;
use InitPHP\Socket\Enum\Transport;
use InitPHP\Socket\Interfaces\{SocketServerInterface, SocketConnectionInterface};
$server = Socket::server(Transport::TLS, '127.0.0.1', 8443, timeout: 5.0)
->option('local_cert', __DIR__ . '/server.pem')
->option('allow_self_signed', true);
$server->listen();
$server->live(function (SocketServerInterface $srv, SocketConnectionInterface $conn) {
$input = $conn->read();
if ($input === null) {
return;
}
if (\preg_match('/^REGISTER\s+([\w-]{3,})$/i', $input, $m) === 1) {
$srv->register($m[1], $conn);
$conn->write("Welcome, {$m[1]}\n");
return;
}
if (\preg_match('/^SEND\s+@([\w-]+)\s+(.*)$/i', $input, $m) === 1) {
$srv->broadcast($m[2], $m[1]);
return;
}
$srv->broadcast(\trim($input));
});use InitPHP\Socket\Socket;
use InitPHP\Socket\Enum\Transport;
$client = Socket::client(Transport::SSL, 'smtp.gmail.com', 465, timeout: 10.0)
->option('verify_peer', false)
->option('verify_peer_name', false);
$client->connect();
$client->write("EHLO [127.0.0.1]\r\n");
echo $client->read(1024);
$client->disconnect();Socket::server(
Transport $transport,
string $host,
int $port,
?Domain $domain = null, // Defaults to Domain::V4 for TCP/UDP. Ignored for TLS/SSL.
?float $timeout = null, // Connect/handshake timeout for TLS/SSL.
): SocketServerInterface;
Socket::client(
Transport $transport,
string $host,
int $port,
?Domain $domain = null,
?float $timeout = null,
): SocketClientInterface;| Method | Purpose |
|---|---|
listen(): static |
Bind and start listening. Does not accept clients. |
live(callable $cb, float $idle = 0.05): void |
Run the accept/dispatch loop until stop() is called. |
tick(callable $cb, float $wait = 0.0): int |
One iteration of the loop. Returns events processed. |
stop(): void |
Cooperatively exit the loop started by live(). |
close(): bool |
Tear everything down (every client + the listening socket). |
broadcast(string $msg, int|string|array|null $ids = null): bool |
Send to all clients or a targeted subset. |
register(int|string $id, SocketConnectionInterface $conn): bool |
Attach an addressable id to a connection. |
getClients(): array |
Map of `id |
AbstractStreamServer (TLS / SSL) additionally exposes:
$server->option(string $key, mixed $value): static // SSL stream context option
$server->timeout(float $seconds): static
$server->blocking(bool $mode = true): static
$server->crypto(?CryptoMethod $method): static| Method | Purpose |
|---|---|
connect(): static |
Open the connection. |
disconnect(): bool |
Close the connection. Idempotent. |
read(int $len = 1024): ?string |
Receive up to $len bytes. Returns null on no data / failure. |
write(string $data): ?int |
Send $data. Returns the number of bytes written, or null on failure. |
AbstractStreamClient (TLS / SSL) adds option(), timeout(), blocking()
and crypto() — same shape as the server side.
InitPHP\Socket\Enum\Transport // TCP, UDP, TLS, SSL
InitPHP\Socket\Enum\Domain // V4, V6, UNIX
InitPHP\Socket\Enum\CryptoMethod // SSLv2/3/23, ANY, TLS, TLSv1_0/1_1/1_2Every package exception implements SocketExceptionInterface, so a single
catch (SocketExceptionInterface $e) covers them all.
SocketExceptionInterface
├── SocketException (extends \RuntimeException)
│ ├── SocketConnectionException
│ └── SocketListenException
└── SocketInvalidArgumentException (extends \InvalidArgumentException)
If you already run an event loop (ReactPHP, Amp, Swoole-bridge, etc.), do
not call live() — invoke tick() from your loop and let the host decide
when to yield:
$server->listen();
while ($yourEventLoop->running()) {
$events = $server->tick(function ($srv, $conn) { /* ... */ }, waitSeconds: 0.0);
if ($events === 0) {
$yourEventLoop->yield();
}
}In-depth guides live under docs/:
- Getting started
- Architecture
- Servers: TCP · UDP · TLS · SSL
- Clients: TCP · UDP · TLS · SSL
- Cookbook: Chat server · SMTP client
- Migrating from 1.x
composer install
composer test # PHPUnit (unit + integration)
composer stan # PHPStan level 8
composer cs-check # PHP-CS-Fixer dry-run
composer cs-fix # Apply style fixes
composer qa # All of the aboveCI runs the full QA pipeline on PHP 8.1, 8.2 and 8.3.
Issues, ideas and pull requests are welcome. Please read the org-wide contributing guide before opening a PR.
Security issues should be reported privately — see SECURITY.md.
- Muhammet ŞAFAK —
<info@muhammetsafak.com.tr>
Released under the MIT License.