2024-11-15 03:17:09 -06:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioTest\Shlink\Core\Geolocation\Middleware;
|
|
|
|
|
|
|
|
use Laminas\Diactoros\ServerRequestFactory;
|
|
|
|
use PHPUnit\Framework\Assert;
|
|
|
|
use PHPUnit\Framework\Attributes\Test;
|
|
|
|
use PHPUnit\Framework\Attributes\TestWith;
|
|
|
|
use PHPUnit\Framework\MockObject\MockObject;
|
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
use Psr\Http\Message\ServerRequestInterface;
|
|
|
|
use Psr\Http\Server\RequestHandlerInterface;
|
|
|
|
use Psr\Log\LoggerInterface;
|
|
|
|
use RuntimeException;
|
2024-11-19 02:08:04 -06:00
|
|
|
use Shlinkio\Shlink\Common\Util\IpAddress;
|
2024-11-15 03:17:09 -06:00
|
|
|
use Shlinkio\Shlink\Core\Config\Options\TrackingOptions;
|
|
|
|
use Shlinkio\Shlink\Core\Geolocation\Middleware\IpGeolocationMiddleware;
|
|
|
|
use Shlinkio\Shlink\IpGeolocation\Exception\WrongIpException;
|
|
|
|
use Shlinkio\Shlink\IpGeolocation\GeoLite2\DbUpdaterInterface;
|
|
|
|
use Shlinkio\Shlink\IpGeolocation\Model\Location;
|
|
|
|
use Shlinkio\Shlink\IpGeolocation\Resolver\IpLocationResolverInterface;
|
|
|
|
use Throwable;
|
|
|
|
|
2024-11-22 01:59:42 -06:00
|
|
|
use const Shlinkio\Shlink\IP_ADDRESS_REQUEST_ATTRIBUTE;
|
|
|
|
|
2024-11-15 03:17:09 -06:00
|
|
|
class IpGeolocationMiddlewareTest extends TestCase
|
|
|
|
{
|
|
|
|
private MockObject & IpLocationResolverInterface $ipLocationResolver;
|
|
|
|
private MockObject & DbUpdaterInterface $dbUpdater;
|
|
|
|
private MockObject & LoggerInterface $logger;
|
|
|
|
private MockObject & RequestHandlerInterface $handler;
|
|
|
|
|
|
|
|
protected function setUp(): void
|
|
|
|
{
|
|
|
|
$this->ipLocationResolver = $this->createMock(IpLocationResolverInterface::class);
|
|
|
|
$this->dbUpdater = $this->createMock(DbUpdaterInterface::class);
|
|
|
|
$this->logger = $this->createMock(LoggerInterface::class);
|
|
|
|
$this->handler = $this->createMock(RequestHandlerInterface::class);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function geolocationIsSkippedIfTrackingIsDisabled(): void
|
|
|
|
{
|
|
|
|
$this->dbUpdater->expects($this->never())->method('databaseFileExists');
|
|
|
|
$this->ipLocationResolver->expects($this->never())->method('resolveIpLocation');
|
|
|
|
$this->logger->expects($this->never())->method('warning');
|
|
|
|
|
|
|
|
$request = ServerRequestFactory::fromGlobals();
|
|
|
|
$this->handler->expects($this->once())->method('handle')->with($request);
|
|
|
|
|
|
|
|
$this->middleware(disableTracking: true)->process($request, $this->handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function warningIsLoggedIfGeoLiteDbDoesNotExist(): void
|
|
|
|
{
|
|
|
|
$this->ipLocationResolver->expects($this->never())->method('resolveIpLocation');
|
|
|
|
$this->dbUpdater->expects($this->once())->method('databaseFileExists')->willReturn(false);
|
|
|
|
$this->logger->expects($this->once())->method('warning')->with(
|
|
|
|
'Tried to geolocate IP address, but a GeoLite2 db was not found.',
|
|
|
|
);
|
|
|
|
|
|
|
|
$request = ServerRequestFactory::fromGlobals();
|
|
|
|
$this->handler->expects($this->once())->method('handle')->with($request);
|
|
|
|
|
|
|
|
$this->middleware()->process($request, $this->handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
2024-11-19 02:08:04 -06:00
|
|
|
#[TestWith([null])]
|
|
|
|
#[TestWith([IpAddress::LOCALHOST])]
|
|
|
|
public function emptyLocationIsReturnedIfIpAddressIsNotLocatable(string|null $ipAddress): void
|
2024-11-15 03:17:09 -06:00
|
|
|
{
|
|
|
|
$this->dbUpdater->expects($this->once())->method('databaseFileExists')->willReturn(true);
|
|
|
|
$this->ipLocationResolver->expects($this->never())->method('resolveIpLocation');
|
|
|
|
$this->logger->expects($this->never())->method('warning');
|
|
|
|
|
2024-11-22 01:59:42 -06:00
|
|
|
$request = ServerRequestFactory::fromGlobals()->withAttribute(IP_ADDRESS_REQUEST_ATTRIBUTE, $ipAddress);
|
2024-11-15 03:17:09 -06:00
|
|
|
$this->handler->expects($this->once())->method('handle')->with($this->callback(
|
|
|
|
function (ServerRequestInterface $req): bool {
|
|
|
|
$location = $req->getAttribute(Location::class);
|
|
|
|
if (! $location instanceof Location) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Assert::assertEmpty($location->countryCode);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
));
|
|
|
|
|
|
|
|
$this->middleware()->process($request, $this->handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
public function locationIsResolvedFromIpAddress(): void
|
|
|
|
{
|
|
|
|
$this->dbUpdater->expects($this->once())->method('databaseFileExists')->willReturn(true);
|
|
|
|
$this->ipLocationResolver->expects($this->once())->method('resolveIpLocation')->with('1.2.3.4')->willReturn(
|
|
|
|
new Location(countryCode: 'ES'),
|
|
|
|
);
|
|
|
|
$this->logger->expects($this->never())->method('warning');
|
|
|
|
|
2024-11-22 01:59:42 -06:00
|
|
|
$request = ServerRequestFactory::fromGlobals()->withAttribute(IP_ADDRESS_REQUEST_ATTRIBUTE, '1.2.3.4');
|
2024-11-15 03:17:09 -06:00
|
|
|
$this->handler->expects($this->once())->method('handle')->with($this->callback(
|
|
|
|
function (ServerRequestInterface $req): bool {
|
|
|
|
$location = $req->getAttribute(Location::class);
|
|
|
|
if (! $location instanceof Location) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Assert::assertEquals('ES', $location->countryCode);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
));
|
|
|
|
|
|
|
|
$this->middleware()->process($request, $this->handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[Test]
|
|
|
|
#[TestWith([
|
|
|
|
new WrongIpException(),
|
|
|
|
'warning',
|
|
|
|
'Tried to locate IP address, but it seems to be wrong. {e}',
|
|
|
|
])]
|
|
|
|
#[TestWith([
|
|
|
|
new RuntimeException('Unknown'),
|
|
|
|
'error',
|
|
|
|
'An unexpected error occurred while trying to locate IP address. {e}',
|
|
|
|
])]
|
|
|
|
public function warningIsPrintedIfAnErrorOccurs(
|
|
|
|
Throwable $exception,
|
|
|
|
string $loggerMethod,
|
|
|
|
string $expectedLoggedMessage,
|
|
|
|
): void {
|
|
|
|
$this->dbUpdater->expects($this->once())->method('databaseFileExists')->willReturn(true);
|
|
|
|
$this->ipLocationResolver
|
|
|
|
->expects($this->once())
|
|
|
|
->method('resolveIpLocation')
|
|
|
|
->with('1.2.3.4')
|
|
|
|
->willThrowException($exception);
|
|
|
|
$this->logger->expects($this->once())->method($loggerMethod)->with($expectedLoggedMessage, ['e' => $exception]);
|
|
|
|
|
2024-11-22 01:59:42 -06:00
|
|
|
$request = ServerRequestFactory::fromGlobals()->withAttribute(IP_ADDRESS_REQUEST_ATTRIBUTE, '1.2.3.4');
|
2024-11-15 03:17:09 -06:00
|
|
|
$this->handler->expects($this->once())->method('handle')->with($this->callback(
|
|
|
|
function (ServerRequestInterface $req): bool {
|
|
|
|
$location = $req->getAttribute(Location::class);
|
|
|
|
if (! $location instanceof Location) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Assert::assertEmpty($location->countryCode);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
));
|
|
|
|
|
|
|
|
$this->middleware()->process($request, $this->handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function middleware(bool $disableTracking = false): IpGeolocationMiddleware
|
|
|
|
{
|
|
|
|
return new IpGeolocationMiddleware(
|
|
|
|
$this->ipLocationResolver,
|
|
|
|
$this->dbUpdater,
|
|
|
|
$this->logger,
|
|
|
|
new TrackingOptions(disableTracking: $disableTracking),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|