A modern, framework-agnostic PHP SDK for the Radius Velocity Fleet Telematics API, built on Saloon. Bearer authentication, an optional OAuth2 refresh-token flow, typed responses, and transient-error backoff — all baked in.
Using Laravel? Reach for the companion bridge
chrisjohnleah/velocity-fleet-api-laravelfor a service provider, config, and a persistent token store.
The Velocity Telematics API exposes a small, focused surface — and this SDK wraps all of it:
| Endpoint | SDK |
|---|---|
| List the customers linked to your user | $velocity->customers()->list() |
| List live device (vehicle) positions for a customer | $velocity->devicePositions()->forCustomer($id) |
- PHP 8.3+
- A Velocity Fleet API token (generated in the UI), or a customer-issued refresh token if you're a third-party integration.
composer require chrisjohnleah/velocity-fleet-apiGenerate a token in the Velocity UI under Account → Account Settings → API Integrations → Create API Token, then:
use ChrisJohnLeah\VelocityFleet\VelocityFleet;
$velocity = VelocityFleet::withToken(getenv('VELOCITY_API_TOKEN'));
// Every customer linked to your user — typed.
foreach ($velocity->customers()->list() as $customer) {
printf("%s (#%s) — %s\n", $customer->name, $customer->number, $customer->product);
}Your customer supplies a Refresh Token. The SDK exchanges it for a short-lived access token on first use (standard OAuth2 refresh_token grant), and refreshes again whenever a call comes back unauthorised:
use ChrisJohnLeah\VelocityFleet\VelocityFleet;
$velocity = VelocityFleet::withRefreshToken(
refreshToken: getenv('VELOCITY_REFRESH_TOKEN'),
clientId: getenv('VELOCITY_CLIENT_ID'), // if your OAuth client requires it
clientSecret: getenv('VELOCITY_CLIENT_SECRET'),
);$positions = $velocity->devicePositions()->forCustomer($customer->id);
echo "{$positions->deviceCount} devices\n";
foreach ($positions->devices as $device) {
printf(
"%s @ %.5f,%.5f — %d %s, ignition %s, seen %s\n",
$device->vehicleRegistration,
$device->lat ?? 0.0,
$device->lon ?? 0.0,
$device->speed ?? 0,
$device->speedMeasureText ?? '',
$device->ignitionOn() ? 'on' : 'off',
$device->occurredAt()?->format('H:i') ?? 'n/a',
);
}
// The same devices are also grouped:
foreach ($positions->deviceGroups as $group) {
echo "{$group->name}: ".count($group->devices)." devices\n";
}Use the right id. The customers response is keyed by each customer's unique id — exposed as
Customer::$id. Pass that toforCustomer(), not the human-facingCustomer::$number.
When you use the refresh-token flow, implement Contracts\TokenStore to keep the rotated token between requests (the in-memory ArrayTokenStore only lives for the current process). The token endpoint may rotate the refresh token, so your put() must always overwrite the previous record:
use ChrisJohnLeah\VelocityFleet\Auth\StoredToken;
use ChrisJohnLeah\VelocityFleet\Contracts\TokenStore;
use ChrisJohnLeah\VelocityFleet\VelocityFleet;
use ChrisJohnLeah\VelocityFleet\VelocityFleetConnector;
final class MyTokenStore implements TokenStore
{
public function get(): ?StoredToken { /* load access/refresh/expiresAt */ }
public function put(StoredToken $token): void { /* overwrite */ }
public function forget(): void { /* delete */ }
}
$velocity = new VelocityFleet(
new VelocityFleetConnector(clientId: '…', clientSecret: '…'),
new MyTokenStore(),
);Failures surface as typed exceptions, all extending Exceptions\VelocityFleetException:
| Exception | When |
|---|---|
NotConnectedException |
No token available (and none could be obtained) |
AuthenticationException |
401 / 403 after a refresh attempt — re-authorise |
ApiException |
Any other API error or transport failure (carries ->status, ->body, ->headers, header(), and retryAfter()) |
use ChrisJohnLeah\VelocityFleet\Exceptions\ApiException;
try {
$velocity->devicePositions()->forCustomer($id);
} catch (ApiException $e) {
report("Velocity API {$e->status}: {$e->getMessage()}");
}The Velocity API is a Django REST Framework service using Bearer (SimpleJWT) access tokens, with token issuance via an OAuth2 endpoint (django-oauth-toolkit). The third-party refresh-token exchange isn't part of the public reference, so the SDK targets the standard OAuth2 refresh_token grant at https://www.velocityfleet.com/o/token/ by default. If your integration documents a different token endpoint or client-authentication requirement, pass it through VelocityFleetConnector (tokenEndpoint, clientId, clientSecret) — no code changes needed.
Anything not yet wrapped in a resource can be sent through the client, which still applies auth, refresh-on-401, and typed error handling:
use ChrisJohnLeah\VelocityFleet\Requests\Customers\GetCustomers;
$customers = $velocity->send(new GetCustomers())->dto();composer test # Pest
composer analyse # PHPStan (max)
composer lint # Pint --test
composer check # all threeTests never hit the network — every request is faked with Saloon's MockClient.
Issues and PRs welcome — see CONTRIBUTING.md. Please report security issues privately per SECURITY.md.
MIT © Chris John Leah. See LICENSE.
Not affiliated with or endorsed by Radius or Velocity Fleet. "Radius", "Velocity" and "Kinesis" are trademarks of their respective owners.