martincamen / php-arr-core
Canonical domain models and value objects for the *arr ecosystem
Installs: 13
Dependents: 1
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/martincamen/php-arr-core
Requires
- php: ^8.3
- guzzlehttp/guzzle: ^7.10
- martincamen/php-file-size: ^1.0
Requires (Dev)
- laravel/pint: ^1.0
- phpstan/phpstan: ^2.0
- phpunit/phpunit: ^12.0
- rector/rector: ^2.3
- roave/security-advisories: dev-latest
This package is auto-updated.
Last update: 2025-12-31 00:56:05 UTC
README
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.
Why does this exist?
The *arr ecosystem is highly cohesive:
- Sonarr, Radarr, Jellyseerr and NZBGet all model:
- media
- downloads
- queues
- 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
What this package is NOT
❌ HTTP clients ❌ API DTOs ❌ Guzzle, Curl, or PSR-18 implementations ❌ Service-specific assumptions
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:
FileSizeDurationProgress
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/
│ ├── FileSize.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\FileSize; use MartinCamen\ArrCore\ValueObject\Progress; use MartinCamen\ArrCore\Enum\DownloadStatus; $item = new DownloadItem( id: ArrId::fromInt(123), name: 'Example.Movie.2024', size: FileSize::fromGB(8.5), progress: Progress::fromPercentage(42), status: DownloadStatus::Downloading );
How service SDKs integrate
Each service SDK:
- Defines API-specific DTOs
- Fetches raw data via HTTP
- Maps DTOs →
php-arr-coredomain models
Example:
$queueItems = $sonarr->queue(); $coreItems = array_map( fn($item) => SonarrToCoreMapper::mapQueueItem($item), $queueItems );
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.