---eonx_docs--- title: Introduction weight: 0 ---eonx_docs---
This package provides helper classes.
The recommended way to install this package is to use Composer:
$ composer require eonx-com/easy-utils
CollectorHelper
: provides methods to facilitate implementation of the Collector Design PatternMath
: provides methods to facilitate numbers manipulation
The Collector Design Pattern is a great method for keeping your code SOLID. However, using it in multiple parts of
your project can lead to significant repetition. The main purpose of the CollectorHelper
is to prevent duplicated code
and facilitate implementation of the Collector Design Pattern in your project.
Most popular PHP frameworks provide features to tag services, and then define all services for a specific tag as dependencies to other services. See the following resources for examples:
Those features help you implement the Collector Design Pattern in your project because they allow you to easily inject a collection of services sharing the same tag into other services.
However, there are some things you need to consider:
- There is no guarantee that all given services are instances of a specific class/interface
- You have no control on the order the services are organised within the given collection
Let's elaborate on the above points.
The service tagging features do not allow you to ensure all services sharing the same tag meet common criteria. Symfony has a feature to automatically tag services based on their class, but nothing stops you from manually tagging a service with the same tag or even one of your dependencies.
This is why we strongly recommend you always filter the given iterable
of services by a given class/interface of your
choice by using the filterByClass()
or filterByClassAsArray()
methods of the CollectorHelper
.
When using service tagging features, you can control the order that the services are organised by simply changing the
order in which you define the services. However, as above, there is nothing stopping you or one of your dependencies
from tagging a service with the same tag. Therefore, you cannot guarantee the order as you cannot modify the
dependencies' service definitions. But the CollectorHelper
can help us!
In some cases, the order of the given services does not matter, so there is no need to do anything. But if your logic
requires the services be used in a specific order, then use the orderHigherPriorityFirst()
and/or
orderLowerPriorityFirst()
methods!
These methods will sort the objects within the given iterable
based on their priority. In order to define an object's
priority, it must implement the EonX\EasyUtils\Common\Helper\HasPriorityInterface
provided by this package. If an object
doesn't implement this interface then its priority will default to 0
automatically.
The convertToArray()
method will convert any iterable to a simple PHP array. It is useful when you want to use array
methods on an iterable
.
For a simple example of when to use the convertToArray()
method, imagine you have a class which accepts an iterable
of "workers" in its constructor. To safely use these "workers", you want to ensure each of them implements the right
interface, so you filter them to keep only the "good workers" by using the array_filter()
function. If the "workers"
were already an array
, then there would be no problem. However, because they are defined as iterable
, you cannot
guarantee you will receive an array
. So use the convertToArray()
method!
use App\Domain\WorkerInterface;
use EonX\EasyUtils\Common\Helper\CollectorHelper;
final class MyClass
{
/**
* @var \App\Domain\WorkerInterface[]
*/
private array $workers;
public function __construct(iterable $workers)
{
// $workers could be any type of iterable, convert it to array
$workers = CollectorHelper::convertToArray($workers);
// Now we are sure $workers is an array, we can use array_filter()
$workers = \array_filter($workers, static function ($worker): bool {
return $worker instanceof WorkerInterface;
});
// $workers is now an array of WorkerInterface for sure
$this->workers = $workers;
}
}
The use case of filtering by class (used above to explain the convertToArray()
method) is very common (at least in our
projects 😃 ), which is why CollectorHelper
provides the filterByClass()
method to do it for you.
The following example is the same as for the convertToArray()
method above. If you have an iterable
and you want to
ensure each item is an instance of a specific class/interface, use the filterByClass()
method!
use App\Domain\WorkerInterface;
use EonX\EasyUtils\Common\Helper\CollectorHelper;
final class MyClass
{
/**
* @var iterable<\App\Domain\WorkerInterface>
*/
private array $workers;
public function __construct(iterable $workers)
{
// $workers now contains only WorkerInterface instances
$workers = CollectorHelper::filterByClass($workers, WorkerInterface::class);
// The filterByClass() method still returns an iterable, a generator more precisely
// If you need an array, you can use the filterByClassAsArray() method
$this->workers = $workers;
}
}
::: tip
The filterByClass()
method still returns an iterable (or, more precisely, a generator). If you need an array
, you
can use the filterByClassAsArray()
method instead.
:::
This method is similar to the filterByClass()
method, but with a little tweak. If you have an iterable
and you want
to make sure each item is an instance of a specific class/interface, but you need the output to be an array
, use the
filterByClassAsArray()
method!
use App\Domain\WorkerInterface;
use EonX\EasyUtils\Common\Helper\CollectorHelper;
final class MyClass
{
/**
* @var \App\Domain\WorkerInterface[]
*/
private array $workers;
public function __construct(iterable $workers)
{
// $workers now contains only WorkerInterface instances
$workers = CollectorHelper::filterByClassAsArray($workers, WorkerInterface::class);
// $workers is now an array containing only WorkerInterface instances
$this->workers = $workers;
}
}
Those methods are similar to the filterByClass()
and filterByClassAsArray()
methods, however they will throw an
exception if at least of the items is not an instance of the given class.
use App\Domain\WorkerInterface;
use EonX\EasyUtils\Common\Helper\CollectorHelper;
final class MyClass
{
/**
* @var \App\Domain\WorkerInterface[]
*/
private array $workers;
public function __construct(iterable $workers)
{
// $workers now contains only WorkerInterface instances
$workers = CollectorHelper::ensureClass(WorkerInterface::class, $workers);
foreach ($workers as $worker) {
// This code will be executed only if all items are instances of WorkerInterface
}
}
}
::: warning
Please note that with the ensureClass()
method, the exception will be thrown only when iterating through the generator.
:::
The orderHigherPriorityFirst()
method will ensure the object with the highest priority is placed first, and the object
with the lowest priority is placed last.
In order to define an object's priority, it must implement the EonX\EasyUtils\Common\Helper\HasPriorityInterface
provided
by this package. If an object doesn't implement this interface then its priority will default to 0
automatically.
// Foo and Bar both implement EonX\EasyUtils\Common\Helper\HasPriorityInterface
$foo = new Foo(); // Has a priority of 10
$bar = new Bar(); // Has a priority of 100
// $foo is added to the array first, and $bar second
$objects = [$foo, $bar];
// $bar is now first as it has a higher priority than $foo
$objects = CollectorHelper::orderHigherPriorityFirst($objects); // [$bar, $foo]
::: tip
The orderHigherPriorityFirst()
method still returns an iterable (or, more precisely, a generator). If you need an array
, you
can use the orderHigherPriorityFirstAsArray()
method instead.
:::
The orderLowerPriorityFirst()
method is the opposite of orderHigherPriorityFirst()
. It will ensure the object with
the lowest priority is placed first, and the object with the highest priority is placed last.
In order to define an object's priority, it must implement the EonX\EasyUtils\Common\Helper\HasPriorityInterface
provided
by this package. If an object doesn't implement this interface then its priority will default to 0
automatically.
// Foo and Bar both implement EonX\EasyUtils\Common\Helper\HasPriorityInterface
$foo = new Foo(); // Has a priority of 10
$bar = new Bar(); // Has a priority of 100
// $foo is added to the array first, and $bar second
$objects = [$foo, $bar];
// $foo is still first as it has a lower priority than $bar
$objects = CollectorHelper::orderLowerPriorityFirst($objects); // [$foo, $bar]
::: tip
The orderLowerPriorityFirst()
method still returns an iterable (or, more precisely, a generator). If you need an array
, you
can use the orderLowerPriorityFirstAsArray()
method instead.
:::
The Math helper provides the following methods:
abs:
returns the absolute value for the given numberadd:
adds two numbers and returns the resultcomp:
compares two numbersdivide:
divides one number by the other and returns the resultmultiply:
multiplies one number by the other and returns the resultround:
rounds the given number and returns the resultsub:
subs tow numbers and returns the result