The official PHP library for the Verne Nautilus platform.
Server-side only. API keys carry full service access and must never be used in browser or client-side code.
PHP 8.1 or later.
composer require vernesoft/sdkuse Vernesoft\Verne;
$verne = new Verne(
relay: $_ENV['VERNE_RELAY_KEY'],
gate: $_ENV['VERNE_GATE_KEY'],
);You can also instantiate services independently if you only need one:
use Vernesoft\Relay;
use Vernesoft\Gate;
$relay = new Relay(apiKey: $_ENV['VERNE_RELAY_KEY']);
$gate = new Gate(apiKey: $_ENV['VERNE_GATE_KEY']);Send events to all subscribed endpoints:
$message = $verne->relay()->messages()->send(
eventType: 'user.created',
payload: ['id' => 'usr_123'],
);Optional parameters:
$message = $verne->relay()->messages()->send(
eventType: 'order.placed',
payload: ['order_id' => '999'],
idempotencyKey: 'evt_abc', // prevent duplicate delivery within 24h
channels: ['team-a'], // restrict to specific endpoint channels
);List previously sent events:
$page = $verne->relay()->messages()->list(limit: 20, eventType: 'user.created');
$page->data; // Message[]
$page->hasMore; // bool
$page->nextCursor; // pass to the next call to paginateManage your end-users. The tenant_id is automatically scoped to your API key.
// Create a user
$identity = $verne->gate()->identities()->create(
schemaId: 'user',
traits: [
'email' => 'user@example.com',
'custom_data' => ['role' => 'editor'],
],
credentials: [
'password' => ['config' => ['password' => 'StrongPassword123!']],
],
state: 'active',
);
// Get a user
$verne->gate()->identities()->get($identity->id);
// Update a user (JSON Patch — RFC 6902)
$verne->gate()->identities()->patch($identity->id, [
['op' => 'replace', 'path' => '/traits/custom_data/role', 'value' => 'admin'],
]);
// Delete a user
$verne->gate()->identities()->delete($identity->id);Exchange your long-lived API key for a short-lived access token:
$token = $verne->gate()->tokens()->create(
subject: 'usr_123',
scopes: ['gate.tokens.read'], // optional
ttlSeconds: 3600, // optional, default 3600, max 86400
);
// $token->accessToken — attach to downstream requests
// $token->expiresAt — ISO 8601 expiryValidate a token:
$info = $verne->gate()->tokens()->introspect($token->accessToken);
if (! $info->active) {
// token is expired or invalid
}Check whether a subject is allowed to perform an action:
$decision = $verne->gate()->authorize(
subject: 'usr_123',
action: 'relay.messages.read',
resource: 'tenant:ten_001',
);
if (! $decision->allowed) {
throw new \RuntimeException('Forbidden');
}All API errors throw a VerneApiException with structured fields:
use Vernesoft\Core\Errors\VerneApiException;
use Vernesoft\Core\Errors\VerneException;
try {
$verne->relay()->messages()->send(eventType: 'ping', payload: []);
} catch (VerneApiException $e) {
$e->getErrorCode(); // e.g. 'invalid_payload', 'unauthorized'
$e->getCode(); // HTTP status code
$e->getRequestId(); // include in support requests
$e->getMessage(); // human-readable message
} catch (VerneException $e) {
// network error or timeout
}Both Verne and the per-service clients accept an optional timeoutSeconds (default 30):
$verne = new Verne(
relay: $_ENV['VERNE_RELAY_KEY'],
timeoutSeconds: 10,
);MIT