nimbly / carton
A simple PSR-11 container implementation offering reflection based autowiring.
Installs: 6 831
Dependents: 5
Suggesters: 0
Security: 0
Stars: 1
Watchers: 2
Forks: 0
Open Issues: 0
Requires
- php: ^8.0
- nimbly/resolve: ^2.1
- psr/container: ^1.0|^2.0
Requires (Dev)
- phpunit/phpunit: ^9.0
- symfony/var-dumper: ^4.3
- vimeo/psalm: ^5.0
Provides
README
A simple PSR-11 container implementation.
Features
- PSR-11 compliant
- Singleton and factory builders
- Service providers
- Nested container support
- Reflection based autowiring
- Aliasing
Install
composer require nimbly/carton
Getting the container
Instantiate container
You can create a new instance of a Container.
$container = new Container;
Singleton instance
Or use the container singleton getInstance
method.
$container = Container::getInstance();
Basic usage
Set an instance
The most basic usage is to just assign an instance to the container itself.
$container->set(SomeInterface::class, new SomeClass);
Of course, you don't need to assign objects - it can be anything you like.
$container->set("timezone", "UTC"); $container->set("foo_fh", \fopen("/tmp/foo", "r"));
Retrieving a value
Grab a value from the container by its key.
$someClass = $container->get(SomeClass::class);
NOTE: Retrieving a value that does not exist will throw a Nimbly\Carton\NotFoundException
.
Checking for instance
You can check for the existance of items registered in the container.
if( $container->has(SomeClass::class) ){ echo "Container has SomeClass::class."; }
Advanced usage
Singleton builder
The singleton builder will ensure only a single instance is ever returned when it is retrieved from the container. The singleton builder requires a callable
that will be invoked when it needs to build your dependency and will pass along the Container
instance as a single parameter.
It also has the added benefit over the set
method by lazily calling your callback. I.e. it will only be created when it is actually needed.
$container->singleton( SomeClass::class, function(Container $container): void { return new SomeClass( $container->get("SomeDependency") ); } );
Factory builder
The factory builder will create new instances each time it is retrieved from the container. The factory builder requires a callable
that will be invoked when it needs to build your dependency and will pass along the Container
instance as a single parameter.
Just like the singleton builder, it has the added benefit over the set method by lazily calling your callback. I.e. it will only be created when it is actually needed.
$container->factory( SomeClass::class, function(Container $container): SomeClass { return new SomeClass( $container->get("SomeDependency") ); } );
Aliases
You can create aliases of your container items. These aliases simply point to an existing container item and fetch that item for you.
$container->set(Bar::class, new Bar); $container->alias(Foo:class, Bar::class); $instance = $container->get(Foo::class); // Returns Bar::class instance.
Alternatively, you can provide an alias or an array of aliases when calling set
, singleton
, or factory
.
$container->singleton( Bar:class, function(Container $container): Bar { return new Bar; }, [Foo::class, Baz::class] ) $instance = $container->get(Bar::class); // Returns Bar::class instance. $instance = $container->get(Foo::class); // Returns Bar::class instance. $instance = $container->get(Baz::class); // Returns Bar::class instance.
Autowiring
You can have instances made for you automatically using the make
method - which will attempt to pull dependencies in from the container itself or recursively attempt to make
them if not found.
class Foo { public function __construct( protected DateTime $date) { } } class Bar { public function __construct( protected Foo $foo) { } } $bar = $container->make(Bar::class);
Dependecy injection on instance methods
Calling an instance method couldn't be easier - Carton will attempt to autoresolve dependencies (autowire) for you when making a call to an instance method.
class BooksController { public function get(ServerRequestInterface $request, string $isbn): Response { return new Response( Books::find($isbn) ); } } $container->set(ServerRequestInterface::class, $serverRequest); $response = $container->call([BooksController::class, "get"], ["isbn" => "123123"]);
Adding additional containers
You can extend Carton with additional PSR-11 compliant container instances by calling the addContainer
method. When Carton attempts to resolve an item, it will always attempt to resolve locally first, and if not found, will loop through any additional containers you have provided.
For example, if you had a configuration manager that implemented ContainerInterface
(PSR-11),
you could add it to Carton.
$container->addContainer( new Config([ new FileLoader(__DIR__ . "/config") ]) ); $container->get("database.connections");
Now you can retrieve your configuration data through the container instance.
Service providers
Service providers allow you to organize your application dependencies in a set of classes.
Create service classes that implement ServiceProviderInterface
.
class MyServiceProvider implements ServiceProviderInterface { public function register(Container $container): void { $container->singleton( MyService::class, function(Container $container): void { return new MyService( $container->get(SomeDependency::class) ); } ); } }
Then register your service providers with the container.
$container->register(new MyServiceProvider);
You can also register multiple services at once and register services by their class name.
// Register group of services at once. $container->register([ new MyServiceProvider, new MyOtherServiceProvider ]); // Register services by class name. $container->register(MyServiceProvider::class); // Register group of services by class name. $container->register([ MyServiceProvider::class, MyOtherServiceProvider::class ]);