0681511699
Related to #95
222 lines
7.5 KiB
PHP
222 lines
7.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Shaarli\Front;
|
|
|
|
use Shaarli\Config\ConfigManager;
|
|
use Shaarli\Container\ShaarliContainer;
|
|
use Shaarli\Front\Exception\LoginBannedException;
|
|
use Shaarli\Front\Exception\UnauthorizedException;
|
|
use Shaarli\Render\PageBuilder;
|
|
use Shaarli\Render\PageCacheManager;
|
|
use Shaarli\Security\LoginManager;
|
|
use Shaarli\TestCase;
|
|
use Shaarli\Updater\Updater;
|
|
use Slim\Http\Request;
|
|
use Slim\Http\Response;
|
|
use Slim\Http\Uri;
|
|
|
|
class ShaarliMiddlewareTest extends TestCase
|
|
{
|
|
protected const TMP_MOCK_FILE = '.tmp';
|
|
|
|
/** @var ShaarliContainer */
|
|
protected $container;
|
|
|
|
/** @var ShaarliMiddleware */
|
|
protected $middleware;
|
|
|
|
public function setUp(): void
|
|
{
|
|
$this->container = $this->createMock(ShaarliContainer::class);
|
|
|
|
touch(static::TMP_MOCK_FILE);
|
|
|
|
$this->container->conf = $this->createMock(ConfigManager::class);
|
|
$this->container->conf->method('getConfigFileExt')->willReturn(static::TMP_MOCK_FILE);
|
|
|
|
$this->container->loginManager = $this->createMock(LoginManager::class);
|
|
|
|
$this->container->environment = ['REQUEST_URI' => 'http://shaarli/subfolder/path'];
|
|
|
|
$this->middleware = new ShaarliMiddleware($this->container);
|
|
}
|
|
|
|
public function tearDown(): void
|
|
{
|
|
unlink(static::TMP_MOCK_FILE);
|
|
}
|
|
|
|
/**
|
|
* Test middleware execution with valid controller call
|
|
*/
|
|
public function testMiddlewareExecution(): void
|
|
{
|
|
$request = $this->createMock(Request::class);
|
|
$request->method('getUri')->willReturnCallback(function (): Uri {
|
|
$uri = $this->createMock(Uri::class);
|
|
$uri->method('getBasePath')->willReturn('/subfolder');
|
|
|
|
return $uri;
|
|
});
|
|
|
|
$response = new Response();
|
|
$controller = function (Request $request, Response $response): Response {
|
|
return $response->withStatus(418); // I'm a tea pot
|
|
};
|
|
|
|
/** @var Response $result */
|
|
$result = $this->middleware->__invoke($request, $response, $controller);
|
|
|
|
static::assertInstanceOf(Response::class, $result);
|
|
static::assertSame(418, $result->getStatusCode());
|
|
}
|
|
|
|
/**
|
|
* Test middleware execution with controller throwing a known front exception.
|
|
* The exception should be thrown to be later handled by the error handler.
|
|
*/
|
|
public function testMiddlewareExecutionWithFrontException(): void
|
|
{
|
|
$request = $this->createMock(Request::class);
|
|
$request->method('getUri')->willReturnCallback(function (): Uri {
|
|
$uri = $this->createMock(Uri::class);
|
|
$uri->method('getBasePath')->willReturn('/subfolder');
|
|
|
|
return $uri;
|
|
});
|
|
|
|
$response = new Response();
|
|
$controller = function (): void {
|
|
$exception = new LoginBannedException();
|
|
|
|
throw new $exception();
|
|
};
|
|
|
|
$pageBuilder = $this->createMock(PageBuilder::class);
|
|
$pageBuilder->method('render')->willReturnCallback(function (string $message): string {
|
|
return $message;
|
|
});
|
|
$this->container->pageBuilder = $pageBuilder;
|
|
|
|
$this->expectException(LoginBannedException::class);
|
|
|
|
$this->middleware->__invoke($request, $response, $controller);
|
|
}
|
|
|
|
/**
|
|
* Test middleware execution with controller throwing a not authorized exception
|
|
* The middle should send a redirection response to the login page.
|
|
*/
|
|
public function testMiddlewareExecutionWithUnauthorizedException(): void
|
|
{
|
|
$request = $this->createMock(Request::class);
|
|
$request->method('getUri')->willReturnCallback(function (): Uri {
|
|
$uri = $this->createMock(Uri::class);
|
|
$uri->method('getBasePath')->willReturn('/subfolder');
|
|
|
|
return $uri;
|
|
});
|
|
|
|
$response = new Response();
|
|
$controller = function (): void {
|
|
throw new UnauthorizedException();
|
|
};
|
|
|
|
/** @var Response $result */
|
|
$result = $this->middleware->__invoke($request, $response, $controller);
|
|
|
|
static::assertSame(302, $result->getStatusCode());
|
|
static::assertSame(
|
|
'/subfolder/login?returnurl=' . urlencode('http://shaarli/subfolder/path'),
|
|
$result->getHeader('location')[0]
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Test middleware execution with controller throwing a not authorized exception.
|
|
* The exception should be thrown to be later handled by the error handler.
|
|
*/
|
|
public function testMiddlewareExecutionWithServerException(): void
|
|
{
|
|
$request = $this->createMock(Request::class);
|
|
$request->method('getUri')->willReturnCallback(function (): Uri {
|
|
$uri = $this->createMock(Uri::class);
|
|
$uri->method('getBasePath')->willReturn('/subfolder');
|
|
|
|
return $uri;
|
|
});
|
|
|
|
$dummyException = new class () extends \Exception {
|
|
};
|
|
|
|
$response = new Response();
|
|
$controller = function () use ($dummyException): void {
|
|
throw $dummyException;
|
|
};
|
|
|
|
$parameters = [];
|
|
$this->container->pageBuilder = $this->createMock(PageBuilder::class);
|
|
$this->container->pageBuilder->method('render')->willReturnCallback(function (string $message): string {
|
|
return $message;
|
|
});
|
|
$this->container->pageBuilder
|
|
->method('assign')
|
|
->willReturnCallback(function (string $key, string $value) use (&$parameters): void {
|
|
$parameters[$key] = $value;
|
|
})
|
|
;
|
|
|
|
$this->expectException(get_class($dummyException));
|
|
|
|
$this->middleware->__invoke($request, $response, $controller);
|
|
}
|
|
|
|
public function testMiddlewareExecutionWithUpdates(): void
|
|
{
|
|
$request = $this->createMock(Request::class);
|
|
$request->method('getUri')->willReturnCallback(function (): Uri {
|
|
$uri = $this->createMock(Uri::class);
|
|
$uri->method('getBasePath')->willReturn('/subfolder');
|
|
|
|
return $uri;
|
|
});
|
|
|
|
$response = new Response();
|
|
$controller = function (Request $request, Response $response): Response {
|
|
return $response->withStatus(418); // I'm a tea pot
|
|
};
|
|
|
|
$this->container->loginManager = $this->createMock(LoginManager::class);
|
|
$this->container->loginManager->method('isLoggedIn')->willReturn(true);
|
|
|
|
$this->container->conf = $this->createMock(ConfigManager::class);
|
|
$this->container->conf->method('get')->willReturnCallback(function (string $key): string {
|
|
return $key;
|
|
});
|
|
$this->container->conf->method('getConfigFileExt')->willReturn(static::TMP_MOCK_FILE);
|
|
|
|
$this->container->pageCacheManager = $this->createMock(PageCacheManager::class);
|
|
$this->container->pageCacheManager->expects(static::once())->method('invalidateCaches');
|
|
|
|
$this->container->updater = $this->createMock(Updater::class);
|
|
$this->container->updater
|
|
->expects(static::once())
|
|
->method('update')
|
|
->willReturn(['update123'])
|
|
;
|
|
$this->container->updater->method('getDoneUpdates')->willReturn($updates = ['update123', 'other']);
|
|
$this->container->updater
|
|
->expects(static::once())
|
|
->method('writeUpdates')
|
|
->with('resource.updates', $updates)
|
|
;
|
|
|
|
/** @var Response $result */
|
|
$result = $this->middleware->__invoke($request, $response, $controller);
|
|
|
|
static::assertInstanceOf(Response::class, $result);
|
|
static::assertSame(418, $result->getStatusCode());
|
|
}
|
|
}
|