Add DispatchStack
This commit is contained in:
parent
2adcbd8636
commit
560b1e8ff0
|
|
@ -0,0 +1,91 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace WellRESTed\Routing;
|
||||||
|
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Psr\Http\Message\ServerRequestInterface;
|
||||||
|
|
||||||
|
class DispatchStack implements DispatchStackInterface
|
||||||
|
{
|
||||||
|
private $stack;
|
||||||
|
private $dispatcher;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
$this->dispatcher = $this->getDispatcher();
|
||||||
|
$this->stack = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Push a new middleware onto the stack.
|
||||||
|
*
|
||||||
|
* This method MUST preserve the order in which middleware added.
|
||||||
|
*
|
||||||
|
* @param mixed $middleware Middleware to dispatch in sequence
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public function add($middleware)
|
||||||
|
{
|
||||||
|
$this->stack[] = $middleware;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dispatch the contained middleware in the order in which they were added.
|
||||||
|
*
|
||||||
|
* The first middleware added to the stack MUST be the first to be
|
||||||
|
* dispatched.
|
||||||
|
*
|
||||||
|
* Each middleware, when dispatched, MUST receive a $next callable that
|
||||||
|
* dispatches the middleware that follows it. The only exception to this is
|
||||||
|
* the last middleware in the stack which much receive a $next callable the
|
||||||
|
* returns the response unchanged.
|
||||||
|
*
|
||||||
|
* If the instance is dispatched with no middleware added, the instance
|
||||||
|
* MUST call $next passing $request and $response and return the returned
|
||||||
|
* response.
|
||||||
|
*
|
||||||
|
* @param ServerRequestInterface $request
|
||||||
|
* @param ResponseInterface $response
|
||||||
|
* @param callable $next
|
||||||
|
* @return ResponseInterface
|
||||||
|
*/
|
||||||
|
public function dispatch(ServerRequestInterface $request, ResponseInterface $response, $next)
|
||||||
|
{
|
||||||
|
$chain = $this->getCallableChain();
|
||||||
|
$response = $chain($request, $response);
|
||||||
|
return $next($request, $response);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------
|
||||||
|
|
||||||
|
protected function getDispatcher()
|
||||||
|
{
|
||||||
|
return new Dispatcher();
|
||||||
|
}
|
||||||
|
|
||||||
|
// ------------------------------------------------------------------------
|
||||||
|
|
||||||
|
private function getCallableChain()
|
||||||
|
{
|
||||||
|
$dispatcher = $this->dispatcher;
|
||||||
|
|
||||||
|
// No-op function to use as the final middleware's $mext.
|
||||||
|
$next = function ($request, $response) {
|
||||||
|
return $response;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Create a chain of callables.
|
||||||
|
//
|
||||||
|
// Each callable wil take $request and $response parameters, and will
|
||||||
|
// contain a dispatcher, the associated middleware, and a $next
|
||||||
|
// that is the links to the next middleware in the chain.
|
||||||
|
foreach (array_reverse($this->stack) as $middleware) {
|
||||||
|
$next = function ($request, $response) use ($dispatcher, $middleware, $next) {
|
||||||
|
return $dispatcher->dispatch($middleware, $request, $response, $next);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return $next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,38 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace WellRESTed\Routing;
|
||||||
|
|
||||||
|
use Psr\Http\Message\ResponseInterface;
|
||||||
|
use Psr\Http\Message\ServerRequestInterface;
|
||||||
|
|
||||||
|
interface DispatchStackInterface extends MiddlewareInterface
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Push a new middleware onto the stack.
|
||||||
|
*
|
||||||
|
* @param mixed $middleware Middleware to dispatch in sequence
|
||||||
|
* @return self
|
||||||
|
*/
|
||||||
|
public function add($middleware);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dispatch the contained middleware in the order in which they were added.
|
||||||
|
*
|
||||||
|
* The first middleware added to the stack is the first to be dispatched.
|
||||||
|
*
|
||||||
|
* Each middleware, when dispatched, will receive a $next callable that
|
||||||
|
* dispatches the middleware that follows it. The only exception to this is
|
||||||
|
* the last middleware in the stack which much receive a $next callable the
|
||||||
|
* returns the response unchanged.
|
||||||
|
*
|
||||||
|
* If the instance is dispatched with no middleware added, the instance
|
||||||
|
* MUST call $next passing $request and $response and return the returned
|
||||||
|
* response.
|
||||||
|
*
|
||||||
|
* @param ServerRequestInterface $request
|
||||||
|
* @param ResponseInterface $response
|
||||||
|
* @param callable $next
|
||||||
|
* @return ResponseInterface
|
||||||
|
*/
|
||||||
|
public function dispatch(ServerRequestInterface $request, ResponseInterface $response, $next);
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,109 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace WellRESTed\Test\Unit\Routing;
|
||||||
|
|
||||||
|
use Prophecy\Argument;
|
||||||
|
use WellRESTed\Routing\DispatchStack;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @coversDefaultClass WellRESTed\Routing\DispatchStack
|
||||||
|
* @uses WellRESTed\Routing\DispatchStack
|
||||||
|
* @uses WellRESTed\Routing\Dispatcher
|
||||||
|
*/
|
||||||
|
class DispatchStackTest extends \PHPUnit_Framework_TestCase
|
||||||
|
{
|
||||||
|
private $request;
|
||||||
|
private $response;
|
||||||
|
private $next;
|
||||||
|
|
||||||
|
public function setUp()
|
||||||
|
{
|
||||||
|
parent::setUp();
|
||||||
|
$this->request = $this->prophesize('Psr\Http\Message\ServerRequestInterface');
|
||||||
|
$this->response = $this->prophesize('Psr\Http\Message\ResponseInterface');
|
||||||
|
$this->next = function ($request, $response) {
|
||||||
|
return $response;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @covers ::__construct
|
||||||
|
* @covers ::getDispatcher
|
||||||
|
*/
|
||||||
|
public function testCreatesInstance()
|
||||||
|
{
|
||||||
|
$stack = new DispatchStack();
|
||||||
|
$this->assertNotNull($stack);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @covers ::add
|
||||||
|
*/
|
||||||
|
public function testAddIsFluid()
|
||||||
|
{
|
||||||
|
$stack = new DispatchStack();
|
||||||
|
$this->assertSame($stack, $stack->add("middleware1"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @covers ::dispatch
|
||||||
|
*/
|
||||||
|
public function testDispachesMiddlewareInOrderAdded()
|
||||||
|
{
|
||||||
|
// Each middelware will add its "name" to this array.
|
||||||
|
$callOrder = [];
|
||||||
|
|
||||||
|
$stack = new DispatchStack();
|
||||||
|
$stack->add(function ($request, $response, $next) use (&$callOrder) {
|
||||||
|
$callOrder[] = "first";
|
||||||
|
return $next($request, $response);
|
||||||
|
});
|
||||||
|
$stack->add(function ($request, $response, $next) use (&$callOrder) {
|
||||||
|
$callOrder[] = "second";
|
||||||
|
return $next($request, $response);
|
||||||
|
});
|
||||||
|
$stack->add(function ($request, $response, $next) use (&$callOrder) {
|
||||||
|
$callOrder[] = "third";
|
||||||
|
return $next($request, $response);
|
||||||
|
});
|
||||||
|
$stack->dispatch($this->request->reveal(), $this->response->reveal(), $this->next);
|
||||||
|
$this->assertEquals(["first", "second", "third"], $callOrder);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testCallsNextAfterDispatchingStack()
|
||||||
|
{
|
||||||
|
$nextCalled = false;
|
||||||
|
$next = function ($request, $response) use (&$nextCalled) {
|
||||||
|
$nextCalled = true;
|
||||||
|
return $response;
|
||||||
|
};
|
||||||
|
|
||||||
|
$middleware = function ($request, $response, $next) use (&$callOrder) {
|
||||||
|
return $next($request, $response);
|
||||||
|
};
|
||||||
|
|
||||||
|
$stack = new DispatchStack();
|
||||||
|
$stack->add($middleware);
|
||||||
|
$stack->add($middleware);
|
||||||
|
$stack->add($middleware);
|
||||||
|
|
||||||
|
$stack->dispatch($this->request->reveal(), $this->response->reveal(), $next);
|
||||||
|
$this->assertTrue($nextCalled);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @covers ::dispatch
|
||||||
|
*/
|
||||||
|
public function testCallsNextAfterDispatchingEmptyStack()
|
||||||
|
{
|
||||||
|
$nextCalled = false;
|
||||||
|
$next = function ($request, $response) use (&$nextCalled) {
|
||||||
|
$nextCalled = true;
|
||||||
|
return $response;
|
||||||
|
};
|
||||||
|
|
||||||
|
$stack = new DispatchStack();
|
||||||
|
$stack->dispatch($this->request->reveal(), $this->response->reveal(), $next);
|
||||||
|
$this->assertTrue($nextCalled);
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue