martincamen/php-arr-core

Canonical domain models and value objects for the *arr ecosystem

Installs: 48

Dependents: 3

Suggesters: 0

Security: 0

Stars: 0

Watchers: 0

Forks: 0

Open Issues: 0

pkg:composer/martincamen/php-arr-core

0.1.4 2026-01-21 09:46 UTC

This package is auto-updated.

Last update: 2026-01-21 09:46:42 UTC


README

Important

This project is still being developed and breaking changes might occur even between patch versions.

The aim is to follow semantic versioning as soon as possible.

A canonical domain model for the *arr ecosystem.

php-arr-core provides shared domain models, value objects, enums and normalization logic used across *arr service SDKs such as Sonarr, Radarr, Jellyseerr and NZBGet.

The goal is to eliminate duplicated DTOs, inconsistent status handling and ad-hoc mappings between services that conceptually model the same things.

Ecosystem

Package Description
radarr-php PHP SDK for Radarr
sonarr-php PHP SDK for Sonarr
jellyseerr-php PHP SDK for Jellyseerr
laravel-radarr Laravel integration for Radarr
laravel-sonarr Laravel integration for Sonarr
laravel-jellyseerr Laravel integration for Jellyseerr

Why does this exist?

The *arr ecosystem is highly cohesive:

  • Sonarr, Radarr, Jellyseerr and NZBGet all model:
    • media (movies, series, episodes)
    • downloads (queue items)
    • file sizes
    • progress
    • statuses
  • Yet each service exposes these concepts using:
    • different naming
    • different units
    • different status values
    • different lifecycle semantics

This results in:

  • duplicated DTOs
  • repeated status mapping logic
  • fragile integrations
  • inconsistent developer experience

php-arr-core solves this by defining one canonical domain model that all *arr SDKs map to.

What this package is

  • ✅ Pure PHP (no framework dependencies)
  • ✅ Canonical domain models
  • ✅ Value objects (FileSize, Duration, Progress, etc.)
  • ✅ Normalized enums and statuses
  • ✅ Mapping helpers and contracts

Core design principles

1. Canonical domain > API representation

APIs change. Domains evolve slowly.

This package models what things are, not how services expose them.

2. Value objects over primitives

Anything that:

  • has units
  • appears in multiple services
  • requires conversion or logic

...is modeled as a value object.

Examples:

  • ArrFileSize (extends martincamen/php-file-size)
  • Duration
  • Progress

3. Status normalization is centralized

Each service uses its own status vocabulary.

All normalization happens once, in core.

SDKs should never contain switch or if blocks for statuses.

4. Mapping happens at the boundary

Service SDKs are responsible for mapping their API DTOs into php-arr-core domain objects.

Core never depends on service-specific code.

Package structure

src/
├── Domain/
│   ├── Media/
│   ├── Download/
│   ├── Request/
│   └── User/
├── ValueObject/
│   ├── ArrFileSize.php
│   ├── Duration.php
│   ├── Progress.php
│   └── ArrId.php
├── Enum/
│   ├── MediaStatus.php
│   ├── DownloadStatus.php
│   └── Service.php
├── Mapping/
│   ├── StatusNormalizer.php
│   └── ServiceCapabilities.php

Example usage

use MartinCamen\ArrCore\Domain\Download\DownloadItem;
use MartinCamen\ArrCore\ValueObject\ArrFileSize;
use MartinCamen\ArrCore\ValueObject\Progress;
use MartinCamen\ArrCore\Enum\DownloadStatus;

$item = new DownloadItem(
    id: ArrId::fromInt(123),
    name: 'Example.Movie.2024',
    size: ArrFileSize::fromGigabytes(8.5),
    progress: Progress::fromPercentage(42),
    status: DownloadStatus::Downloading
);

How service SDKs integrate

Each service SDK:

  1. Defines API-specific DTOs
  2. Fetches raw data via HTTP
  3. Maps DTOs → php-arr-core domain models

Example:

use MartinCamen\Sonarr\Sonarr;

$sonarr = Sonarr::create('localhost', 8989, 'your-api-key');

// Action-based API returns typed responses
$downloads = $sonarr->downloads()->all();   // DownloadPage
$series = $sonarr->series()->all();          // SeriesCollection
$status = $sonarr->system()->status();       // SystemStatus

Supported services

This package is designed to be used by:

  • Sonarr SDK
  • Radarr SDK
  • Jellyseerr SDK
  • NZBGet SDK
  • Prowlarr SDK (planned)

Versioning & stability

php-arr-core follows semantic versioning.

Breaking changes only occur when:

  • a domain concept changes
  • a value object's behavior changes

New services and fields should be additive whenever possible.

Contributing

Contributions are welcome, especially:

  • new value objects
  • improved normalization
  • additional service mappings

Before adding new models, consider:

Is this a domain concept or an API detail?

If unsure, open an issue.