Creating DTOs
Learn all the different ways to create DTO instances in SimpleDTO.
Creation Methods Overview
Section titled “Creation Methods Overview”SimpleDTO provides multiple ways to create instances:
| Method | Use Case | Example | 
|---|---|---|
| fromArray() | From associative array | UserDTO::fromArray($data) | 
| fromJson() | From JSON string | UserDTO::fromJson($json) | 
| fromRequest() | From HTTP request (Laravel) | UserDTO::fromRequest($request) | 
| fromModel() | From Eloquent model (Laravel) | UserDTO::fromModel($user) | 
| fromEntity() | From Doctrine entity (Symfony) | UserDTO::fromEntity($user) | 
| fromXml() | From XML string | UserDTO::fromXml($xml) | 
| fromYaml() | From YAML string | UserDTO::fromYaml($yaml) | 
| fromCsv() | From CSV string | UserDTO::fromCsv($csv) | 
| validateAndCreate() | With validation | UserDTO::validateAndCreate($data) | 
| new | Direct instantiation | new UserDTO(...) | 
From Array
Section titled “From Array”Basic Usage
Section titled “Basic Usage”$dto = UserDTO::fromArray([    'name' => 'John Doe',    'email' => 'john@example.com',    'age' => 30,]);With Nested Arrays
Section titled “With Nested Arrays”$dto = UserDTO::fromArray([    'name' => 'John Doe',    'email' => 'john@example.com',    'address' => [        'street' => '123 Main St',        'city' => 'New York',        'country' => 'USA',    ],]);With Extra Keys (Ignored)
Section titled “With Extra Keys (Ignored)”// Extra keys are automatically ignored$dto = UserDTO::fromArray([    'name' => 'John Doe',    'email' => 'john@example.com',    'age' => 30,    'extra_field' => 'ignored',  // Ignored    'another_field' => 'ignored', // Ignored]);From JSON
Section titled “From JSON”Basic Usage
Section titled “Basic Usage”$json = '{"name":"John Doe","email":"john@example.com","age":30}';$dto = UserDTO::fromJson($json);From JSON File
Section titled “From JSON File”$json = file_get_contents('user.json');$dto = UserDTO::fromJson($json);With Nested JSON
Section titled “With Nested JSON”$json = '{    "name": "John Doe",    "email": "john@example.com",    "address": {        "street": "123 Main St",        "city": "New York"    }}';$dto = UserDTO::fromJson($json);From HTTP Request
Section titled “From HTTP Request”Laravel Request
Section titled “Laravel Request”use Illuminate\Http\Request;
class UserController extends Controller{    public function store(Request $request)    {        $dto = UserDTO::fromRequest($request);
        // Or with validation        $dto = UserDTO::validateAndCreate($request->all());    }}Symfony Request
Section titled “Symfony Request”use Symfony\Component\HttpFoundation\Request;
class UserController{    public function store(Request $request)    {        $dto = UserDTO::fromArray($request->request->all());    }}Plain PHP
Section titled “Plain PHP”// From $_POST$dto = UserDTO::fromArray($_POST);
// From $_GET$dto = UserDTO::fromArray($_GET);
// From php://input$json = file_get_contents('php://input');$dto = UserDTO::fromJson($json);From Database Models
Section titled “From Database Models”Laravel Eloquent
Section titled “Laravel Eloquent”use App\Models\User;
// From single model$user = User::find(1);$dto = UserDTO::fromModel($user);
// From collection$users = User::all();$dtos = $users->map(fn($user) => UserDTO::fromModel($user));
// Or use DataCollection$dtos = DataCollection::make($users, UserDTO::class);Symfony Doctrine
Section titled “Symfony Doctrine”use App\Entity\User;
// From single entity$user = $entityManager->find(User::class, 1);$dto = UserDTO::fromEntity($user);
// From multiple entities$users = $repository->findAll();$dtos = array_map(    fn($user) => UserDTO::fromEntity($user),    $users);
// Or use DataCollection$dtos = DataCollection::make($users, UserDTO::class);From XML
Section titled “From XML”Basic Usage
Section titled “Basic Usage”$xml = '<?xml version="1.0"?><user>    <name>John Doe</name>    <email>john@example.com</email>    <age>30</age></user>';
$dto = UserDTO::fromXml($xml);From XML File
Section titled “From XML File”$xml = file_get_contents('user.xml');$dto = UserDTO::fromXml($xml);From YAML
Section titled “From YAML”Basic Usage
Section titled “Basic Usage”$yaml = 'name: John Doeemail: john@example.comage: 30';
$dto = UserDTO::fromYaml($yaml);From YAML File
Section titled “From YAML File”$yaml = file_get_contents('user.yaml');$dto = UserDTO::fromYaml($yaml);From CSV
Section titled “From CSV”Basic Usage
Section titled “Basic Usage”$csv = '"name","email","age""John Doe","john@example.com",30';
$dto = UserDTO::fromCsv($csv);From CSV File
Section titled “From CSV File”$csv = file_get_contents('user.csv');$dto = UserDTO::fromCsv($csv);With Validation
Section titled “With Validation”Validate and Create
Section titled “Validate and Create”use Event4u\DataHelpers\SimpleDTO\Attributes\Required;use Event4u\DataHelpers\SimpleDTO\Attributes\Email;
class UserDTO extends SimpleDTO{    public function __construct(        #[Required]        public readonly string $name,
        #[Required, Email]        public readonly string $email,    ) {}}
// Validates automatically$dto = UserDTO::validateAndCreate([    'name' => 'John Doe',    'email' => 'john@example.com',]);
// Throws ValidationException if invalidtry {    $dto = UserDTO::validateAndCreate([        'name' => '',  // ❌ Required        'email' => 'invalid',  // ❌ Invalid email    ]);} catch (ValidationException $e) {    echo $e->getMessage();}Direct Instantiation
Section titled “Direct Instantiation”Using Constructor
Section titled “Using Constructor”$dto = new UserDTO(    name: 'John Doe',    email: 'john@example.com',    age: 30);With Named Arguments
Section titled “With Named Arguments”// ✅ Good - clear and explicit$dto = new UserDTO(    name: 'John Doe',    email: 'john@example.com',    age: 30);
// ❌ Bad - unclear order$dto = new UserDTO('John Doe', 'john@example.com', 30);From Other DTOs
Section titled “From Other DTOs”Clone and Modify
Section titled “Clone and Modify”$dto1 = UserDTO::fromArray([    'name' => 'John Doe',    'email' => 'john@example.com',    'age' => 30,]);
// Create new DTO with modified values$dto2 = new UserDTO(    name: 'Jane Doe',  // Changed    email: $dto1->email,    age: $dto1->age);Convert Between DTOs
Section titled “Convert Between DTOs”class UserDTO extends SimpleDTO{    public function __construct(        public readonly string $name,        public readonly string $email,    ) {}}
class UserResourceDTO extends SimpleDTO{    public function __construct(        public readonly string $name,        public readonly string $email,        public readonly string $createdAt,    ) {}
    public static function fromUserDTO(UserDTO $user): self    {        return new self(            name: $user->name,            email: $user->email,            createdAt: now()->toIso8601String()        );    }}
$userDto = UserDTO::fromArray(['name' => 'John', 'email' => 'john@example.com']);$resourceDto = UserResourceDTO::fromUserDTO($userDto);Factory Methods
Section titled “Factory Methods”Custom Factory Methods
Section titled “Custom Factory Methods”class UserDTO extends SimpleDTO{    public function __construct(        public readonly string $name,        public readonly string $email,        public readonly string $role,    ) {}
    public static function admin(string $name, string $email): self    {        return new self(            name: $name,            email: $email,            role: 'admin'        );    }
    public static function guest(string $name, string $email): self    {        return new self(            name: $name,            email: $email,            role: 'guest'        );    }}
// Use factory methods$admin = UserDTO::admin('John Doe', 'john@example.com');$guest = UserDTO::guest('Jane Doe', 'jane@example.com');Bulk Creation
Section titled “Bulk Creation”From Array of Arrays
Section titled “From Array of Arrays”$data = [    ['name' => 'John', 'email' => 'john@example.com'],    ['name' => 'Jane', 'email' => 'jane@example.com'],    ['name' => 'Bob', 'email' => 'bob@example.com'],];
$dtos = array_map(    fn($item) => UserDTO::fromArray($item),    $data);Using DataCollection
Section titled “Using DataCollection”use Event4u\DataHelpers\SimpleDTO\DataCollection;
$data = [    ['name' => 'John', 'email' => 'john@example.com'],    ['name' => 'Jane', 'email' => 'jane@example.com'],    ['name' => 'Bob', 'email' => 'bob@example.com'],];
$collection = DataCollection::make($data, UserDTO::class);
// Now you can use collection methods$filtered = $collection->filter(fn($dto) => str_contains($dto->email, 'john'));$mapped = $collection->map(fn($dto) => $dto->name);Best Practices
Section titled “Best Practices”Use Static Factory Methods
Section titled “Use Static Factory Methods”// ✅ Good - clear intent$dto = UserDTO::fromArray($data);
// ❌ Bad - unclear$dto = new UserDTO(...$data);Validate Early
Section titled “Validate Early”// ✅ Good - validate at creation$dto = UserDTO::validateAndCreate($request->all());
// ❌ Bad - validate later$dto = UserDTO::fromArray($request->all());$dto->validate();Use Type-Specific Methods
Section titled “Use Type-Specific Methods”// ✅ Good - use specific method$dto = UserDTO::fromJson($json);
// ❌ Bad - manual parsing$data = json_decode($json, true);$dto = UserDTO::fromArray($data);Handle Errors Gracefully
Section titled “Handle Errors Gracefully”try {    $dto = UserDTO::validateAndCreate($data);} catch (ValidationException $e) {    // Handle validation errors    return response()->json([        'errors' => $e->errors()    ], 422);}Code Examples
Section titled “Code Examples”The following working examples demonstrate DTO creation:
- Basic DTO - Simple DTO with required properties
- DTO Factory - Factory pattern for DTOs
- Wrapping - Wrapping existing data
- Optional Properties - Handling optional properties
All examples are fully tested and can be run directly:
php examples/simple-dto/creating-dtos/basic-dto.phpphp examples/simple-dto/creating-dtos/dto-factory.phpRelated Tests
Section titled “Related Tests”The functionality is thoroughly tested. Key test files:
- SimpleDTOTest.php - Core DTO functionality
- DTOFactoryTest.php - Factory pattern tests
- OptionalPropertiesTest.php - Optional properties tests
Run the tests:
# Run all SimpleDTO teststask test:unit -- --filter=SimpleDTO
# Run specific test filevendor/bin/pest tests/Unit/SimpleDTO/SimpleDTOTest.phpSee Also
Section titled “See Also”- Type Casting - Automatic type conversion
- Validation - Validate your data
- Property Mapping - Map property names
- Collections - Working with collections
