mesilov/kinescope-php-sdk

Unofficial PHP SDK for Kinescope API - video management platform

Maintainers

Package info

github.com/mesilov/kinescope-php-sdk

pkg:composer/mesilov/kinescope-php-sdk

Statistics

Installs: 139

Dependents: 0

Suggesters: 0

Stars: 1

Open Issues: 1

0.5.0 2026-05-19 19:54 UTC

README

Unofficial PHP SDK for Kinescope API — a video management platform for uploading, transcoding (up to 4K), protection, and delivery of video content.

Requirements

  • PHP >= 8.4
  • Extensions: ext-json, ext-curl, ext-mbstring
  • A PSR-18 HTTP client for API requests (e.g., Guzzle or Symfony HTTP Client)
  • A PSR-7/PSR-17 implementation for API requests (e.g., nyholm/psr7)
  • Symfony components compatibility: ^5.4|^6.0|^7.0|^8.0

Installation

composer require mesilov/kinescope-php-sdk

You also need an HTTP client and PSR-7 implementation for Kinescope API requests. For example, with Guzzle:

composer require guzzlehttp/guzzle nyholm/psr7

Quick Start

use Kinescope\Core\Credentials;
use Kinescope\Services\ServiceFactory;

// Create factory with API key
$factory = new ServiceFactory(Credentials::fromString('your-api-key'));

// Or read from KINESCOPE_API_KEY environment variable
$factory = ServiceFactory::fromEnvironment();

// Use services
$videos = $factory->videos()->list();
$projects = $factory->projects()->list();
$folders = $factory->folders()->list('project-id');
$playlists = $factory->playlists()->list();
$statistics = $factory->statistics()->forAccount();

Available Services

Service Access Description
Videos $factory->videos() Read/list/search videos
Projects $factory->projects() Read/list projects
Folders $factory->folders() Folder listing and tree navigation
Playlists $factory->playlists() Playlist and playlist-entities listing
Statistics $factory->statistics() Done-video count and total duration aggregation

In-memory Video Search

Use InMemoryVideoSearch when videos are already loaded and you need to map an external embed URL or lesson title back to a VideoDTO / video ID without another API request:

use Kinescope\Core\Pagination;
use Kinescope\Services\Videos\InMemoryVideoSearch;

$videoPage = $factory->videos()->list(
    pagination: new Pagination(perPage: Pagination::MAX_PER_PAGE),
    projectId: 'project-id',
);

$videos = $videoPage->getData();
$search = new InMemoryVideoSearch();

$video = $search->byEmbedLink(
    $videos,
    'https://kinescope.io/embed/oDko3nwPjHwpzmqUgxJmKB',
);

$videoId = $video?->id;

$matches = $search->byName($videos, '1.3 Сегментация и ёмкость рынка');

byEmbedLink() accepts only the canonical https://kinescope.io/embed/{slug} form. URLs with query strings, fragments, trailing slashes, iframe HTML, and non-embed Kinescope links return null.

byName() uses deterministic normalized substring matching: trim, multibyte lowercase, ё/е equivalence, whitespace collapsing, and one leading lesson-number prefix removal. It is not BM25, fuzzy search, stemming, or ranked full-text search.

CLI

The package ships a standalone Symfony Console entry point:

vendor/bin/kinescope list

Credentials are resolved from KINESCOPE_API_KEY or --api-key / -k.

Read-only commands use singular resources and explicit actions:

# List projects or show one project
vendor/bin/kinescope kinescope:project:list --format=table
vendor/bin/kinescope kinescope:project:show 00000000-0000-0000-0000-000000000000

# List folders for a project or show one folder
vendor/bin/kinescope kinescope:folder:list \
  --project-id=00000000-0000-0000-0000-000000000000 \
  --format=json
vendor/bin/kinescope kinescope:folder:show 11111111-1111-1111-1111-111111111111 \
  --project-id=00000000-0000-0000-0000-000000000000

# List videos for a project or folder, or show one video
vendor/bin/kinescope kinescope:video:list \
  --project-id=00000000-0000-0000-0000-000000000000 \
  --folder-id=11111111-1111-1111-1111-111111111111 \
  --format=json
vendor/bin/kinescope kinescope:video:show 22222222-2222-2222-2222-222222222222

# Include sanitized asset summaries in video rows
vendor/bin/kinescope kinescope:video:list \
  --project-id=00000000-0000-0000-0000-000000000000 \
  --include-assets \
  --format=json

# Inspect sanitized assets for one video
vendor/bin/kinescope kinescope:video:asset:list 22222222-2222-2222-2222-222222222222

# Show account, project, or folder statistics
vendor/bin/kinescope kinescope:statistics:show
vendor/bin/kinescope kinescope:statistics:show \
  --project-id=00000000-0000-0000-0000-000000000000 \
  --format=json
vendor/bin/kinescope kinescope:statistics:show \
  --folder-id=11111111-1111-1111-1111-111111111111

List commands support table or deterministic json output. Resource show commands print pretty JSON. kinescope:statistics:show supports table and json; without a selector it reports account-wide statistics, or it can be scoped with exactly one of --project-id or --folder-id. Asset output exposes video_stream_size, video_stream_size_mb, and booleans such as has_url, has_download_link, and downloadable; raw signed CDN URLs and download links are not printed by default.

AI Agent Skill

The package ships a kinescope-cli AI skill under skills/kinescope-cli/ and declares it through extra.skills.source for llm/skills. The skill helps coding agents use vendor/bin/kinescope safely: choose the current kinescope:* commands, prefer JSON for machine parsing, keep KINESCOPE_API_KEY out of output, and avoid exposing signed asset URLs.

Install the Composer skill sync plugin in the consumer project:

composer require --dev llm/skills

Allow the plugin and trust this SDK as a skill donor:

{
  "config": {
    "allow-plugins": {
      "llm/skills": true
    }
  },
  "extra": {
    "skills": {
      "trusted": ["mesilov/kinescope-php-sdk"],
      "aliases": [".claude/skills"]
    }
  }
}

Then sync the skill:

composer skills:update mesilov/kinescope-php-sdk --alias=.claude/skills

llm/skills writes the real skill directory to .agents/skills/ by default, which works for Codex-style repo skills. The --alias=.claude/skills option mirrors the same target for Claude Code without keeping a second copy. If you already added mesilov/kinescope-php-sdk to extra.skills.trusted, you can omit the package argument and run composer skills:update.

DTO timestamp properties such as createdAt, updatedAt, deletedAt, and generatedAt are Carbon\CarbonImmutable instances. toArray() keeps API field names such as created_at and serializes date values as ISO JSON strings, except that asset stream-size metadata is exported as video_stream_size to avoid implying a real downloaded file size.

AssetDTO::$videoStreamSize maps raw API assets[].file_size. This is Kinescope stream metadata, not a guaranteed downloaded file size on disk. For download validation, progress after HTTP metadata is available, disk checks, and storage accounting, use HTTP Content-Length, transfer-reported bytes, bytes written, or final filesize().

Statistics

$account = $factory->statistics()->forAccount();
$project = $factory->statistics()->forProject('project-id');
$folder = $factory->statistics()->forFolder('folder-id');

printf(
    "%d done videos, %d seconds total\n",
    $account->videosCount,
    $account->getTotalSeconds(),
);

Video Downloader + Events

VideoDownloader fetches video metadata through Videos, selects the requested downloadable asset, and transfers the selected video bytes through a dedicated file-transfer boundary. By default it uses CurlFileTransfer, which writes directly to the in-progress file without materializing a PSR-18 response body.

It also supports event subscriptions for the download lifecycle:

  • DownloadStartedEvent
  • DownloadProgressEvent
  • DownloadCompletedEvent
  • DownloadFailedEvent
use Kinescope\Enum\QualityPreference;
use Kinescope\Event\Download\DownloadProgressEvent;
use Kinescope\Services\Videos\VideoDownloader;

$downloader = new VideoDownloader($factory->videos());

$downloader->on(DownloadProgressEvent::class, function (DownloadProgressEvent $event): void {
    printf("Progress: %.1f%%\n", $event->percent);
});

$filePath = $downloader->downloadVideo(
    videoId: 'your-video-id',
    destinationDir: __DIR__ . '/downloads',
    quality: QualityPreference::BEST,
);

For custom transfer behavior, inject FileTransferInterface. The downloader still owns metadata lookup, asset selection, lifecycle events, .part handling, and completed-size validation:

use Kinescope\Services\Videos\Download\FileTransferInterface;
use Kinescope\Services\Videos\Download\FileTransferProgress;
use Kinescope\Services\Videos\Download\FileTransferRequest;
use Kinescope\Services\Videos\Download\FileTransferResult;
use Kinescope\Services\Videos\VideoDownloader;
use RuntimeException;
use Symfony\Component\Filesystem\Filesystem;

final readonly class AppFileTransfer implements FileTransferInterface
{
    public function transfer(FileTransferRequest $request, ?callable $onProgress = null): FileTransferResult
    {
        $source = fopen($request->url, 'rb');
        $target = fopen($request->outputPath, 'wb');
        $bytesWritten = 0;

        if ($source === false) {
            throw new RuntimeException('Transfer stream cannot be opened.');
        }

        if ($target === false) {
            fclose($source);

            throw new RuntimeException('Transfer output cannot be opened.');
        }

        try {
            while (! feof($source)) {
                $chunk = fread($source, 1024 * 1024);

                if ($chunk === false || $chunk === '') {
                    continue;
                }

                $written = fwrite($target, $chunk);

                if ($written === false) {
                    throw new RuntimeException('Transfer stream cannot be written.');
                }

                $bytesWritten += $written;

                if ($onProgress !== null) {
                    $onProgress(new FileTransferProgress($bytesWritten, $request->expectedBytes));
                }
            }
        } finally {
            fclose($source);
            fclose($target);
        }

        return new FileTransferResult($request->outputPath, $bytesWritten, $request->expectedBytes);
    }
}

$downloader = new VideoDownloader(
    videos: $factory->videos(),
    filesystem: new Filesystem(),
    fileTransfer: new AppFileTransfer(),
);

Symfony applications may implement this interface with HttpClientInterface::request() using buffer: false and stream(); symfony/http-client is not required by the SDK itself.

Default transfer policy:

  • cURL GET, HTTP/HTTPS only, follows up to 5 HTTP/HTTPS redirects.
  • TLS peer and host verification are enabled.
  • Only final 2xx HTTP statuses are successful.
  • Connection setup timeout is 10 seconds; there is no fixed total transfer timeout.
  • Stalled transfers fail below 1024 bytes/sec for 60 seconds.
  • Kinescope API bearer credentials are not sent to video download URLs automatically; only explicit FileTransferRequest headers are used.
  • Progress events are throttled by VideoDownloader at 10 MiB intervals.
  • Downloads are written to a sibling .part file first, renamed only after the written byte count matches the transfer-reported byte count or, when absent, the selected asset size, and removed on handled transfer or validation failures.

Development

Setup

# Initialize project (first run)
make docker-init

# Start Docker environment
make docker-up

# Install dependencies
make composer-install

Testing

# Unit tests
make test-unit

# Integration tests (requires API key)
make test-integration

# Full test suite
make test

Code Quality

# Run all linters
make lint-all

# Static analysis
make lint-phpstan

# Code style check (dry-run)
make lint-cs-fixer

# Fix code style
make lint-cs-fixer-fix

License

MIT. See LICENSE for details.

Changelog

See CHANGELOG.md for release history and migration notes.