2021-01-04 20:15:42 +01:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioTest\Shlink\Rest\Middleware\ShortUrl;
|
|
|
|
|
|
|
|
use Laminas\Diactoros\Response;
|
|
|
|
use Laminas\Diactoros\ServerRequestFactory;
|
|
|
|
use PHPUnit\Framework\Assert;
|
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
use Prophecy\Argument;
|
|
|
|
use Prophecy\PhpUnit\ProphecyTrait;
|
|
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
|
|
|
use Psr\Http\Message\ServerRequestInterface;
|
|
|
|
use Psr\Http\Server\RequestHandlerInterface;
|
|
|
|
use Shlinkio\Shlink\Core\Domain\DomainServiceInterface;
|
2022-09-23 19:03:32 +02:00
|
|
|
use Shlinkio\Shlink\Core\Domain\Entity\Domain;
|
2022-09-23 18:30:07 +02:00
|
|
|
use Shlinkio\Shlink\Core\ShortUrl\Model\Validation\ShortUrlInputFilter;
|
2021-01-04 20:15:42 +01:00
|
|
|
use Shlinkio\Shlink\Rest\ApiKey\Role;
|
|
|
|
use Shlinkio\Shlink\Rest\Entity\ApiKey;
|
|
|
|
use Shlinkio\Shlink\Rest\Middleware\ShortUrl\OverrideDomainMiddleware;
|
|
|
|
|
|
|
|
class OverrideDomainMiddlewareTest extends TestCase
|
|
|
|
{
|
|
|
|
use ProphecyTrait;
|
|
|
|
|
|
|
|
private OverrideDomainMiddleware $middleware;
|
|
|
|
private ObjectProphecy $domainService;
|
|
|
|
private ObjectProphecy $apiKey;
|
|
|
|
private ObjectProphecy $handler;
|
|
|
|
|
|
|
|
protected function setUp(): void
|
|
|
|
{
|
|
|
|
$this->apiKey = $this->prophesize(ApiKey::class);
|
|
|
|
$this->handler = $this->prophesize(RequestHandlerInterface::class);
|
|
|
|
|
|
|
|
$this->domainService = $this->prophesize(DomainServiceInterface::class);
|
|
|
|
$this->middleware = new OverrideDomainMiddleware($this->domainService->reveal());
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @test */
|
|
|
|
public function nextMiddlewareIsCalledWhenApiKeyDoesNotHaveProperRole(): void
|
|
|
|
{
|
|
|
|
$request = $this->requestWithApiKey();
|
|
|
|
$response = new Response();
|
|
|
|
$hasRole = $this->apiKey->hasRole(Role::DOMAIN_SPECIFIC)->willReturn(false);
|
|
|
|
$handle = $this->handler->handle($request)->willReturn($response);
|
|
|
|
$getDomain = $this->domainService->getDomain(Argument::cetera());
|
|
|
|
|
|
|
|
$result = $this->middleware->process($request, $this->handler->reveal());
|
|
|
|
|
|
|
|
self::assertSame($response, $result);
|
|
|
|
$hasRole->shouldHaveBeenCalledOnce();
|
|
|
|
$handle->shouldHaveBeenCalledOnce();
|
|
|
|
$getDomain->shouldNotHaveBeenCalled();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideBodies
|
|
|
|
*/
|
|
|
|
public function overwritesRequestBodyWhenMethodIsPost(Domain $domain, array $body, array $expectedBody): void
|
|
|
|
{
|
|
|
|
$request = $this->requestWithApiKey()->withMethod('POST')->withParsedBody($body);
|
|
|
|
$hasRole = $this->apiKey->hasRole(Role::DOMAIN_SPECIFIC)->willReturn(true);
|
|
|
|
$getRoleMeta = $this->apiKey->getRoleMeta(Role::DOMAIN_SPECIFIC)->willReturn(['domain_id' => '123']);
|
|
|
|
$getDomain = $this->domainService->getDomain('123')->willReturn($domain);
|
|
|
|
$handle = $this->handler->handle(Argument::that(
|
|
|
|
function (ServerRequestInterface $req) use ($expectedBody): bool {
|
|
|
|
Assert::assertEquals($req->getParsedBody(), $expectedBody);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
))->willReturn(new Response());
|
|
|
|
|
|
|
|
$this->middleware->process($request, $this->handler->reveal());
|
|
|
|
|
|
|
|
$hasRole->shouldHaveBeenCalledOnce();
|
|
|
|
$getRoleMeta->shouldHaveBeenCalledOnce();
|
|
|
|
$getDomain->shouldHaveBeenCalledOnce();
|
|
|
|
$handle->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideBodies(): iterable
|
|
|
|
{
|
2021-07-22 20:48:58 +02:00
|
|
|
yield 'no domain provided' => [
|
|
|
|
Domain::withAuthority('foo.com'),
|
|
|
|
[],
|
|
|
|
[ShortUrlInputFilter::DOMAIN => 'foo.com'],
|
|
|
|
];
|
2021-01-04 20:15:42 +01:00
|
|
|
yield 'other domain provided' => [
|
2021-07-22 20:48:58 +02:00
|
|
|
Domain::withAuthority('bar.com'),
|
2021-01-31 07:44:46 +01:00
|
|
|
[ShortUrlInputFilter::DOMAIN => 'foo.com'],
|
|
|
|
[ShortUrlInputFilter::DOMAIN => 'bar.com'],
|
2021-01-04 20:15:42 +01:00
|
|
|
];
|
|
|
|
yield 'same domain provided' => [
|
2021-07-22 20:48:58 +02:00
|
|
|
Domain::withAuthority('baz.com'),
|
2021-01-31 07:44:46 +01:00
|
|
|
[ShortUrlInputFilter::DOMAIN => 'baz.com'],
|
|
|
|
[ShortUrlInputFilter::DOMAIN => 'baz.com'],
|
2021-01-04 20:15:42 +01:00
|
|
|
];
|
|
|
|
yield 'more body params' => [
|
2021-07-22 20:48:58 +02:00
|
|
|
Domain::withAuthority('doma.in'),
|
2021-01-31 07:44:46 +01:00
|
|
|
[ShortUrlInputFilter::DOMAIN => 'baz.com', 'something' => 'else', 'foo' => 123],
|
|
|
|
[ShortUrlInputFilter::DOMAIN => 'doma.in', 'something' => 'else', 'foo' => 123],
|
2021-01-04 20:15:42 +01:00
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideMethods
|
|
|
|
*/
|
|
|
|
public function setsRequestAttributeWhenMethodIsNotPost(string $method): void
|
|
|
|
{
|
2021-07-22 20:48:58 +02:00
|
|
|
$domain = Domain::withAuthority('something.com');
|
2021-01-04 20:15:42 +01:00
|
|
|
$request = $this->requestWithApiKey()->withMethod($method);
|
|
|
|
$hasRole = $this->apiKey->hasRole(Role::DOMAIN_SPECIFIC)->willReturn(true);
|
|
|
|
$getRoleMeta = $this->apiKey->getRoleMeta(Role::DOMAIN_SPECIFIC)->willReturn(['domain_id' => '123']);
|
|
|
|
$getDomain = $this->domainService->getDomain('123')->willReturn($domain);
|
|
|
|
$handle = $this->handler->handle(Argument::that(
|
|
|
|
function (ServerRequestInterface $req): bool {
|
2021-01-31 07:44:46 +01:00
|
|
|
Assert::assertEquals($req->getAttribute(ShortUrlInputFilter::DOMAIN), 'something.com');
|
2021-01-04 20:15:42 +01:00
|
|
|
return true;
|
|
|
|
},
|
|
|
|
))->willReturn(new Response());
|
|
|
|
|
|
|
|
$this->middleware->process($request, $this->handler->reveal());
|
|
|
|
|
|
|
|
$hasRole->shouldHaveBeenCalledOnce();
|
|
|
|
$getRoleMeta->shouldHaveBeenCalledOnce();
|
|
|
|
$getDomain->shouldHaveBeenCalledOnce();
|
|
|
|
$handle->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideMethods(): iterable
|
|
|
|
{
|
|
|
|
yield 'GET' => ['GET'];
|
|
|
|
yield 'PUT' => ['PUT'];
|
|
|
|
yield 'PATCH' => ['PATCH'];
|
|
|
|
yield 'DELETE' => ['DELETE'];
|
|
|
|
}
|
|
|
|
|
|
|
|
private function requestWithApiKey(): ServerRequestInterface
|
|
|
|
{
|
|
|
|
return ServerRequestFactory::fromGlobals()->withAttribute(ApiKey::class, $this->apiKey->reveal());
|
|
|
|
}
|
|
|
|
}
|