Additional functionality for Middleware extender

Implements the remove, insertBefore, insertAfter and replace
functionality for middlewares.

The IoC container now holds one array of middleware (bindings) per
frontend stack - the extender operates on that array, before it is
wrapped in a middleware "pipe".

Fixes #1957, closes #1971.
This commit is contained in:
Matthew Kilgore 2020-01-24 21:20:08 +01:00 committed by Franz Liedke
parent 6f6958dd6b
commit 93c6babc61
5 changed files with 125 additions and 47 deletions

View File

@ -49,7 +49,19 @@ class AdminServiceProvider extends AbstractServiceProvider
return $routes; return $routes;
}); });
$this->app->singleton('flarum.admin.middleware', function (Application $app) { $this->app->singleton('flarum.admin.middleware', function () {
return [
HttpMiddleware\ParseJsonBody::class,
HttpMiddleware\StartSession::class,
HttpMiddleware\RememberFromCookie::class,
HttpMiddleware\AuthenticateWithSession::class,
HttpMiddleware\CheckCsrfToken::class,
HttpMiddleware\SetLocale::class,
Middleware\RequireAdministrateAbility::class,
];
});
$this->app->singleton('flarum.admin.handler', function (Application $app) {
$pipe = new MiddlewarePipe; $pipe = new MiddlewarePipe;
// All requests should first be piped through our global error handler // All requests should first be piped through our global error handler
@ -59,21 +71,15 @@ class AdminServiceProvider extends AbstractServiceProvider
$app->tagged(Reporter::class) $app->tagged(Reporter::class)
)); ));
$pipe->pipe($app->make(HttpMiddleware\ParseJsonBody::class)); foreach ($this->app->make('flarum.admin.middleware') as $middleware) {
$pipe->pipe($app->make(HttpMiddleware\StartSession::class)); $pipe->pipe($this->app->make($middleware));
$pipe->pipe($app->make(HttpMiddleware\RememberFromCookie::class)); }
$pipe->pipe($app->make(HttpMiddleware\AuthenticateWithSession::class));
$pipe->pipe($app->make(HttpMiddleware\CheckCsrfToken::class));
$pipe->pipe($app->make(HttpMiddleware\SetLocale::class));
$pipe->pipe($app->make(Middleware\RequireAdministrateAbility::class));
event(new ConfigureMiddleware($pipe, 'admin')); event(new ConfigureMiddleware($pipe, 'admin'));
return $pipe;
});
$this->app->afterResolving('flarum.admin.middleware', function (MiddlewarePipe $pipe) {
$pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.admin.routes'))); $pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.admin.routes')));
return $pipe;
}); });
$this->app->bind('flarum.assets.admin', function () { $this->app->bind('flarum.assets.admin', function () {

View File

@ -45,7 +45,20 @@ class ApiServiceProvider extends AbstractServiceProvider
return $routes; return $routes;
}); });
$this->app->singleton('flarum.api.middleware', function (Application $app) { $this->app->singleton('flarum.api.middleware', function () {
return [
HttpMiddleware\ParseJsonBody::class,
Middleware\FakeHttpMethods::class,
HttpMiddleware\StartSession::class,
HttpMiddleware\RememberFromCookie::class,
HttpMiddleware\AuthenticateWithSession::class,
HttpMiddleware\AuthenticateWithHeader::class,
HttpMiddleware\CheckCsrfToken::class,
HttpMiddleware\SetLocale::class,
];
});
$this->app->singleton('flarum.api.handler', function (Application $app) {
$pipe = new MiddlewarePipe; $pipe = new MiddlewarePipe;
$pipe->pipe(new HttpMiddleware\HandleErrors( $pipe->pipe(new HttpMiddleware\HandleErrors(
@ -54,22 +67,15 @@ class ApiServiceProvider extends AbstractServiceProvider
$app->tagged(Reporter::class) $app->tagged(Reporter::class)
)); ));
$pipe->pipe($app->make(HttpMiddleware\ParseJsonBody::class)); foreach ($this->app->make('flarum.api.middleware') as $middleware) {
$pipe->pipe($app->make(Middleware\FakeHttpMethods::class)); $pipe->pipe($this->app->make($middleware));
$pipe->pipe($app->make(HttpMiddleware\StartSession::class)); }
$pipe->pipe($app->make(HttpMiddleware\RememberFromCookie::class));
$pipe->pipe($app->make(HttpMiddleware\AuthenticateWithSession::class));
$pipe->pipe($app->make(HttpMiddleware\AuthenticateWithHeader::class));
$pipe->pipe($app->make(HttpMiddleware\CheckCsrfToken::class));
$pipe->pipe($app->make(HttpMiddleware\SetLocale::class));
event(new ConfigureMiddleware($pipe, 'api')); event(new ConfigureMiddleware($pipe, 'api'));
return $pipe;
});
$this->app->afterResolving('flarum.api.middleware', function (MiddlewarePipe $pipe) {
$pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.api.routes'))); $pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.api.routes')));
return $pipe;
}); });
} }

View File

@ -11,11 +11,14 @@ namespace Flarum\Extend;
use Flarum\Extension\Extension; use Flarum\Extension\Extension;
use Illuminate\Contracts\Container\Container; use Illuminate\Contracts\Container\Container;
use Laminas\Stratigility\MiddlewarePipe;
class Middleware implements ExtenderInterface class Middleware implements ExtenderInterface
{ {
protected $middlewares = []; protected $addMiddlewares = [];
protected $removeMiddlewares = [];
protected $replaceMiddlewares = [];
protected $insertBeforeMiddlewares = [];
protected $insertAfterMiddlewares = [];
protected $frontend; protected $frontend;
public function __construct(string $frontend) public function __construct(string $frontend)
@ -25,17 +28,74 @@ class Middleware implements ExtenderInterface
public function add($middleware) public function add($middleware)
{ {
$this->middlewares[] = $middleware; $this->addMiddlewares[] = $middleware;
return $this;
}
public function replace($originalMiddleware, $newMiddleware)
{
$this->replaceMiddlewares[$originalMiddleware] = $newMiddleware;
return $this;
}
public function remove($middleware)
{
$this->removeMiddlewares[] = $middleware;
return $this;
}
public function insertBefore($originalMiddleware, $newMiddleware)
{
$this->replaceMiddlewares[$originalMiddleware] = $newMiddleware;
return $this;
}
public function insertAfter($originalMiddleware, $newMiddleware)
{
$this->replaceMiddlewares[$originalMiddleware] = $newMiddleware;
return $this; return $this;
} }
public function extend(Container $container, Extension $extension = null) public function extend(Container $container, Extension $extension = null)
{ {
$container->resolving("flarum.{$this->frontend}.middleware", function (MiddlewarePipe $pipe) use ($container) { $container->extend("flarum.{$this->frontend}.middleware", function ($existingMiddleware) {
foreach ($this->middlewares as $middleware) { foreach ($this->addMiddlewares as $addMiddleware) {
$pipe->pipe($container->make($middleware)); $existingMiddleware[] = $addMiddleware;
} }
foreach ($this->replaceMiddlewares as $originalMiddleware => $newMiddleware) {
$existingMiddleware = array_replace($existingMiddleware,
array_fill_keys(
array_keys($existingMiddleware, $originalMiddleware),
$newMiddleware
)
);
}
foreach ($this->insertBeforeMiddlewares as $originalMiddleware => $newMiddleware) {
array_splice($existingMiddleware,
array_search($originalMiddleware, $existingMiddleware),
0,
$newMiddleware
);
}
foreach ($this->insertAfterMiddlewares as $originalMiddleware => $newMiddleware) {
array_splice($existingMiddleware,
array_search($originalMiddleware, $existingMiddleware) + 1,
0,
$newMiddleware
);
}
$existingMiddleware = array_diff($existingMiddleware, $this->removeMiddlewares);
return $existingMiddleware;
}); });
} }
} }

View File

@ -59,7 +59,20 @@ class ForumServiceProvider extends AbstractServiceProvider
$this->setDefaultRoute($routes); $this->setDefaultRoute($routes);
}); });
$this->app->singleton('flarum.forum.middleware', function (Application $app) { $this->app->singleton('flarum.forum.middleware', function () {
return [
HttpMiddleware\ParseJsonBody::class,
HttpMiddleware\CollectGarbage::class,
HttpMiddleware\StartSession::class,
HttpMiddleware\RememberFromCookie::class,
HttpMiddleware\AuthenticateWithSession::class,
HttpMiddleware\CheckCsrfToken::class,
HttpMiddleware\SetLocale::class,
HttpMiddleware\ShareErrorsFromSession::class
];
});
$this->app->singleton('flarum.forum.handler', function (Application $app) {
$pipe = new MiddlewarePipe; $pipe = new MiddlewarePipe;
// All requests should first be piped through our global error handler // All requests should first be piped through our global error handler
@ -69,22 +82,15 @@ class ForumServiceProvider extends AbstractServiceProvider
$app->tagged(Reporter::class) $app->tagged(Reporter::class)
)); ));
$pipe->pipe($app->make(HttpMiddleware\ParseJsonBody::class)); foreach ($this->app->make('flarum.forum.middleware') as $middleware) {
$pipe->pipe($app->make(HttpMiddleware\CollectGarbage::class)); $pipe->pipe($this->app->make($middleware));
$pipe->pipe($app->make(HttpMiddleware\StartSession::class)); }
$pipe->pipe($app->make(HttpMiddleware\RememberFromCookie::class));
$pipe->pipe($app->make(HttpMiddleware\AuthenticateWithSession::class));
$pipe->pipe($app->make(HttpMiddleware\CheckCsrfToken::class));
$pipe->pipe($app->make(HttpMiddleware\SetLocale::class));
$pipe->pipe($app->make(HttpMiddleware\ShareErrorsFromSession::class));
event(new ConfigureMiddleware($pipe, 'forum')); event(new ConfigureMiddleware($pipe, 'forum'));
return $pipe;
});
$this->app->afterResolving('flarum.forum.middleware', function (MiddlewarePipe $pipe) {
$pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.forum.routes'))); $pipe->pipe(new HttpMiddleware\DispatchRoute($this->app->make('flarum.forum.routes')));
return $pipe;
}); });
$this->app->bind('flarum.assets.forum', function () { $this->app->bind('flarum.assets.forum', function () {

View File

@ -63,9 +63,9 @@ class InstalledApp implements AppInterface
$pipe->pipe(new OriginalMessages); $pipe->pipe(new OriginalMessages);
$pipe->pipe( $pipe->pipe(
new BasePathRouter([ new BasePathRouter([
$this->subPath('api') => 'flarum.api.middleware', $this->subPath('api') => 'flarum.api.handler',
$this->subPath('admin') => 'flarum.admin.middleware', $this->subPath('admin') => 'flarum.admin.handler',
'/' => 'flarum.forum.middleware', '/' => 'flarum.forum.handler',
]) ])
); );
$pipe->pipe(new RequestHandler($this->container)); $pipe->pipe(new RequestHandler($this->container));