Skip to content

Commit e19f01c

Browse files
garvinhickinggithub-actions[bot]
authored andcommitted
[TASK] Create stub for DTO explanation
This is a basic first step definition of a DTO Addresses #4822
1 parent eeee42a commit e19f01c

File tree

2 files changed

+127
-0
lines changed

2 files changed

+127
-0
lines changed

Documentation/ExtensionArchitecture/BestPractises/SoftwareDesignPrinciples.rst

+37
Original file line numberDiff line numberDiff line change
@@ -19,3 +19,40 @@ extensions for TYPO3.
1919

2020
We also recommend to study common `software design
2121
patterns <https://designpatternsphp.readthedocs.io/en/latest/>`__.
22+
23+
.. _concept-dto:
24+
25+
DTO / Data Transfer Objects
26+
----------------------------
27+
28+
A very common pattern in Extbase extensions is a "DTO" ("Data Transfer Object").
29+
30+
A DTO is an instance of a basic class that usually only has a constructor,
31+
getters and setters. It is not meant to be an extension of an Extbase `AbstractEntity`.
32+
33+
This DTO serves as pure data storage. You can use it to receive and retrieve
34+
data in a `<f:form>` fluid
35+
`CRUD <https://en.wikipedia.org/wiki/Create,_read,_update_and_delete>`__
36+
("Create Read Update Delete") setup.
37+
38+
Later on, the DTO can be accessed and converted into a "proper" Extbase domain model
39+
entity: The DTO getters retrieve the data, and the Extbase domain model entity's setters
40+
receive that data:
41+
42+
.. literalinclude:: _dto.php
43+
:language: php
44+
:caption: Example of DTO and AbstractEntity used in an Extbase controller
45+
46+
DTOs are helpful because:
47+
48+
* They allow to decouple pure data from processed/validated data in entities.
49+
* They allow to structure data into distinct data models.
50+
* They allow to use good type hinting and using setters/getters instead of using
51+
untyped and unstructured PHP arrays for data storage.
52+
* They can be used to hide implementation details of your actual entities by transferring
53+
data like filter settings that internally gets applied to actual data models.
54+
55+
56+
Some more reading:
57+
58+
* `<https://usetypo3.com/dtos-in-extbase/>`__
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,90 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
// The actual "plain" DTO, just setters and getters
6+
class PersonDTO
7+
{
8+
protected string $first_name;
9+
protected string $last_name;
10+
11+
public function getFirstName(): string
12+
{
13+
return $this->first_name;
14+
}
15+
16+
public function setFirstName(string $first_name): void
17+
{
18+
$this->first_name = $first_name;
19+
}
20+
21+
public function getLastName(): string
22+
{
23+
return $this->last_name;
24+
}
25+
26+
public function setLastName(string $last_name): void
27+
{
28+
$this->last_name = $last_name;
29+
}
30+
}
31+
32+
// The Extbase domain model entity.
33+
// Note that the getters and setters can easily be mapped
34+
// to the DTO due to their same names!
35+
class Person extends \TYPO3\CMS\Extbase\DomainObject\AbstractEntity
36+
{
37+
protected string $first_name;
38+
protected string $last_name;
39+
40+
public function getFirstName(): string
41+
{
42+
return $this->first_name;
43+
}
44+
45+
public function setFirstName(string $first_name): void
46+
{
47+
$this->first_name = $first_name;
48+
}
49+
50+
public function getLastName(): string
51+
{
52+
return $this->last_name;
53+
}
54+
55+
public function setLastName(string $last_name): void
56+
{
57+
$this->last_name = $last_name;
58+
}
59+
}
60+
61+
// An Extbase controller utilizing DTO-to-entity transfer
62+
class DtoController extends TYPO3\CMS\Extbase\Mvc\Controller\ActionController
63+
{
64+
public function __construct(protected MyVendor\MyExtension\Domain\Repository\PersonRepository $personRepository)
65+
{
66+
}
67+
68+
function createAction(): Psr\Http\Message\ResponseInterface
69+
{
70+
// Set up a DTO to be filled with input data.
71+
// The Fluid template would use <f:form> and its helpers.
72+
$this->view->assign('personDTO', new PersonDTO());
73+
}
74+
75+
function saveAction(PersonDTO $personDTO): Psr\Http\Message\ResponseInterface
76+
{
77+
// Transfer all data to a proper Extbase entity.
78+
// Create an empty entity first:
79+
$person = \TYPO3\CMS\Core\Utility\GeneralUtility::makeInstance(Person::class);
80+
81+
// Use setters/getters for propagation
82+
$person->setFirstName($personDTO->getFirstName());
83+
$person->setLastName($personDTO->getLastName());
84+
85+
// Persist the extbase entity
86+
$this->personRepository->add($person);
87+
88+
// The "old" DTO needs to further processing.
89+
}
90+
}

0 commit comments

Comments
 (0)