A PSR-11 compliant service container built for the kadro framework. It combines features of a Service Locator and Dependency Injection Container, designed to manage configurations, services, and dependencies efficiently. LeMarchand is complemented by two core components:
- Solver: Handles dynamic dependency resolution.
- Factory: Manages object instantiation and caching.
- PSR-11 Compliant: Implements the standard container interface, providing
has
andget
methods. - Dynamic Dependency Resolution: Supports complex resolution strategies via the
Solver
. - Efficient Instance Management: Includes a
Factory
to cache instances and manage singletons. - Flexible Configuration: Easily load settings, register services, and resolve dependencies.
- Extensible: Designed for use in applications that require advanced dependency management.
composer require hexmakina/le-marchand
You can initialize the container with a configuration array:
$settings = [
'app' => [
'name' => 'KORAL',
'production_host' => 'engine.hexmakina.be',
'session_start_options' => ['session_name' => 'koral-alias'],
'time_window_start' => '-3 months',
'time_window_stop' => '+1 month',
],
'controller_namespaces' => [
'App\\Controllers\\',
'HexMakina\\koral\\Controllers\\',
'HexMakina\\kadro\\Controllers\\'
]
];
$box = new LeMarchand($settings);
Settings are stored under the settings
key:
$box->has('settings'); // returns true
$box->has('settings.app.name'); // returns true
$box->get('settings.app.name'); // returns 'KORAL'
You can register additional services manually:
$box->register('HexMakina\Crudites\ConnectionInterface', $connection);
LeMarchand is the main container class. It:
- Implements the PSR-11
ContainerInterface
. - Manages a centralized collection of services and configurations.
- Provides mechanisms for retrieving and checking for services (
get
andhas
).
- Singleton Pattern: Ensures only one instance of the container exists.
- Configuration Management: Supports hierarchical settings (e.g.,
settings.database.host
). - Integration with Solver and Factory for advanced dependency resolution and instantiation.
The Solver
class extends LeMarchand’s capabilities by dynamically resolving dependencies using multiple strategies:
- Settings Probe: Resolves hierarchical keys (e.g.,
settings.app.name
) from the configuration array. - Class Probe: Instantiates a class if it exists.
- Interface Probe: Maps interfaces to implementations via a wiring configuration.
- Cascade Probe: Dynamically resolves classes using a namespace cascade for patterns like MVC.
// Resolve a setting
$solver->probeSettings('settings.app.name'); // Returns 'KORAL'
// Resolve a class
$solver->probeClasses(App\Controllers\UserController::class);
// Resolve an interface
$solver->probeInterface(App\Contracts\LoggerInterface::class);
// Resolve dynamically using namespaces
$solver->probeCascade('Controllers\UserController');
The Factory
class handles the creation and caching of class instances. It:
- Dynamically resolves constructor dependencies using the container.
- Caches instances to prevent redundant object creation.
- Handles singleton creation for classes with private constructors.
$factory = new Factory($box);
// Retrieve a cached instance or create a new one
$instance = $factory->serve(App\Services\EmailService::class);
// Build a new instance of a class
$newInstance = $factory->build(App\Models\User::class);
// Create a singleton
$singleton = $factory->buildSingleton(new \ReflectionClass(App\Services\Singleton::class), ['getInstance', []]);
LeMarchand adheres to PSR-11 standards for error handling:
Psr\Container\NotFoundExceptionInterface
: Thrown when a service or configuration is not found.Psr\Container\ContainerExceptionInterface
: Thrown when there’s an issue resolving a service.
LeMarchand supports complex resolution scenarios using the Solver
:
// Resolve a service with constructor dependencies
$controller = $box->get(App\Controllers\UserController::class);
The Factory
simplifies singleton management:
$singleton = $factory->buildSingleton(
new \ReflectionClass(App\Services\Logger::class),
['getInstance', []]
);
Feel free to submit issues or pull requests. Contributions are welcome!
The LeMarchand box, also known as the Lament Configuration, was introduced in The Hellbound Heart novella and later appeared in the Hellraiser film series. This project pays homage to its mystique by solving the "lamentations" of dependency management.