bk203/vici-php

Pure-PHP client implementation of the strongSwan VICI protocol to monitor, configure and control the IKE daemon charon.

Maintainers

Package info

github.com/bk203/vici-php

pkg:composer/bk203/vici-php

Statistics

Installs: 83

Dependents: 0

Suggesters: 0

Stars: 1

Open Issues: 0

1.0.0 2026-06-08 09:26 UTC

This package is auto-updated.

Last update: 2026-06-08 09:27:13 UTC


README

CI

A pure-PHP client implementation of strongSwan's VICI protocol. Use it from PHP to monitor, configure, and control the IKE daemon charon.

  • Covers every command and event documented in the VICI README.
  • Pluggable transport: Unix domain socket (default) or TCP, plus a generic StreamTransport for injection and testing. An opt-in ReconnectingTransport wrapper recovers from charon restarts on Unix sockets.
  • Blocking Session for commands, plus an EventListener for long-running event subscriptions.
  • Streaming list commands (list-sas, list-conns, ...) expose event streams as PHP generators.
  • Fully typed, PHPStan level 8 clean, zero runtime dependencies.

Requirements

  • PHP 8.4+
  • A reachable charon VICI socket. Defaults to the Unix socket /var/run/charon.vici; TCP listeners are supported as well.

Install

composer require bk203/vici-php

Quick start

use Bk203\Vici\Session;
use Bk203\Vici\Transport\UnixSocketTransport;

$session = new Session(new UnixSocketTransport('/var/run/charon.vici'));

$info = $session->version();
printf(
    "%s %s (%s, %s, %s)\n",
    $info['daemon'],
    $info['version'],
    $info['sysname'],
    $info['release'],
    $info['machine'],
);

$session->close();

If you're happy with the defaults, new Session() alone connects to /var/run/charon.vici.

Connect over TCP

use Bk203\Vici\Session;
use Bk203\Vici\Transport\TcpSocketTransport;

$session = new Session(new TcpSocketTransport(
    host: '10.0.0.1',
    port: 4502,
    connectTimeout: 5.0,
    readTimeout: 30.0,
));

Inject a custom stream

use Bk203\Vici\Session;
use Bk203\Vici\Transport\StreamTransport;

$stream = stream_socket_client('unix:///run/strongswan/charon.vici');
$session = new Session(new StreamTransport($stream, readTimeout: 10.0));

Long-lived connections (Unix socket reconnect)

For daemons or while (true) loops where charon may restart and recreate the Unix socket file, use ReconnectingTransport. It reconnects automatically when a single send(), receive(), or hasData() call fails with ConnectionException:

use Bk203\Vici\Session;
use Bk203\Vici\Transport\ReconnectingTransport;

$session = new Session(new ReconnectingTransport(
    path: '/var/run/charon.vici',
    readTimeout: 30.0,
));

while (true) {
    sleep(60);
    $info = $session->version();
}

ReconnectingTransport is opt-in; new Session() alone still uses a plain UnixSocketTransport with no automatic recovery.

When the transport implements {@see \Bk203\Vici\Transport\ReconnectableTransportInterface} (including ReconnectingTransport), Session automatically replays daemon-side EVENT_REGISTER calls after each reconnect.

Limitations

  • request() / requireSuccess() retry once on ConnectionException. Multi-packet commands (streamedRequest(), EventListener::listen()) do not auto-resume mid-operation; catch ConnectionException and restart the command or listener loop.
  • TimeoutException is not retried (slow charon is not treated as a dead socket).
  • Custom transports can implement ReconnectableTransportInterface and receive the same restore hook via setOnReconnect().

Common workflows

Load a connection

$session->loadConn([
    'home' => [
        'version' => 2,
        'proposals' => ['aes256-sha256-modp2048'],
        'local_addrs' => ['192.0.2.1'],
        'remote_addrs' => ['198.51.100.1'],
        'local' => ['auth' => 'psk', 'id' => 'home@example.org'],
        'remote' => ['auth' => 'psk', 'id' => 'gw@example.org'],
        'children' => [
            'net' => [
                'local_ts' => ['10.0.0.0/24'],
                'remote_ts' => ['10.1.0.0/24'],
                'esp_proposals' => ['aes256-sha256'],
            ],
        ],
    ],
]);

Booleans are encoded as yes / no, integers and floats are stringified, and null values are skipped — matching the conventions used by the Python and Ruby reference clients.

Stream active SAs

foreach ($session->listSas() as $entry) {
    foreach ($entry as $name => $sa) {
        echo "{$name}: {$sa['state']} ({$sa['local-host']} -> {$sa['remote-host']})\n";
    }
}

Streaming commands return a Generator. If you break out early, the library automatically drains any remaining stream packets and unregisters the backing event before the next command can run, so the connection stays in sync.

Initiate / terminate

$session->initiate(['child' => 'net', 'timeout' => 30_000]);
$session->terminate(['ike' => 'home']);

Commands whose response carries success = no raise CommandException carrying the command name and the decoded response.

Listen for events

use Bk203\Vici\Event;
use Bk203\Vici\EventListener;

$listener = (new EventListener($session))
    ->on(Event::IKE_UPDOWN, function (string $name, array $message): void {
        $up = ($message['up'] ?? null) === 'yes' ? 'up' : 'down';
        echo "IKE {$up}: " . json_encode($message) . "\n";
    })
    ->on(Event::CHILD_UPDOWN, function (string $name, array $message): void {
        echo "Child event: " . json_encode($message) . "\n";
    })
    ->register([Event::IKE_UPDOWN, Event::CHILD_UPDOWN]);

try {
    $listener->listen(timeout: 60.0);
} finally {
    $listener->unregister();
}

A Session allows only one in-flight command at a time, so give a listener its own Session if you also want to issue commands concurrently from another thread/process. $listener->next($timeout) returns the next event (or null on timeout) if you'd rather poll than run a blocking loop.

Command reference

All commands return decoded arrays (or generators for streaming variants). Every helper ultimately delegates to Session::request() / Session::streamedRequest(), so you can always fall back to the generic API for new commands:

$response = $session->request('custom-command', ['arg' => 'value']);
Category Methods
Control version(), stats(), reloadSettings(), initiate(), terminate(), rekey(), redirect(), install(), uninstall()
Streaming listSas(), listPolicies(), listConns(), listCerts(), listAuthorities()
Configuration getConns(), getAuthorities(), loadConn(), unloadConn(), loadCert(), loadKey(), unloadKey(), getKeys(), loadToken(), loadShared(), unloadShared(), getShared(), flushCerts(), clearCreds(), loadAuthority(), unloadAuthority(), loadPool(), unloadPool(), getPools()
Diagnostics getAlgorithms(), getCounters(), resetCounters()

Event constants live on Bk203\Vici\Event (LOG, CONTROL_LOG, LIST_SA, LIST_POLICY, LIST_CONN, LIST_CERT, LIST_AUTHORITY, IKE_UPDOWN, IKE_REKEY, IKE_UPDATE, CHILD_UPDOWN, CHILD_REKEY, ALERT).

Error handling

All exceptions extend Bk203\Vici\Exception\ViciException:

Exception Thrown when
ConnectionException Underlying socket cannot connect, closes mid-stream, or stream_select() fails. Exposes ->context (ConnectionFailureContext) and ->getDetailedMessage() with stream metadata, endpoint, partial I/O progress, and PHP error text
TimeoutException Read/connect timeout elapses
ProtocolException Framing or message-encoding violation on the wire
CommandUnknownException Server replies with CMD_UNKNOWN
CommandException Command completes with success = no; exposes ->command and ->response
EventRegistrationException Server replies with EVENT_UNKNOWN to EVENT_REGISTER / EVENT_UNREGISTER

For long-lived loops, log the detailed form when a connection fails:

use Bk203\Vici\Exception\ConnectionException;

try {
    $session->version();
} catch (ConnectionException $e) {
    error_log($e->getDetailedMessage());
    // Inspect $e->context?->endpoint for "socket file exists" vs stale fd
    // Inspect $e->context?->streamMeta['eof'] and $e->context?->phpError
    throw $e;
}

Known issues

Stacked initiate commands with short client timeouts can lock the VICI socket

initiate can run for a long time on the charon side while IKE negotiation retries play out. That sequence has its own timeout (the timeout field in the command message, in milliseconds), independent of the transport read timeout on your Session.

If the transport read timeout is shorter than that whole charon-side sequence, the client raises TimeoutException before charon sends CMD_RESPONSE. Catching that exception and immediately sending another initiate — or any other command — on the same socket leaves charon still busy with the earlier command. Repeating this pattern desynchronizes the VICI control channel over time: the socket stops responding, every caller cascades into TimeoutException, and the lock affects all clients on that socket, including swanctl and other tools.

Mitigations

  • Set transport read timeouts well above the initiate message timeout, or omit a read timeout for long-running control commands.
  • Do not retry initiate on the same Session after a client-side timeout; treat a wedged socket as requiring a new connection or a charon restart.
  • Keep at most one in-flight initiate per connection; wait for charon to finish (success, failure, or its own timeout) before trying again.

Architecture

  • Bk203\Vici\Transport\TransportInterface — 32-bit length-prefixed framing (max 512 KiB), implemented by UnixSocketTransport, TcpSocketTransport, and StreamTransport.
  • Bk203\Vici\Protocol\{Packet, PacketCodec, PacketType} — packet layer: the 8-bit type + optional 8-bit-length name tag + optional message payload.
  • Bk203\Vici\Message\{MessageEncoder, MessageDecoder, ElementType} — the hierarchical message tree: sections, key/value pairs (16-bit value length), and lists.
  • Bk203\Vici\Session — the high-level API: command dispatch, event interleaving, reference-counted event (de-)registration, and typed wrappers for every VICI command.
  • Bk203\Vici\EventListener — convenience layer over Session for long-running event subscriptions.

Development

composer install
composer test            # PHPUnit (unit + integration, all in-process)
composer test:unit
composer test:integration
composer stan            # PHPStan level 8
composer cs              # PHP-CS-Fixer dry run
composer cs:fix          # PHP-CS-Fixer apply

Integration tests use an in-process MockViciServer backed by a socket pair, so no real charon daemon or container is required.

Docker development environment

A bare Ubuntu 24.04 container runs strongSwan charon plus PHP 8.4 so you can exercise the library against a real VICI socket without installing anything on the host. charon is started by the entrypoint and listens on /var/run/charon.vici for the lifetime of the container.

docker compose build
docker compose run --rm app composer install
docker compose run --rm app composer test          # full suite (unit + integration)
docker compose run --rm app bash                   # interactive shell

With the container shell you can hit the live daemon directly:

docker compose run --rm app php -r \
  'require "vendor/autoload.php"; print_r((new Bk203\Vici\Session())->version());'

The compose file bind-mounts the repository at /app, so host-side edits are picked up immediately. NET_ADMIN is granted to leave the door open for initiate() / kernel IPsec experiments, but it is not required for VICI commands like version(), stats(), load-conn, or list-sas.

License

MIT. See LICENSE.

Acknowledgements

Modelled after the reference Python and Go VICI clients from the strongSwan project.