rammewerk / hydrator
Strongly typed hydrator for PHP 8.4+.
Requires
- php: >=8.4
- ext-ctype: *
Requires (Dev)
- phpstan/phpstan: ^2.1.0
- phpunit/phpunit: ^11.5
README
The Rammewerk Hydrator is a fast, reflection-based hydrator for PHP 8.4+, built to effortlessly map arrays onto PHP objects with minimal configuration.
It’s not an ORM - no annotations or heavy frameworks - just a simple yet powerful mapper. It even includes a lazy-loading HydratorCollection for handling large datasets.
Installation
composer require rammewerk/hydrator
Basic usage
Let's consider a simple entity/dto:
final class Product { public string $name = ''; public ?string $sku = null; public float $price = 0; public StatusEnum $status = StatusEnum::draft; public ?\DateTime $created_at = null; public ?ProductDetails $details = null; }
We can populate this from an array:
$data = [ 'name' => 'Some product', 'sku' => '1020', 'status' => 'active', 'created_at' => '2022-01-01', 'details' => [ 'color' => 'red', 'size' => 'large', ], ]; $hydrator = new \Rammewerk\Component\Hydrator(Product::class); $product = $hydrator->hydrate($data); echo $product->name; // Some product echo $product->details?->color; // red
In return, we got a typed entity.
This might be useful to automatically convert data from database to an entity, or from an API etc.
Features
- Reflection-based hydrator with caching
- Converts scalars, date/time objects, enums, union/intersection types
- Handles promoted/readonly properties, default/null values
- Automatically hydrates nested objects
- HydratorCollection for quickly hydrating large datasets
- Full IDE autocompletion
Benefits
- Enhanced type safety via entities
- Automatic class mapping from arrays
- Clear IDE type hints
- Improved code quality
Allowed Property Types
The hydrator ensures each property is set to its declared type by converting raw data as needed (e.g., booleans stored as integers). It supports:
- Built-in types:
string
,int
,float
,bool
,array
- Date types:
DateTime
,DateTimeImmutable
,DateTimeZone
- Backed enums - Using the enum’s
from()
method - Unit enums - Using the enum’s case name as matching value
- Union/Intersection types (e.g.,
string|int
,InterfaceA&InterfaceB
) - Classes (e.g.,
public AnotherClass $class
) — nested objects are hydrated if valid data is provided - Arrays of classes via the
mapArray()
method. More about nested objects here.
Note: One should try not to use union/intersection types, as they may lead to unexpected results. Use with caution.
Nested hydration
If the hydrator encounters a property of type class, it will hydrate the class if given data is an array.
class OrderItems { public ?Product $product = null; } $hydrator = new Hydrator(OrderItems::class)->hydrate([ 'order_id' => 100, 'product' => [ 'name' => 'Some product', 'sku' => '1020', ], ]);
To handle an array of classes, you can use the built-in mapArray method:
class OrderItem { /** @var Product[] */ public array $products = []; } $hydrator = new Hydrator(OrderItem::class); // Notice: the mapArray is an immutable method, so we need to assign the result $hydrator = $hydrator->mapArray('products', Product::class); $orderItem = $hydrator->hydrate([ 'order_id' => 100, 'products' => [ ['name' => 'Some product', 'sku' => '1020'], ['name' => 'Some product', 'sku' => '1020'], ], ]); $orderItem->products[0] instanceof Product; // true
Hydrator Collection
use Rammewerk\Component\Hydrator\Hydrator; use Rammewerk\Component\Hydrator\HydratorCollection; $collection = new HydratorCollection(new Hydrator(Product::class), $data); foreach( $collection as $entity ) { echo $entity->name; } // Get all as an array of entities: Product[] $entities = $collection->toArray();
Collections are lazy-loaded, and will only hydrate the entities when you iterate over them. This is useful for large
datasets, as it will only hydrate the entities that are actually needed. If needed all at once, you can use the
toArray()
method.
Constructor parameters
When your class uses constructor parameters, the hydrator automatically resolves them, but with a few rules:
- Promoted parameters (
public string $firstName
): Treated like properties, but assigned through the constructor. If no default is set, a dataset value is required. - Non-promoted parameters (
string $firstName
): These aren’t set directly from the dataset. They’re constructed with their default or null if nullable. If a non-promoted parameter is non-nullable and lacks a default, it throws an exception. After construction, a corresponding property (if it exists) is separately hydrated from the dataset.
Example:
public function __construct( // Promoted parameters: public readonly string $firstName, // Must be in dataset public int $age, // Must be in dataset public ?string $email, // If not in dataset, defaults to null public bool $active = true, // Optional, uses default if missing // Non-promoted parameters: int $required_id, // No default, not nullable => throws exception public ?string $lastName, // Nullable => defaults to null int $something = 0, // Optional => defaults to 0 ){}
Read more on this below in the section about object design.
Read only properties
To add readonly properties, they must be initiated through the constructor.
class Entity { public function __construct( public readonly int $id ) { } }
Why Use the Hydrator?
The hydrator helps convert raw data (e.g., from a database) into typed PHP objects without tedious manual checks. It enforces strict types, simplifies property access, and makes your code more maintainable.
Without hydrator
// Get all users, returns as an array $user = $this->userRepository->getAll(); foreach($users as $user) { if( isset($user['email']) && is_string($user['email']) && ! empty($user['email']) ) { $this->sendEmail($user['email']); } }
With Hydrator
$users = $this->userRepository->getAll(); foreach ($users as $user) { if (!$user->email) continue; $this->sendEmail($user->email); }
In the hydrator version, your IDE can autocomplete $user
properties because it knows each user is a User
object. It
also ensures that only defined properties on the User entity are hydrated - no unexpected data creeping in.
There’s no hidden
__call()
magic or dynamic property illusions - just strictly typed objects, so you always know exactly what data you’re working with.
To achieve this, simply return a HydratorCollection from your repository:
class User { public string $email = ''; public string $name = ''; public int $id = 0; }
use Rammewerk\Component\Hydrator\Hydrator; use Rammewerk\Component\Hydrator\HydratorCollection; class UserRepository { /** * @return HydratorCollection<User> */ public function getAll(): HydratorCollection { $data = $this->fetchAll("SELECT * FROM `users`"); return new HydratorCollection(new Hydrator(User::class), $data); } }
Now you get a collection of actual User
objects, each property strictly typed, making your code cleaner and more
reliable.
Be Mindful of Object Design
While the hydrator is powerful, it doesn’t replace good object design.
Avoid making your classes reliant on the hydrator by declaring uninitialized or non-optional properties without defaults. If you attempt to access such properties before the hydrator sets them, you’ll encounter exceptions. Instead, use sensible defaults or optional properties, and ensure your constructor aligns with how data is actually provided. See below for an example.
Set reasonable defaults:
public ?string $name = '';
The $name
property is nullable, but it’s initialized with an empty string. This means the hydrator will never set it
to null
; instead, it will use ''
, even when null
is provided. To allow null values, initialize the property like
this:
public ?string $name = null;
Define Required Properties via Constructor
public int $id;
Avoid declaring non-optional properties without initializing them. This is bad practice as it can lead to uninitialized properties.
Best Practice: Define required properties through the constructor. If a property should be read-only, use the readonly keyword in the constructor’s promoted parameters.
public string $name = ''; // Optional property with a default value public function __construct( public string $id, // Required property public readonly string $uid, // Required and read-only property ) {}
Benefits:
- Ensures Initialization: All required properties are set upon object creation.
- Immutability: Using readonly enforces that certain properties cannot be modified after initialization.
- Clear Intent: Makes the code more predictable and easier to understand.
Limited Hydration for Non-Promoted Parameters
Non-promoted constructor parameters aren’t automatically hydrated because they often don’t match a specific property or may be used for computed values. The hydrator’s role is to fill properties, not infer custom constructor logic.
If you need non-promoted parameters, give them defaults, make them nullable, or manually instantiate your object before hydration:
$entity = new Entity('John Doe', 100); $hydrator = new \Rammewerk\Component\Hydrator($entity);
Preserving Properties Set by the Constructor
The hydrator won’t overwrite properties already set to values different from their default. This prevents overwriting values explicitly set by your constructor. For example:
class Entity { public int $age = 0; public function __construct(int $age = 18) { $this->age = $age; } }
If the hydrator is given ['age' => 20], it won’t override age (already set to 18 by the constructor). This behavior ensures constructor-defined values are respected.
Hydrator Callback for More Customizable Hydration
The hydrator can be used as a callback, returning the hydrated object. The first parameter of hydrate()
is an
array of data to hydrate with, and the second parameter is a callback that customizes hydration.
Note: Data given in the first parameter will override values returned by the callback.
Example Usage
$data = ['name' => 'Johnny']; $postData = $_POST; unset($postData['_token']); $entity = $hydrator->hydrate($data, static fn(PropertyHandler $prop) => match ($prop->name) { 'email' => $postData['email_address'] ?? null, // 'name' will be ignored because it is set in the first parameter. 'name' => $postData['first_name'] ?? null, default => null, });
By using this approach, you can define a default set of properties in the first parameter, while the callback provides additional values that are used unless explicitly overridden by the initial array.
Error handling
All exceptions extend Rammewerk\Component\Hydrator\Error\HydratorException. Catch them to manage invalid data.
try { $hydrator->hydrate(...); } catch (\Rammewerk\Component\Hydrator\Error\HydratorException $e) { // It's an error }
Contribution
If you have any issues or would like to contribute to the development of this library, feel free to open an issue or pull request.
License
The Rammewerk Hydrator is open-sourced software licensed under the MIT license.
Keywords: DTO, mapper, data-mapper and populator
Converters
- String
- Scalar (int, float, bool, string)
- Null will be converted to empty string, unless nullable
- Int
- Float
- Bool
- DateTime
- DateTimeImmutable
- DateTimeZone
- Enum
- Union
- Intersection
- Array