infuse-di / infuse
A PSR-11 compliant dependency injection library
1.1.0
2024-12-27 20:01 UTC
Requires
- php: 8.2.*|8.3.*|8.4.*
- psr/container: ^2.0.0
Requires (Dev)
- friendsofphp/php-cs-fixer: ^3.65
- phpstan/phpstan: ^1.12.13
- phpunit/phpunit: ^11.5.2
- robiningelbrecht/phpunit-pretty-print: ^1.3.1
README
Infuse 🍃
A minimal PSR-11 implementation.
Features
- ✔️ Autowiring (powered by Reflection)
- ✔️ Simple API (only 3 methods)
- ✔️ Container can be built from definitions
- ✔️ Singletons
- ✔️ Detects circular dependencies
- ⏳ Compilable for production
- ⏳ PHPStan generics support for container bindings
Documentation
Infuse has a very minimal API with only has three methods:
/** * Finds an entry of the container by its identifier and returns it. * * @param string $id identifier of the entry to look for * * @return mixed entry * * @throws NotFoundExceptionInterface no entry was found for the provided identifier * @throws ContainerExceptionInterface error while retrieving the entry */ public function get(string $id): mixed; /** * Returns true if the container can return an entry for the given identifier. * Returns false otherwise. * * `has($id)` returning true does not mean that `get($id)` will not throw an exception. * It does however mean that `get($id)` will not throw a `NotFoundExceptionInterface`. * * @param string $id identifier of the entry to look for */ public function has(string $id): bool; /** * @param \Closure(Container): mixed $definition * * @throws ContainerExceptionInterface if provided id is not unique */ public function bind(string $id, \Closure $definition): void;
Example:
use Infuse\Container; $container = new Container(); // Container::bind tells the Container how to build a Bar object $container->bind(Bar::class, function () { return new Bar(); }); // Container::has checks if there's a binding with the provided id $container->has(Bar::class); // true // the callable parameter receives the Container itself as argument $container->bind(Foo::class, function (Container $c) { $bar = $c->get(Bar::class); return new Foo($bar): }); // Container::get retrieves an instance from the Container, the bound callable will be called at this moment $foo = $container->get(Foo::class); $isFoo = $foo instanceof Foo; // true // This will throw a ContainerException, ids must be unique $container->bind(Bar::class, function (Container $c) { return new Bar(); }); // This will throw a NotFoundException, this id has not been bound $container->get(Zee::class); // You can bind basically anything $container->bind('some_array', fn () => ['hello' => 'world']); $container->bind('some_scalar', fn () => 42);
You can also create a ready to use Container from a definitions array:
// definitions.php <?php use Infuse\Container; // should be shaped as array<string, callable> return [ GeoLocationService::class => function (Container $c) { $config = $c->get('config'); return new GeoLocationService($config['GEOLOCATION_API_KEY']); }, 'config' => function () { return [ 'GEOLOCATION_API_KEY' => $_ENV['GEOLOCATION_API_KEY'], ]; }, ];
// something.php use Infuse\ContainerFactory; $definitions = require __DIR__ . '/definitions.php'; $container = ContainerFactory::FromDefinitions($definitions); $container->has('config'); // true $container->has(GeoLocationService::class); // true
Mark your classes with the Singleton attribute to always receive the same instance:
use Infuse\Attributes\Singleton; #[Singleton] class SomeSingleton {} $container = new Container(); $sameInstance = $container->get(SomeSingleton::class); $asThisOne = $container->get(SomeSingleton::class);
Installing
The recommended way to install Infuse is through Composer.
composer require infuse-di/infuse
Tests
To execute the test suite, you'll need to install all development dependencies.
git clone https://github.com/davisenra/infuse
composer install
composer test
License
This project is licensed under the MIT license. See License File for more information.