This package provides a pipeline pattern implementation. It is based on middleware approach.
PHP version >= 7.1
For php 5.6 use version 1:
"require" : {
"andydune/pipeline": "1.*"
}
Installation using composer:
composer require andydune/pipeline
Or if composer was not installed globally:
php composer.phar require andydune/pipeline
Or edit your composer.json
:
"require" : {
"andydune/pipeline": "^2"
}
And execute command:
php composer.phar update
Operations in pipeline stages, can be anything that satisfies the callable
type-hint. So closures and anything that's invokable is good.
use AndyDune\Pipeline\Pipeline;
$pipeline = new Pipeline();
$stages = [
function($contest, $next) {
$contest += 100;
return $next($contest);
},
function($contest, $next) {
$contest += 10;
return $next($contest);
}
];
$result = $pipeline->through($stages)->send(1)
->then(function ($context) {
$context += 1000;
return $context;});
$result // 1111
Basically each stage can be Closure
:
$pipeline = new Pipeline();
$pipeline->pipe(function ($context, $next) {
$context['closure'] = 'was here';
return $next($context);
});
It can be an instance of class with callable interface:
$instance = new class() {
public function __invoke($context, $next)
{
$context['invoke'] = 'was here';
return $next($context);
}
}
$pipeline = new Pipeline();
$pipeline->pipe($instance);
It can be an instance of class with any method:
$instance = new class() {
public function doIt($context, $next)
{
$context['invoke'] = 'was here';
return $next($context);
}
}
$pipeline = new Pipeline();
$pipeline->pipe($instance, 'doIt');
It can be a class name with method __invoke
or any method you describe:
class Trim
{
public function __invoke($context, $next)
{
$context['class_invoke'] = 'was here';
return $next($context);
}
}
$pipeline = new Pipeline();
$pipeline->pipe(Trim::class);
It can be a class name with any method you describe:
class Trim
{
public function handle($context, $next)
{
$context['class_invoke'] = 'was here';
return $next($context);
}
}
$pipeline = new Pipeline();
$pipeline->pipe(Trim::class, 'handle');
You can use methods which don't execute $next function. It gets some data and return results. There is special method: 'pipeForContainer'
Example class.
namespace AndyDune\Pipeline\Example;
class Methods
{
// return result - no calling next()
public function addBraceLeft($string)
{
return '(' . $string;
}
// It has middleware interface
public function addBraceRight($string, callable $next)
{
$string = $string . ')';
return $next($string);
}
}
$instance = new Methods();
$pipeline = new Pipeline();
$pipeline->send('puh');
$pipeline->pipeForContainer($instance, 'addBraceLeft');
$pipeline->pipe(Methods::class, 'addBraceRight');
$result = $pipeline->execute();
$result == '(puh)';
String can de passed as pipeline stage. By default it is a class name.
It's make possible by using default provider: AndyDune\Pipeline\PipeIsClassName
You can set your own provider, with injections you need. Your class must implements interface: Interop\Container\ContainerInterface
If was not found stage with your provider system will try to use default provider. There is stack of providers.
Use your provider as parameter for pipeline constructor:
use Interop\Container\ContainerInterface;
class SpecialPipelineContainer implements ContainerInterface
{
/**
* @var \Rzn\ObjectBuilder
*/
protected $objectBuilder;
public function get($name)
{
/*
do anything to build object using $name
*/
return $object;
}
public function has($name)
{
/*
do anything to check required object can be build
*/
return $exist;
}
}
$container = new SpecialPipelineContainer();
$pipeLine = new Pipeline($container);
$pipeLine->pipe('any string to get stage from container');
Sometimes you need to pass any number of additional parameters to pipe stage during description. It needs for test and it very good for more flexibility of stage. It increases reuse of class.
Lets look at the example. It is very simple example.
Here is example class for stage:
namespace AndyDune\Pipeline\Example;
class PowerOfNumber
{
public function __invoke($data, callable $next, $power = 2)
{
if (is_array($power)) {
array_walk($power, function (&$value, $key) use ($data) {
$value = pow($data, $value);
});
return $next($power);
}
$data = $this->handle($data, $power);
return $next($data);
}
protected function handle($number, $power)
{
return pow($number, $power);
}
}
Lets use it:
use use AndyDune\Pipeline\Pipeline;
use AndyDune\Pipeline\Example;
$pipeline = new Pipeline();
$pipeline->send(2);
$pipeline->pipe(PowerOfNumber::class);
$result = $pipeline->execute(); // == 4
$pipeline = new Pipeline();
$pipeline->send(2);
$pipeline->pipe(PowerOfNumber::class, null, 3);
$result = $pipeline->execute(); // == 8
$pipeline = new Pipeline();
$pipeline->send(2);
$pipeline->pipe(PowerOfNumber::class, null, 4);
$result = $pipeline->execute(); // == 16
You can use default pipeline stage creator with injection into stage objects services or any objects from outside.
There is injection with interfaces.
use use AndyDune\Pipeline\Pipeline;
$pipeline = new Pipeline();
$pipeline->addInitializer(function($stageObject) use ($someService) {
if ($stageObject instanceof SomeServiceAwareInterface) {
$stageObject->setSomeService($someService)
}
});
$pipeline->pipe(ClassHaveInterface::class);
$result = $pipeline->execute();
This use method addInitializer
which receive callable parameter.
Package has not integrated exception catch support. It is simple for you to include exception try-catch block into one of pipeline stages.
use AndyDune\Pipeline\Pipeline;
$pipeline = new Pipeline();
$pipeline->send(['zub' => 'kovoy']);
$pipeline->pipe(function ($context, $next) {
try {
return $next($context);
} catch (Exception $e) {
$context['exception'] = 'caught';
}
return $context;
});
$pipeline->pipe(function ($context, $next) {
$context['action'] = 'before_exception';
throw new Exception();
return $next($context); // it will be never execute
});
// This stage will never be executed
$pipeline->pipe(function ($context, $next) {
$context['after_exception'] = 'ignored';
return $next($context);
});
$result = $pipeline->execute();
array_key_exists('zub', $result); // true
array_key_exists('exception', $result); // true
array_key_exists('action', $result); // false
array_key_exists('after_exception', $result); // false
There is a class you may use as stage for your pipeline for catch exception in this package.
$pipeline = new Pipeline();
$pipeline->send(['zub' => 'kovoy']);
$pipeline->pipe(AndyDune\Pipeline\Stage\ExceptionCatch::class);
$pipeline->pipe(function ($context, $next) {
$context['action'] = 'before_exception';
throw new Exception('jump');
});
$result = $pipeline->execute();
$result instancheof \Exception // true
You have service for retrieve data. And you don't need to change its code. Just use it as a stage in the pipeline.
// Description
$pipeline = new Pipeline();
$pipeline->pipe(function($key, $next) {
/*
* Cache adapter with (PSR-16) interface
* Ones a have used Symfony Cache Component [https://github.com/symfony/cache]
* It's for example
*/
$cache = new FilesystemCache();
if ($cache->has($key)) {
return $cache->get($key);
}
$data = $next($catId);
$cache->set($key, $data);
return $data;
});
$pipeline->pipe(DataRetrieveClass::class, 'getImportantData');
// Execute
$results = $pipeline->send($key)->execute();
By default string as stage for pipeline means name of class.
If you create pipeline object without parameters service container implements AndyDune\Pipeline\PipeIsClassName
. It only creates instance of given class and return it.
If this package is used as part of of Zend FW 3 you can use Zend's services for retrieve instances.
First you must install package with composer.
Then copy file vendor/andydune/pipeline/config/pipeline.global.php
to config/autoload/pipeline.global.php
Description fot your pipeline in factory:
use Zend\ServiceManager\Factory\FactoryInterface;
use Interop\Container\ContainerInterface;
class DunhillFactory implements FactoryInterface
{
public function __invoke(ContainerInterface $container, $requestedName, array $options = null)
{
$pipeline = $container->get('pipeline');
$pipeline->pipe('service name within Zend3');
$pipeline->pipe(function($data, $next) {
return $next($data);
});
return $pipeline;
}
}
If you don't work with Zend services use pipeline directly.
Laravel pipeline - it do very the same I wanted, but I need more tools for testing purpose. And I don't like laraver service manager. it has not common interface. My pipeline may use service managers based on container-interop with PSR-11 interfaces.
League\Pipeline - it has good documentation. But has bad feature - poor workflow control. To stop execution it need to use exception.