Important: #96332 - Extbase Validators can use dependency injection¶
See forge#96332
Description¶
In contrast to what has been outlined with this changelog, Extbase validators can use dependency injection in v11 again.
Using dependency injection in Extbase validators is possible again, and it will be available as standard functionality in TYPO3 v12.
All options below are only needed for extensions that really need to find fully compatible ways for dependency injection in their validators. In case single extensions have already been adapted to use the strategy from the breaking changelog, no further adaption is needed.
Extensions still have to apply some manual code changes to single validators
if they should be dependency injection aware, though: Validators that implement
method set
can use __
or inject*
methods
for dependency injection. Method set
will be added to Validator
in v12 as mandatory method, and Abstract
will implement it. Extensions
with dependency injection-aware validators additionally need to set the class
public: true
and shared: false
in Services.
. This
will be done automatically in v12.
Note
All standard validators of EXT:extbase and EXT:form will be marked final
in TYPO3 v12.
Extension authors should consider this when refactoring validators in TYPO3 v11 already.
A typical Extbase validator that uses dependency injection in v10 and extends Abstract
looks like this in v10:
class MyCustomValidator extends AbstractValidator
{
public function injectSomething(Something $something)
{
$this->something = $something;
}
}
An extension that keeps dependency injection in v11 can now look like this:
class MyCustomValidator extends AbstractValidator
{
public function injectSomething(Something $something)
{
$this->something = $something;
}
public function setOptions(array $options): void
{
// This method is upwards compatible with TYPO3 v12, it will be implemented
// by AbstractValidator in v12 directly and is part of v12 ValidatorInterface.
// @todo: Remove this method when v11 compatibility is dropped.
$this->initializeDefaultOptions($options);
}
}
An extension that keeps compatibility with v10 and v11 at the same time and needs dependency injection for custom validators, may need an additional quirk to retain v10 compatibility. It looks like this:
class MyCustomValidator extends AbstractValidator
{
public function injectSomething(Something $something)
{
$this->something = $something;
}
public function __construct(array $options = []) {
// Retain v10 compatibility if the validator has options. This is
// especially important if there are *mandatory* options, otherwise
// option initialization will be called twice in v11, which may fail.
// @todo: Remove this method when v10 compatibility is dropped.
if ((new Typo3Version())->getMajorVersion() < 11) {
parent::__construct($options);
}
}
public function setOptions(array $options): void
{
// This method is upwards compatible with TYPO3 v12, it will be implemented
// by AbstractValidator in v12 directly and is part of v12 ValidatorInterface.
// @todo: Remove this method when v11 compatibility is dropped.
$this->initializeDefaultOptions($options);
}
}
Extensions compatible with v11 and v12 can streamline the code like this:
class MyCustomValidator extends AbstractValidator
{
public function __construct(Something $something) {
$this->something = $something;
}
public function setOptions(array $options): void
{
// @todo: Remove this method when v11 compatibility is dropped.
$this->initializeDefaultOptions($options);
}
}
The v12 and above version of this validator can then looks like this:
class MyCustomValidator extends AbstractValidator
{
public function __construct(private readonly Something $something) {
}
}
In all of the above cases, whenever Extbase validators need native dependency injection
without manual General
calls for their dependencies, and
if TYPO3 v11 should be supported, these validators must set public: true
and
shared: false
in Services.
:
# This is obsolete when the extension does not support TYPO3 v11 anymore.
# @todo: Remove this when v11 compatibility is dropped.
MyVendor\MyExtension\Validation\Validator\MyCustomValidator:
public: true
shared: false