Skip to content

Commit

Permalink
Fixes wrong IP address when using a reverse proxy (#2236)
Browse files Browse the repository at this point in the history
Added reverse proxy support to preserve forwarded IPs
  • Loading branch information
x7airworker authored Jul 22, 2020
1 parent eaac786 commit 451a557
Show file tree
Hide file tree
Showing 7 changed files with 128 additions and 2 deletions.
11 changes: 11 additions & 0 deletions src/Admin/AdminServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
use Flarum\Http\UrlGenerator;
use Flarum\Locale\LocaleManager;
use Flarum\Settings\Event\Saved;
use Illuminate\Support\Arr;
use Laminas\Stratigility\MiddlewarePipe;

class AdminServiceProvider extends AbstractServiceProvider
Expand All @@ -48,6 +49,7 @@ public function register()
$this->app->singleton('flarum.admin.middleware', function () {
return [
'flarum.admin.error_handler',
'flarum.admin.proxy_middleware',
HttpMiddleware\ParseJsonBody::class,
HttpMiddleware\StartSession::class,
HttpMiddleware\RememberFromCookie::class,
Expand All @@ -66,6 +68,15 @@ public function register()
);
});

$this->app->bind('flarum.admin.proxy_middleware', function () {
$config = $this->app->make('flarum.config');

return new HttpMiddleware\ProxyAddress(
Arr::get($config, 'reverse_proxy.enabled', false),
Arr::get($config, 'reverse_proxy.allowed', ['127.0.0.1'])
);
});

$this->app->singleton('flarum.admin.handler', function () {
$pipe = new MiddlewarePipe;

Expand Down
11 changes: 11 additions & 0 deletions src/Api/ApiServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
use Flarum\Http\RouteCollection;
use Flarum\Http\RouteHandlerFactory;
use Flarum\Http\UrlGenerator;
use Illuminate\Support\Arr;
use Laminas\Stratigility\MiddlewarePipe;

class ApiServiceProvider extends AbstractServiceProvider
Expand All @@ -45,6 +46,7 @@ public function register()
$this->app->singleton('flarum.api.middleware', function () {
return [
'flarum.api.error_handler',
'flarum.api.proxy_middleware',
HttpMiddleware\ParseJsonBody::class,
Middleware\FakeHttpMethods::class,
HttpMiddleware\StartSession::class,
Expand All @@ -64,6 +66,15 @@ public function register()
);
});

$this->app->bind('flarum.api.proxy_middleware', function () {
$config = $this->app->make('flarum.config');

return new HttpMiddleware\ProxyAddress(
Arr::get($config, 'reverse_proxy.enabled', false),
Arr::get($config, 'reverse_proxy.allowed', ['127.0.0.1'])
);
});

$this->app->singleton('flarum.api.handler', function () {
$pipe = new MiddlewarePipe;

Expand Down
2 changes: 1 addition & 1 deletion src/Api/Controller/CreateDiscussionController.php
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ public function __construct(Dispatcher $bus, Floodgate $floodgate)
protected function data(ServerRequestInterface $request, Document $document)
{
$actor = $request->getAttribute('actor');
$ipAddress = Arr::get($request->getServerParams(), 'REMOTE_ADDR', '127.0.0.1');
$ipAddress = $request->getAttribute('ipAddress');

if (! $request->getAttribute('bypassFloodgate')) {
$this->floodgate->assertNotFlooding($actor);
Expand Down
2 changes: 1 addition & 1 deletion src/Api/Controller/CreatePostController.php
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ protected function data(ServerRequestInterface $request, Document $document)
$actor = $request->getAttribute('actor');
$data = Arr::get($request->getParsedBody(), 'data', []);
$discussionId = Arr::get($data, 'relationships.discussion.data.id');
$ipAddress = Arr::get($request->getServerParams(), 'REMOTE_ADDR', '127.0.0.1');
$ipAddress = $request->getAttribute('ipAddress');

if (! $request->getAttribute('bypassFloodgate')) {
$this->floodgate->assertNotFlooding($actor);
Expand Down
11 changes: 11 additions & 0 deletions src/Forum/ForumServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@
use Flarum\Settings\Event\Saved;
use Flarum\Settings\Event\Saving;
use Flarum\Settings\SettingsRepositoryInterface;
use Illuminate\Support\Arr;
use Laminas\Stratigility\MiddlewarePipe;
use Symfony\Component\Translation\TranslatorInterface;

Expand Down Expand Up @@ -57,6 +58,7 @@ public function register()
$this->app->singleton('flarum.forum.middleware', function () {
return [
'flarum.forum.error_handler',
'flarum.forum.proxy_middleware',
HttpMiddleware\ParseJsonBody::class,
HttpMiddleware\CollectGarbage::class,
HttpMiddleware\StartSession::class,
Expand All @@ -76,6 +78,15 @@ public function register()
);
});

$this->app->bind('flarum.forum.proxy_middleware', function () {
$config = $this->app->make('flarum.config');

return new HttpMiddleware\ProxyAddress(
Arr::get($config, 'reverse_proxy.enabled', false),
Arr::get($config, 'reverse_proxy.allowed', ['127.0.0.1'])
);
});

$this->app->singleton('flarum.forum.handler', function () {
$pipe = new MiddlewarePipe;

Expand Down
21 changes: 21 additions & 0 deletions src/Http/Exception/ProxyNotAllowedException.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php

/*
* This file is part of Flarum.
*
* For detailed copyright and license information, please view the
* LICENSE file that was distributed with this source code.
*/

namespace Flarum\Http\Exception;

use Exception;
use Flarum\Foundation\KnownError;

class ProxyNotAllowedException extends Exception implements KnownError
{
public function getType(): string
{
return 'reverse_proxy_not_allowed';
}
}
72 changes: 72 additions & 0 deletions src/Http/Middleware/ProxyAddress.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,72 @@
<?php

/*
* This file is part of Flarum.
*
* For detailed copyright and license information, please view the
* LICENSE file that was distributed with this source code.
*/

namespace Flarum\Http\Middleware;

use Flarum\Http\Exception\ProxyNotAllowedException;
use Illuminate\Support\Arr;
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\MiddlewareInterface as Middleware;
use Psr\Http\Server\RequestHandlerInterface;

class ProxyAddress implements Middleware
{
/**
* @var bool
*/
protected $enabled;

/**
* @var array
*/
protected $allowedAddresses;

/**
* @param bool $enabled
* @param array $allowedAddresses
*/
public function __construct($enabled, $allowedAddresses)
{
$this->enabled = $enabled;
$this->allowedAddresses = $allowedAddresses;
}

private function wildcardMatch(string $ipAddress): bool
{
foreach ($this->allowedAddresses as $allowedAddress) {
if (fnmatch($allowedAddress, $ipAddress)) {
return true;
}
}

return false;
}

public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
{
$serverParams = $request->getServerParams();
$ipAddress = Arr::get($serverParams, 'REMOTE_ADDR', '127.0.0.1');

if ($this->enabled) {
if ($this->wildcardMatch($ipAddress)) {
// standard header for proxies, see: https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/X-Forwarded-For
$ipAddress = Arr::get($serverParams, 'X_FORWARDED_FOR', $ipAddress);
$ipAddress = Arr::get($serverParams, 'HTTP_CLIENT_IP', $ipAddress);
$ipAddress = Arr::get($serverParams, 'X_PROXYUSER_IP', $ipAddress);
} else {
throw new ProxyNotAllowedException();
}
}

$request = $request->withAttribute('ipAddress', $ipAddress);

return $handler->handle($request);
}
}

0 comments on commit 451a557

Please sign in to comment.