Skip to content

Commit

Permalink
[FEATURE] Allow deactivation of PSR-15 middlewares
Browse files Browse the repository at this point in the history
Similar to other places that use dependency configurations
via arrays, the middleware stack configuration now allows
disabling single middlewares by setting disabled=true.

Change-Id: I42c741062b5f6952577e164939593f0553b1ad31
Resolves: #83907
Related: #83906
Related: #83725
Releases: master
Reviewed-on: https://review.typo3.org/55725
Reviewed-by: Mathias Schreiber <mathias.schreiber@typo3.com>
Tested-by: Mathias Schreiber <mathias.schreiber@typo3.com>
Tested-by: TYPO3com <no-reply@typo3.com>
Reviewed-by: Anja Leichsenring <aleichsenring@ab-softlab.de>
Tested-by: Anja Leichsenring <aleichsenring@ab-softlab.de>
Reviewed-by: Benjamin Franzke <bfr@qbus.de>
Reviewed-by: Christian Kuhn <lolli@schwarzbu.ch>
Tested-by: Christian Kuhn <lolli@schwarzbu.ch>
  • Loading branch information
lolli42 committed Feb 17, 2018
1 parent 85ab2f5 commit 6239953
Show file tree
Hide file tree
Showing 7 changed files with 151 additions and 19 deletions.
4 changes: 4 additions & 0 deletions typo3/sysext/core/Classes/Http/MiddlewareStackResolver.php
Expand Up @@ -118,6 +118,10 @@ protected function sanitizeMiddlewares(array $allMiddlewares): array

$sanitizedMiddlewares = [];
foreach ($middlewaresOfStack as $name => $middleware) {
if (isset($middleware['disabled']) && $middleware['disabled'] === true) {
// Skip this middleware if disabled by configuration
continue;
}
$sanitizedMiddlewares[$name] = $middleware['target'];
}

Expand Down
2 changes: 1 addition & 1 deletion typo3/sysext/core/Classes/Http/NormalizedParams.php
Expand Up @@ -21,7 +21,7 @@
/**
* This class provides normalized server parameters in HTTP request context.
* It normalizes reverse proxy scenarios and various other web server specific differences
* of native the PSR-7 request object parameters (->getServerParams() / $GLOBALS['_SERVER']).
* of the native PSR-7 request object parameters (->getServerParams() / $GLOBALS['_SERVER']).
*
* An instance of this class is available as PSR-7 ServerRequestInterface attribute:
* $normalizedParams = $request->getAttribute('normalizedParams')
Expand Down
Expand Up @@ -37,21 +37,34 @@ To add a middleware to the "frontend" or "backend" middleware stack, create the

.. code-block:: php
return [
// stack name: currently 'frontend' or 'backend'
'frontend' => [
'middleware-identifier' => [
'target' => \ACME\Ext\Middleware::class,
'description' => '',
'before' => [
'another-middleware-identifier',
],
'after' => [
'yet-another-middleware-identifier',
],
]
]
];
.. index:: Backend, Frontend, PHP-API, NotScanned
return [
// stack name: currently 'frontend' or 'backend'
'frontend' => [
'middleware-identifier' => [
'target' => \ACME\Ext\Middleware::class,
'description' => '',
'before' => [
'another-middleware-identifier',
],
'after' => [
'yet-another-middleware-identifier',
],
]
]
];
If extensions need to shut down or substitute existing middlewares with an own solution, they can
disable an existing middleware by adding the following code in :file:`Configuration/RequestMiddlewares.php`: of their
extension.

.. code-block:: php
return [
'frontend' => [
'middleware-identifier' => [
'disabled' => true,
],
],
];
.. index:: Backend, Frontend, PHP-API
@@ -0,0 +1,8 @@
<?php
return [
'testStack' => [
'firstMiddleware' => [
'target' => 'aClassName',
],
]
];
@@ -0,0 +1,8 @@
<?php
return [
'testStack' => [
'secondMiddleware' => [
'target' => 'anotherClassName',
],
]
];
@@ -0,0 +1,11 @@
<?php
return [
'testStack' => [
'firstMiddleware' => [
'disabled' => true,
],
'secondMiddleware' => [
'target' => 'anotherClassName',
],
]
];
88 changes: 88 additions & 0 deletions typo3/sysext/core/Tests/Unit/Http/MiddlewareStackResolverTest.php
@@ -0,0 +1,88 @@
<?php
declare(strict_types = 1);
namespace TYPO3\CMS\Core\Tests\Unit\Http;

/*
* This file is part of the TYPO3 CMS project.
*
* It is free software; you can redistribute it and/or modify it under
* the terms of the GNU General Public License, either version 2
* of the License, or any later version.
*
* For the full copyright and license information, please read the
* LICENSE.txt file that was distributed with this source code.
*
* The TYPO3 project - inspiring people to share!
*/

use Prophecy\Argument;
use TYPO3\CMS\Core\Cache\Frontend\PhpFrontend;
use TYPO3\CMS\Core\Http\MiddlewareStackResolver;
use TYPO3\CMS\Core\Package\Package;
use TYPO3\CMS\Core\Package\PackageManager;
use TYPO3\CMS\Core\Service\DependencyOrderingService;
use TYPO3\TestingFramework\Core\Unit\UnitTestCase;

/**
* Test case
*/
class MiddlewareStackResolverTest extends UnitTestCase
{
/**
* @test
*/
public function resolveReturnsMiddlewareStack()
{
$package1 = $this->prophesize(Package::class);
$package1->getPackagePath()->willReturn(__DIR__ . '/' . 'Fixtures/Package1/');
$package2 = $this->prophesize(Package::class);
$package2->getPackagePath()->willReturn(__DIR__ . '/' . 'Fixtures/Package2/');
$packageManagerProphecy = $this->prophesize(PackageManager::class);
$packageManagerProphecy->getActivePackages()->willReturn([$package1->reveal(), $package2->reveal()]);
$dependencyOrderingServiceProphecy = $this->prophesize(DependencyOrderingService::class);
$dependencyOrderingServiceProphecy->orderByDependencies(Argument::cetera())->willReturnArgument(0);
$phpFrontendCacheProphecy = $this->prophesize(PhpFrontend::class);
$phpFrontendCacheProphecy->has(Argument::cetera())->willReturn(false);
$phpFrontendCacheProphecy->set(Argument::cetera())->willReturn(false);

$subject = new MiddlewareStackResolver(
$packageManagerProphecy->reveal(),
$dependencyOrderingServiceProphecy->reveal(),
$phpFrontendCacheProphecy->reveal()
);
$expected = [
'secondMiddleware' => 'anotherClassName',
'firstMiddleware' => 'aClassName',
];
$this->assertEquals($expected, $subject->resolve('testStack'));
}

/**
* @test
*/
public function resolveAllowsDisablingAMiddleware()
{
$package1 = $this->prophesize(Package::class);
$package1->getPackagePath()->willReturn(__DIR__ . '/' . 'Fixtures/Package1/');
$package2 = $this->prophesize(Package::class);
$package2->getPackagePath()->willReturn(__DIR__ . '/' . 'Fixtures/Package2Disables1/');
$packageManagerProphecy = $this->prophesize(PackageManager::class);
$packageManagerProphecy->getActivePackages()->willReturn([$package1->reveal(), $package2->reveal()]);
$dependencyOrderingServiceProphecy = $this->prophesize(DependencyOrderingService::class);
$dependencyOrderingServiceProphecy->orderByDependencies(Argument::cetera())->willReturnArgument(0);
$phpFrontendCacheProphecy = $this->prophesize(PhpFrontend::class);
$phpFrontendCacheProphecy->has(Argument::cetera())->willReturn(false);
$phpFrontendCacheProphecy->set(Argument::cetera())->willReturn(false);

$subject = new MiddlewareStackResolver(
$packageManagerProphecy->reveal(),
$dependencyOrderingServiceProphecy->reveal(),
$phpFrontendCacheProphecy->reveal()
);
$expected = [
// firstMiddleware is missing, RequestMiddlewares.php of Package2 sets disables=true on firstMiddleware
'secondMiddleware' => 'anotherClassName',
];
$this->assertEquals($expected, $subject->resolve('testStack'));
}
}

0 comments on commit 6239953

Please sign in to comment.