bluebillywig / bb-sapi-php-sdk
Requires
- bluebillywig/vmsrpc: ^0.97.6
- composer/ca-bundle: ^1.3
- guzzlehttp/guzzle: ^7.5
Requires (Dev)
- dev-master
- v2.0.0
- v1.0.4
- v1.0.3
- v1.0.2
- v1.0.1
- v1.0.0
- dev-fix/create-release-token
- dev-release/v2.0
- dev-feature/refactor-sdk-entities-and-contracts
- dev-add-unit-test-channel-mediacliplist
- dev-channel-list-and-abs-mediaclip-src
- dev-added-thumbnail-path-parsing
- dev-readme-update-with-install
- dev-add-linting-to-pr-check
- dev-fix-for-wrong-mediaclip-nested-entity-name
- dev-started-on-unit-tests
- dev-license-date-update-and-various-fixes
- dev-11895-ability-to-edit-mediaclip
- dev-fix-mediaclip-upload-fail
- dev-introduce-entity-helpers
- dev-change-promise-chaining-to-coroutines
- dev-added-docstring-to-sdk-methods
- dev-allow-for-nested-entity-recursion
This package is auto-updated.
Last update: 2026-03-20 14:12:44 UTC
README
This PHP SDK provides abstractions to interact with the Blue Billywig Server API.
Requirements
- PHP >= 8.1
Installation
composer require bluebillywig/bb-sapi-php-sdk
Quick Start
use BlueBillywig\Sdk; $sdk = Sdk::withRPCTokenAuthentication( 'my-publication', 1, // token ID 'shared-secret' // shared secret ); // List media clips $response = $sdk->mediaclip->list(); $data = $response->getDecodedBody(); print_r($data);
Authentication
The SDK uses HOTP-based RPC token authentication. You need a token ID and shared secret from your Blue Billywig publication settings.
use BlueBillywig\Sdk; use BlueBillywig\Authentication\RPCTokenAuthenticator; // Recommended: use the convenience factory $sdk = Sdk::withRPCTokenAuthentication('my-publication', $tokenId, $sharedSecret); // Or provide a custom authenticator $authenticator = new RPCTokenAuthenticator($tokenId, $sharedSecret); $sdk = new Sdk('my-publication', $authenticator);
Clock synchronization: The RPC token is time-based (HOTP). Both client and server clocks must be reasonably synchronized (within the token expiration window, default 120 seconds). Significant clock drift will cause authentication failures.
Entities
All entities support standard CRUD operations where applicable:
| Entity | List | Get | Create | Update | Delete |
|---|---|---|---|---|---|
$sdk->mediaclip |
Yes | Yes | Yes | Yes | Yes |
$sdk->playlist |
Yes | Yes | Yes | Yes | Yes |
$sdk->channel |
Yes | Yes | Yes | Yes | Yes |
$sdk->playout |
Yes | Yes | Yes | Yes | Yes |
$sdk->subtitle |
Yes | Yes | Yes | Yes | Yes |
$sdk->thumbnail |
- | - | - | - | - |
Sync and Async
Every entity method is available in both synchronous and asynchronous variants. Async methods return a GuzzleHttp\Promise\PromiseInterface.
// Synchronous $response = $sdk->mediaclip->list(); // Asynchronous $promise = $sdk->mediaclip->listAsync(); $response = $promise->wait();
Media Clips
// List $response = $sdk->mediaclip->list(15, 0, 'createddate desc'); // Get (with optional language and job inclusion) $response = $sdk->mediaclip->get(123); $response = $sdk->mediaclip->get(123, 'en', false); // Create $response = $sdk->mediaclip->create(['title' => 'My Video']); // Update $response = $sdk->mediaclip->update(123, ['title' => 'Updated Title']); // Delete (with optional purge) $response = $sdk->mediaclip->delete(123); $response = $sdk->mediaclip->delete(123, true); // purge
Playlists, Channels, Playouts, Subtitles
// All follow the same pattern $response = $sdk->playlist->list(); $response = $sdk->playlist->get(1); $response = $sdk->playlist->create(['title' => 'My Playlist']); $response = $sdk->playlist->update(1, ['title' => 'Updated']); $response = $sdk->playlist->delete(1);
File Uploads
The SDK supports single-chunk and multi-part uploads to S3 via presigned URLs.
// 1. Initialize the upload $initResponse = $sdk->mediaclip->initializeUpload('/path/to/video.mp4'); $initResponse->assertIsOk(); $uploadData = $initResponse->getDecodedBody(); // 2. Execute the upload $success = $sdk->mediaclip->helper->executeUpload('/path/to/video.mp4', $uploadData); // 3. (Optional) Track upload progress foreach ($sdk->mediaclip->helper->uploadProgressGenerator( $uploadData['listPartsUrl'], $uploadData['headObjectUrl'], $uploadData['chunks'], ) as $progress) { echo "Upload progress: {$progress}%\n"; }
Async upload using coroutines:
use GuzzleHttp\Promise\Coroutine; $promise = Coroutine::of(function () use ($sdk, $mediaClipPath) { $response = (yield $sdk->mediaclip->initializeUploadAsync($mediaClipPath)); $response->assertIsOk(); yield $sdk->mediaclip->helper->executeUploadAsync($mediaClipPath, $response->getDecodedBody()); }); $promise->wait();
Thumbnails
// Generate an absolute thumbnail URL with dimensions $url = $sdk->thumbnail->helper->getAbsoluteImagePath('/path/to/image.jpg', 640, 360); // => https://my-publication.bbvms.com/image/640/360/path/to/image.jpg
Response Handling
All entity methods return a SapiResponse object:
$response = $sdk->mediaclip->get(123); // Check status if ($response->isOk()) { $data = $response->getDecodedBody(); } // Or assert (throws on non-2xx) $response->assertIsOk(); $data = $response->getDecodedBody(); // Access response details $response->getStatusCode(); // e.g. 200 $response->getBody()->getContents(); // raw body string $response->getJsonBody(); // parse as JSON $response->getXmlBody(); // parse as XML $response->getDecodedBody(); // try JSON first, then XML $response->getStatusCodeCategory(); // HTTPStatusCodeCategory enum $response->getRequest(); // the original Request object // Batch response utilities Response::allOk($responses); // true if all 2xx Response::assertAllOk($responses); // throws on first non-2xx Response::getFailedResponses($responses); // generator yielding non-2xx responses
Error Handling
The SDK throws typed exceptions for HTTP errors:
use BlueBillywig\Exception\HTTPRequestException; use BlueBillywig\Exception\HTTPClientErrorRequestException; use BlueBillywig\Exception\HTTPServerErrorRequestException; try { $response = $sdk->mediaclip->get(999); $response->assertIsOk(); } catch (HTTPClientErrorRequestException $e) { // 4xx error echo "Client error {$e->getCode()}: {$e->getMessage()}\n"; echo "Response body: {$e->getResponseBody()}\n"; } catch (HTTPServerErrorRequestException $e) { // 5xx error echo "Server error {$e->getCode()}: {$e->getMessage()}\n"; }
Configuration
$sdk = Sdk::withRPCTokenAuthentication('my-publication', $tokenId, $sharedSecret, [ // Pass any Guzzle client options // See: https://docs.guzzlephp.org/en/stable/request-options.html ]);
Development
# Install dependencies composer install # Lint composer run lint # Run tests composer run test:unit # Run tests with coverage composer run test:unit:coverage