Configure less. Do more.
Framework-agnostic Dependency Injection tool and PSR-11 implementation provider.
PHP 7.4|8+
composer require free-elephants/di
Your entry php script (index.php or some background-job runner)
$components = require 'components.php';
$di = (new \FreeElephants\DI\InjectorBuilder)->buildFromArray($components);
$app = $di->createInstance(\YourApplication::class);
$app->run();
Your components.php
file with dependencies description should look like this:
<?php
return [
'instances' => [
\PDO::class => new \PDO(getenv('DB_DNS'), getenv('DB_USER'), getenv('DB_PASS')),
],
'register' => [
\YourApplication::class,
\ControllerFactory::class,
\SomeService::class,
\AnotherService::class,
\Psr\Log\LoggerInterface::class => \Symfony\Component\Console\Logger\ConsoleLogger::class,
],
'callable' => [
// if function provided as key value
// first argument passed to callable is psr container
// second is key
Foo::class => function(\Psr\Container\ContainerInterface $container, string $key) {
return (new Foo())->setSomething($container->get('something'));
},
// if array provided as key value
// first argument passed to callable is psr container
// remaining element as ...args tail
Bar::class => [ // array where first element is callable, other is values for last arguments
function(\Psr\Container\ContainerInterface $container, $firstArg, string $secondArg) {
return new Bar($firstArg, $secondArg);
},
100,
500,
],
],
'loggers' => [
// For suitable logger injections use map, where keys are your services, that implement LoggerAwareInterface
// and value is logger instances
LoggerAwareClass::class => $logger,
AnotherLoggerAwareClass::class => $anotherLogger,
],
];
The main idea: all your components should expect all dependencies as constructor arguments. All other work entrust to Injector. You do not have to want to instantiate any classes directly in your code. Your must inject some factories instead.
<?php
// getenv('ENV') -> 'test'
$components = (new \FreeElephants\DI\EnvAwareConfigLoader(__DIR__ . '/config', 'ENV'))->readConfig('components');
$di = (new \FreeElephants\DI\InjectorBuilder)->buildFromArray($components);
EnvAwareConfigLoader
load config/components.php
and merge it with config/components.test.php
if it exists.
Default value is false
.
Default value is false
. When you set it true
, you can register only specific interfaces instances.
All final typed dependency will be lazy-instantiated by chain!
Default value is true
.
Default value is false
.
Allow to set LoggerInterface into LoggerAwareInterface instances after constructing, or use loggers map if type present.
В ООП можно выделить две большие группы классов по их ответственности: сущности и сервисы.
Сущности содержат данные и методы для работы с ними. Они чаще всего обладают состоянием и требуются во множестве экзепляров, создаваемых во время исполнения программы. Например:
- доменные объекты (например Entities в контексте Doctrine, Models у Propel)
- Value Objects
- объекты для передачи данных (Data Transfer Objects)
- Request / Response из PSR 7
Сервисы отвечают за всё остальное:
- обрабатывают запрос пользователя (контроллеры, команды)
- оперируют сущностями
- обеспечивают коммуникацию между компонентами системы
- инстанцируют другие сущности и сервисы (фабрики, локаторы)
- предоставляют прикладную функциональность (протоколы, хранилища, роутинг)
Сервис часто требуются в единственном экземпляре, и, редко меняет собственное состояние во время исполнения. Сервисы могут быть описаны до этапа исполнения, например в статическом файле, и быть получены в коде по требованию, или созданы единожды при запуске приложения.
Сущности не должны зависеть от сервисов. В то время, как сервисы наоборот — часто оперируют экземплярами сущностей. При этом сервисы могут использовать другие сервисы.
Наиболее явный и надёжный способ внедрения зависимостей, это инъекция в конструктор:
- нельзя создать экземпляр неготовый к использованию
- зависимости класса обозначены контрактом в одном месте
Type Hinting и рефлексия в php позволяют собрать готовый к использованию сервис на основе сигнатуры его конструктора, без лишних конфигурационных файлов и магии. Такой подход использован в free-elephants/di. Этот способ хорошо поддерживает рефакторинг, т.к. используется только нативный php-код, не требует статического описания зависимостей в yml, xml или аннотациях.