update readme
This commit is contained in:
parent
bb13950966
commit
46b98d16a6
88 changed files with 7546 additions and 176 deletions
213
implementation/16-caching/data/pages/09-dependency-injector.md
Normal file
213
implementation/16-caching/data/pages/09-dependency-injector.md
Normal file
|
@ -0,0 +1,213 @@
|
|||
[<< previous](08-inversion-of-control.md) | [next >>](10-invoker.md)
|
||||
|
||||
### Dependency Injector
|
||||
|
||||
A dependency injector resolves the dependencies of your class and makes sure that the correct objects are injected when
|
||||
the class is instantiated.
|
||||
|
||||
Again the psr has defined an [interface](https://www.php-fig.org/psr/psr-11/) for dependency injection that we can work
|
||||
with. Almost all common dependency injection containers implement this interface, so it is a good starting point to look
|
||||
for a [suitable solution on packagist](https://packagist.org/providers/psr/container-implementation).
|
||||
|
||||
I choose the [PHP-DI](https://packagist.org/packages/php-di/php-di) container, as it is easy to configure and provides some very [powerfull features](https://php-di.org/#autowiring)
|
||||
out of the box.
|
||||
|
||||
After installing the container through composer create a new file with the name 'dependencies.php' in your config folder:
|
||||
|
||||
```php
|
||||
<?php declare(strict_types = 1);
|
||||
|
||||
$builder = new \DI\ContainerBuilder();
|
||||
$builder->addDefinitions([
|
||||
\Psr\Http\Message\ResponseInterface::class => \DI\create(\Laminas\Diactoros\Response::class),
|
||||
\Psr\Http\Message\ServerRequestInterface::class => fn () => \Laminas\Diactoros\ServerRequestFactory::fromGlobals(),
|
||||
]);
|
||||
|
||||
return $builder->build();
|
||||
```
|
||||
|
||||
In this file we create a containerbuilder, add some definitions to it and return the container.
|
||||
As the container supports autowiring we only need to define services where we want to use a specific implementation of
|
||||
an interface.
|
||||
|
||||
In the example i used two different ways of defining the service. The first is by using the 'create' method of PHP-DI to
|
||||
tell the container that it should create a Diactoros\Response object when ever I query a ResponseInterface, in the second
|
||||
exampler I choose to write a small factory closure that wraps the Laminas Requestfactory.
|
||||
|
||||
Make sure to read the documentation on definition types on the [PHP-DI website](https://php-di.org/doc/php-definitions.html#definition-types),
|
||||
as we will use that extensively.
|
||||
|
||||
Of course your `Bootstrap.php` will also need to be changed. Before you were setting up `$request` and `$response` with `new` calls. Switch that to the dependency container. We do not need to get the response here, as the container will create and use it internally
|
||||
to create our Handler-Object
|
||||
|
||||
```php
|
||||
$container = require __DIR__ . '/../config/dependencies.php';
|
||||
assert($container instanceof \Psr\Container\ContainerInterface);
|
||||
|
||||
$request = $container->get(\Psr\Http\Message\ServerRequestInterface::class);
|
||||
assert($request instanceof \Psr\Http\Message\ServerRequestInterface);
|
||||
```
|
||||
|
||||
The other part that has to be changed is the dispatching of the route. Before you had the following code:
|
||||
|
||||
```php
|
||||
$className = $routeInfo[1];
|
||||
$handler = new $className($response);
|
||||
assert($handler instanceof \Psr\Http\Server\RequestHandlerInterface)
|
||||
foreach ($routeInfo[2] as $attributeName => $attributeValue) {
|
||||
$request = $request->withAttribute($attributeName, $attributeValue);
|
||||
}
|
||||
$response = $handler->handle($request);
|
||||
```
|
||||
|
||||
Change that to the following:
|
||||
|
||||
```php
|
||||
/** @var RequestHandlerInterface $handler */
|
||||
$className = $routeInfo[1];
|
||||
$handler = $container->get($className);
|
||||
assert($handler instanceof RequestHandlerInterface);
|
||||
foreach ($routeInfo[2] as $attributeName => $attributeValue) {
|
||||
$request = $request->withAttribute($attributeName, $attributeValue);
|
||||
}
|
||||
$response = $handler->handle($request);
|
||||
```
|
||||
|
||||
Make sure to use the container fetch the response object in the catch blocks as well:
|
||||
|
||||
```php
|
||||
} catch (MethodNotAllowed) {
|
||||
$response = $container->get(ResponseInterface::class);
|
||||
assert($response instanceof ResponseInterface);
|
||||
$response = $response->withStatus(405);
|
||||
$response->getBody()->write('Not Allowed');
|
||||
} catch (NotFound) {
|
||||
$response = $container->get(ResponseInterface::class);
|
||||
assert($response instanceof ResponseInterface);
|
||||
$response = $response->withStatus(404);
|
||||
$response->getBody()->write('Not Found');
|
||||
}
|
||||
```
|
||||
|
||||
Now all your controller constructor dependencies will be automatically resolved with PHP-DI.
|
||||
|
||||
We can now use that to inject all kinds of services. Often we need to work with the Current time to do some comparisons
|
||||
in an application. Of course we are writing S.O.L.I.D. and testable code so that we would never be so crazy as to call
|
||||
`$time = new \DateTimeImmutable();` in our Action directly, because then we would need to change the system time of we
|
||||
want to work with a different date in a test.
|
||||
|
||||
Therefore we are creating a new Namespace called 'Service\Time' where we introduce a Now-Interface and an Implementation
|
||||
that creates us a DateTimeImmutable object with the current date and time.
|
||||
|
||||
src/Service/Time/Now.php:
|
||||
```php
|
||||
namespace Lubian\NoFramework\Service\Time;
|
||||
|
||||
interface Now
|
||||
{
|
||||
public function __invoke(): \DateTimeImmutable;
|
||||
}
|
||||
```
|
||||
src/Service/Time/SystemClockNow.php:
|
||||
```php
|
||||
namespace Lubian\NoFramework\Service\Time;
|
||||
|
||||
final class SystemClockNow implements Now
|
||||
{
|
||||
|
||||
public function __invoke(): \DateTimeImmutable
|
||||
{
|
||||
return new \DateTimeImmutable;
|
||||
}
|
||||
}
|
||||
```
|
||||
If we want to use that Service in our HelloAction we just need to add it as another argument for the Constructor and
|
||||
update the handle-method to use the new class property:
|
||||
|
||||
```php
|
||||
<?php declare(strict_types=1);
|
||||
|
||||
namespace Lubian\NoFramework\Action;
|
||||
|
||||
use Lubian\NoFramework\Service\Time\SystemClockNow;
|
||||
use Psr\Http\Message\ResponseInterface;
|
||||
use Psr\Http\Message\ServerRequestInterface;
|
||||
use Psr\Http\Server\RequestHandlerInterface;
|
||||
|
||||
final class Hello implements RequestHandlerInterface
|
||||
{
|
||||
public function __construct(
|
||||
private ResponseInterface $response,
|
||||
private SystemClockNow $now,
|
||||
)
|
||||
{
|
||||
}
|
||||
|
||||
public function handle(ServerRequestInterface $request): ResponseInterface
|
||||
{
|
||||
$name = $request->getAttribute('name', 'Stranger');
|
||||
$nowAsString = ($this->now)()->format('H:i:s');
|
||||
$body = $this->response->getBody();
|
||||
|
||||
$body->write('Hello ' . $name . '!');
|
||||
$body->write(' The Time is ' . $nowAsString);
|
||||
|
||||
return $this->response
|
||||
->withBody($body)
|
||||
->withStatus(200);
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
If you open the route in your browser you should see that the current time gets displayed. This happens because PHP-DI
|
||||
automatically figures out what classes are requested in the constructor and tries to create the objects needed.
|
||||
|
||||
But we do not want to depend on the SystemClockNow implementation in our class because that would violate our sacred
|
||||
S.O.L.I.D. principles therefore we need to change the Typehint to the Now interface:
|
||||
|
||||
```php
|
||||
public function __construct(
|
||||
private ResponseInterface $response,
|
||||
private Now $now,
|
||||
)
|
||||
```
|
||||
|
||||
When we are now accessing the Handler in the Browser we get an Error because we have not defined which implementation
|
||||
should be use to satisfy dependencies on the Now interface. So lets add that definition to our dependencies file:
|
||||
|
||||
```php
|
||||
\Lubian\NoFramework\Service\Time\Now::class => fn () => new \Lubian\NoFramework\Service\Time\SystemClockNow(),
|
||||
```
|
||||
|
||||
we could also use the PHP-DI create method to delegate the object creation to the container implementation:
|
||||
```php
|
||||
\Lubian\NoFramework\Service\Time\Now::class => DI\create(\Lubian\NoFramework\Service\Time\SystemClockNow::class),
|
||||
```
|
||||
|
||||
this way the container can try to resolve any dependencies that the class might have internally, but prefer the other
|
||||
method because we are not depending on this specific dependency injection implementation.
|
||||
|
||||
Either way the container should now be able to correctly resolve the dependency on the Now interfacen when you are
|
||||
requesting the Hello action.
|
||||
|
||||
If you run phpstan now, you will get some errors, because the get method on the ContainerInterface returns 'mixed'. As
|
||||
we will adress these issues later, lets tell phpstan that we know about the issue and we can ignore it for now. This way
|
||||
we wont get any warnings for this particular issue, but for any other issues we add to our code.
|
||||
|
||||
Update the phpstan.neon file to include a "baseline" file:
|
||||
|
||||
```
|
||||
includes:
|
||||
- phpstan-baseline.neon
|
||||
|
||||
parameters:
|
||||
level: 9
|
||||
paths:
|
||||
- src
|
||||
```
|
||||
|
||||
if we run phpstan with './vendor/bin/phpstan analyse --generate-baseline' it will add all current errors to that file and
|
||||
ignore them in the future. You can also add that command to your composer.json for easier access. I have called it just
|
||||
'baseline'
|
||||
|
||||
[<< previous](08-inversion-of-control.md) | [next >>](10-invoker.md)
|
Loading…
Add table
Add a link
Reference in a new issue