Download the PHP package earc/di without Composer

On this page you can find all versions of the php package earc/di. It is possible to download/install these versions without Composer. Possible dependencies are resolved automatically.

FAQ

After the download, you have to make one include require_once('vendor/autoload.php');. After that you have to import the classes with use statements.

Example:
If you use only one package a project is not needed. But if you use more then one package, without a project it is not possible to import the classes with use statements.

In general, it is recommended to use always a project to download your libraries. In an application normally there is more than one library needed.
Some PHP packages are not free to download and because of that hosted in private repositories. In this case some credentials are needed to access such packages. Please use the auth.json textarea to insert credentials, if a package is coming from a private repository. You can look here for more information.

  • Some hosting areas are not accessible by a terminal or SSH. Then it is not possible to use Composer.
  • To use Composer is sometimes complicated. Especially for beginners.
  • Composer needs much resources. Sometimes they are not available on a simple webspace.
  • If you are using private repositories you don't need to share your credentials. You can set up everything on our site and then you provide a simple download link to your team member.
  • Simplify your Composer build process. Use our own command line tool to download the vendor folder as binary. This makes your build process faster and you don't need to expose your credentials for private repositories.
Please rate this library. Is it a good library?

Informations about the package di

earc/di

Standalone lightweight dependency injection component of the eArc libraries.

If you need to decouple your components, restrict injection access or want to make your app components explicit use the earc/component-di library which builds on the top of earc/di.

table of contents

pro/cons

pro

cons

installation

Install the earc dependency injection library via composer.

You can even use it with symfony.

basic usage

A new dependency resolver can be initialized with no arguments. Use this in your index.php, bootstrap or configuration script.

After that classes and parameters can be accessed via the di_* functions.

di_get returns the same instance in successive calls. If you need a new object use di_make. It returns a new instance on each call. In both functions the use of the fully qualified class name as parameter is mandatory.

Classes injected by di_get or di_make must not have constructor parameters. Surprisingly *cough* there is no need for constructor parameters.

Method and even function injection is supported.

There is no need for any further dependency configuration!

Of course you need to import the parameters though.

parameters

A parameter is key value pair. The key is a string and the value can be of any type - even a closure. There may be other restricting factors though. If you choose to organize your parameters in YAML-files you might be restricted to string, int, float, bool and array.

usage

di_param('key_name') returns the parameter value that belongs to the key key_name.

You can check for existence by di_has_param('key_name').

If a parameter is generated dynamically you can use di_set_param('key_name', $value) to make it globally available. A immutable request object would make in most use cases a valuable dynamically generated parameter.

Hint: A mutable global parameters is no parameter at all. Otherwise it would introduce really huge side effects.

If the parameter is not set a NotFoundException is thrown by di_param. You can suppress this behaviour by supplying a default parameter other than null as second argument. Instead of throwing an exception the default parameter is returned.

dot syntax

In big applications the parameter key names can cause naming conflicts. Therefore it is a good idea to organize your parameter keys in a tree hierarchy like the namespaces in php. Arrays give you that tree hierarchy for free but it is not easy to see the tree behind

It gets even harder if you have to search parameter_name in unknown code as it may be used by several different parameters. Therefore earc/di supports the dot syntax for di_param, di_has_param and di_set_param.

In dot syntax the above would be

import

di_import_param takes an (potentially multidimensional) array as argument. This keeps it flexible for all implementation details and frameworks. You can hardcode your Parameters

or use the popular YAML-format

di_import_param is different to di_set_param as parameters may be overwritten. Thus libraries are able to offer default parameter which may (or may not) be changed by software using this library.

best practice

Define in your Project one ParameterInterface per module. Define all parameter keys (dot syntax) via constants in the ParameterInterface. Let all classes using parameters implement the ParameterInterface of the module.

For example the ParameterInterface of the earc/event-tree let you know all Parameters that might be relevant to you. You don't need to know all the documentation, just one file. Such explicit programming could be a live saver in sparsely documented projects.

factories

If you want to use objects of third party libraries which are not using earc/di, they are sometimes not easy to build. Think of the doctrine EntityManager. It is such a common object in apps using doctrine, it is a serious obstacle to inject a factory instance instead of the real object.

Use di_register_factory instead to registers a callable to a fully qualified class name.

Thereafter the callable is used for building the object.

If an instance was retrieved by get already, you must call clear_cache($fCQN) manually to get the instance from the factory.

If you register a second factory to the same fully qualified class name, the second factory is used. By passing null as factory parameter you can unregister a factory.

decoration

Using an interface as argument of di_get or di_make is a good idea. earc/di uses the argument as type hint. A class as argument states that all potentially injected classes have to inherit from the type hinted class. An interface as argument results in a potentially wider range of classes to inject as they only need to implement the interface. But that freedom comes with a trade-off: earc/di does not know the class to build and inject anymore. Therefore each interface used as parameter for di_get or di_make must be decorated first.

But decoration is much more. It is where the last decision is made what to inject and this as many times as it is needed.

It is where the inversion of control takes place, the service locator part.

Assume you write a library. You add some cool features and bugfixes every month. You have a job, a wife, a few kids, so your reaction time on issues and merge requests does not perform very well. Your library is used as third party library in production nevertheless. After an upgrade an error in a method is detected in production. The programming engineers need a quick way to fix your library. They do not want to fork your library, because keeping a forked library up to date could be a big task. Wouldn't it be nice if they could just exchange the method.

di_decorate does the job. Extending the class, overwriting the erroneous method and decorating the original class is all it needs.

Everywhere the error-prone service was injected now the class with the fix demands its place.

earc/di enables you to decorate abstract classes.

Decoration is something you do in the configuration part of your software. To be not too cumbersome there is a way to do this in a configuration file. On calling DI::importParameter() the parameter earc.di.class_decoration is imported.

Please note the order of the calls is important.

For debugging purpose di_is_decorated and di_get_decorator are handy functions. But be aware that it debugs the current decoration, not the result of a decorator chain.

To clear a decoration decorate a class by itself.

namespace decoration

Sometimes you mirror the file structure from one project in another project. Then it may be desirable to decorate a class automatically by (re)placing it in the mirrored file structure. That is what namespace decoration is for. You can activate this feature by setting the earc.di.namespace_decoration parameter before calling DI::importParameter().

mocking

In most testing libraries mocks are objects. Thus you can't use decoration for replacing the original class. di_mock is made for this use cases.

Keep in mind: Static accessed methods (via di_static) can be mocked via decoration only as di_static returns a string and di_mock takes an object.

You can check if an service is mocked by di_is_mocked.

If you need the real service again use di_clear_mock.

You can even clear all existing mocks.

Mocking is applied after decoration. But the di_*mock functions does not take any decoration into account. That gives you more grip on the mocking process but more care is also needed. Keep in mind, you have to mock the decorator not the decorated class.

As you see in the example code mocks are not forced to follow the type hints. This means you can pass as mock whatever you want. (As long as your code do not pass your Services as type hinted arguments around.)

tagging

Maybe you solve a problem by implementing the chain of responsibility - design pattern. Only the third party software knows which services add to this implementation. This leaves four questions:

  1. How to register to a base service without instantiating it?
  2. How to tell the base service on instantiation without instantiating all handlers?
  3. Where is the best place to write the information?
  4. Where is the best place to store the information?

Three answers solves earc/di for you through tagging.

The third party can store the relevant piece of information by executing

Your base class can retrieve the service classes by iterating over di_get_tagged('tag.name')

If you would have pass a third argument to di_tag $argument would hold this argument instead of null.

Of course the services should implement an interface and the base service should check for it to avoid failure on name conflicts or forgotten methods. And yes, logging handlers not implementing the interface is a good idea. But you know about the architecture your software needs (and can afford) best.

Hint: Decoration is not applied to the tags but to di_get of course.

troubleshooting

earc/di has dropped circular dependency detection in favour of performance. If you experience an error like the following in the earc/di code its cause is most likely a circular dependency. Search for a circular class dependencies in your code.

exceptions

advanced usage

Some usages are not obvious but desirable, like function decoration. If you call your own functions with di_static they can be decorated using di_decorate.

performance considerations

The classes that handle the behaviour of earc/di have in sum round about 250 lines of code, fast array calculations mainly. But nevertheless some big app or server limitations may force a second thought on performance.

Objects retrieved by di_get can not be garbage collected until someone calls di_clear_cache. If you use an object only once or for a tiny moment it might save a bit of memory if you use di_make instead.

Calling di_has performs existence checks for classes and interfaces. This checks can trigger an autoload. Use them wisely. The same is true for di_is_decorated in the case you use namespace decoration.

Namespace decoration uses string substitution which is considerable slower than key lookups on arrays. Nevertheless if you have to decorate hundreds of classes it outperforms explicit configuration since it is used only on classes which are active for the request, wheres configuration must be processed on every request for all decorated classes.

architectural considerations

Don't rely on the singleton behaviour of di_get. Its main purpose are performance considerations. If your architecture need to get always the same instance for a class make it explicit and use a real singleton instead.

In earc/di each type hint is set globally. This means each type hint forces exactly one class. If you feel pushed to use di_decorate outside of the configuration part of your app to change that behaviour you experience a bad architecture smell. It shows that some of your classes demand more than they type hint. Maybe your interfaces do not follow the interface segregation principle, maybe you need just another interface or maybe some of your classes do not follow the single responsibility principle.

integration with other di systems

You are able to completely rewrite the behavioral logic behind the scene. This enables earc/di to integrate with nearly all dependency injection systems. Extend the Resolver::class or the ParameterBag::class to your need or implement the corresponding Interfaces. Register your class(es) by the DI::init method. Now the di_* functions follow the logic you have implemented.

If the third party di-system uses a container then integration is a beginners task. You can find a ready to use example for symfony in the bridge folder. Don't forget to switch your symfony service definitions to public.

If you use symfony register the SymfonyDICompilerPass and you are ready go (or to migrate step by step).

There is no limit. Create your own one to rule them all and make your live easy again.

releases

release 3.1

release 3.0

release 2.4

release 2.3

release 2.2

release 2.1

release 2.0

release 1.0

release 0.1

the first official release


All versions of di with dependencies

PHP Build Version
Package Version
Requires php Version ^8.0 || ^7.2
Composer command for our command line client (download client) This client runs in each environment. You don't need a specific PHP version etc. The first 20 API calls are free. Standard composer command

The package earc/di contains the following files

Loading the files please wait ....