Ensured database connection is closed even if an error is thrown during dispatch process

This commit is contained in:
Alejandro Celaya 2019-07-31 20:08:46 +02:00
parent e6a63a9b85
commit 406de16a0d
2 changed files with 35 additions and 20 deletions

View file

@ -19,16 +19,13 @@ class CloseDbConnectionMiddleware implements MiddlewareInterface
$this->em = $em; $this->em = $em;
} }
/**
* Process an incoming server request and return a response, optionally delegating
* response creation to a handler.
*/
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{ {
$handledRequest = $handler->handle($request); try {
$this->em->getConnection()->close(); return $handler->handle($request);
$this->em->clear(); } finally {
$this->em->getConnection()->close();
return $handledRequest; $this->em->clear();
}
} }
} }

View file

@ -8,6 +8,7 @@ use Doctrine\ORM\EntityManagerInterface;
use PHPUnit\Framework\TestCase; use PHPUnit\Framework\TestCase;
use Prophecy\Prophecy\ObjectProphecy; use Prophecy\Prophecy\ObjectProphecy;
use Psr\Http\Server\RequestHandlerInterface; use Psr\Http\Server\RequestHandlerInterface;
use RuntimeException;
use Shlinkio\Shlink\Common\Middleware\CloseDbConnectionMiddleware; use Shlinkio\Shlink\Common\Middleware\CloseDbConnectionMiddleware;
use Zend\Diactoros\Response; use Zend\Diactoros\Response;
use Zend\Diactoros\ServerRequest; use Zend\Diactoros\ServerRequest;
@ -20,35 +21,52 @@ class CloseDbConnectionMiddlewareTest extends TestCase
private $handler; private $handler;
/** @var ObjectProphecy */ /** @var ObjectProphecy */
private $em; private $em;
/** @var ObjectProphecy */
private $conn;
public function setUp(): void public function setUp(): void
{ {
$this->handler = $this->prophesize(RequestHandlerInterface::class); $this->handler = $this->prophesize(RequestHandlerInterface::class);
$this->em = $this->prophesize(EntityManagerInterface::class); $this->em = $this->prophesize(EntityManagerInterface::class);
$this->conn = $this->prophesize(Connection::class);
$this->conn->close()->will(function () {
});
$this->em->getConnection()->willReturn($this->conn->reveal());
$this->em->clear()->will(function () {
});
$this->middleware = new CloseDbConnectionMiddleware($this->em->reveal()); $this->middleware = new CloseDbConnectionMiddleware($this->em->reveal());
} }
/** @test */ /** @test */
public function connectionIsClosedWhenMiddlewareIsProcessed() public function connectionIsClosedWhenMiddlewareIsProcessed(): void
{ {
$req = new ServerRequest(); $req = new ServerRequest();
$resp = new Response(); $resp = new Response();
$conn = $this->prophesize(Connection::class);
$closeConn = $conn->close()->will(function () {
});
$getConn = $this->em->getConnection()->willReturn($conn->reveal());
$clear = $this->em->clear()->will(function () {
});
$handle = $this->handler->handle($req)->willReturn($resp); $handle = $this->handler->handle($req)->willReturn($resp);
$result = $this->middleware->process($req, $this->handler->reveal()); $result = $this->middleware->process($req, $this->handler->reveal());
$this->assertSame($result, $resp); $this->assertSame($result, $resp);
$getConn->shouldHaveBeenCalledOnce(); $this->em->getConnection()->shouldHaveBeenCalledOnce();
$closeConn->shouldHaveBeenCalledOnce(); $this->conn->close()->shouldHaveBeenCalledOnce();
$clear->shouldHaveBeenCalledOnce(); $this->em->clear()->shouldHaveBeenCalledOnce();
$handle->shouldHaveBeenCalledOnce(); $handle->shouldHaveBeenCalledOnce();
} }
/** @test */
public function connectionIsClosedEvenIfExceptionIsThrownOnInnerMiddlewares(): void
{
$req = new ServerRequest();
$expectedError = new RuntimeException();
$this->handler->handle($req)->willThrow($expectedError)
->shouldBeCalledOnce();
$this->em->getConnection()->shouldBeCalledOnce();
$this->conn->close()->shouldBeCalledOnce();
$this->em->clear()->shouldBeCalledOnce();
$this->expectExceptionObject($expectedError);
$this->middleware->process($req, $this->handler->reveal());
}
} }