2018-11-11 23:08:23 +03:00
|
|
|
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioTest\Shlink\Common\IpGeolocation\GeoLite2;
|
|
|
|
|
|
|
|
use GuzzleHttp\ClientInterface;
|
|
|
|
use GuzzleHttp\Exception\ClientException;
|
|
|
|
use PHPUnit\Framework\TestCase;
|
|
|
|
use Prophecy\Argument;
|
|
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
|
|
|
use Shlinkio\Shlink\Common\Exception\RuntimeException;
|
|
|
|
use Shlinkio\Shlink\Common\IpGeolocation\GeoLite2\DbUpdater;
|
|
|
|
use Shlinkio\Shlink\Common\IpGeolocation\GeoLite2\GeoLite2Options;
|
|
|
|
use Symfony\Component\Filesystem\Exception as FilesystemException;
|
|
|
|
use Symfony\Component\Filesystem\Filesystem;
|
|
|
|
use Zend\Diactoros\Response;
|
|
|
|
|
|
|
|
class DbUpdaterTest extends TestCase
|
|
|
|
{
|
2018-11-20 21:30:27 +03:00
|
|
|
/** @var DbUpdater */
|
2018-11-11 23:08:23 +03:00
|
|
|
private $dbUpdater;
|
2018-11-20 21:30:27 +03:00
|
|
|
/** @var ObjectProphecy */
|
2018-11-11 23:08:23 +03:00
|
|
|
private $httpClient;
|
2018-11-20 21:30:27 +03:00
|
|
|
/** @var ObjectProphecy */
|
2018-11-11 23:08:23 +03:00
|
|
|
private $filesystem;
|
2018-11-20 21:30:27 +03:00
|
|
|
/** @var GeoLite2Options */
|
2018-11-11 23:08:23 +03:00
|
|
|
private $options;
|
|
|
|
|
2019-02-16 12:53:45 +03:00
|
|
|
public function setUp(): void
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
|
|
|
$this->httpClient = $this->prophesize(ClientInterface::class);
|
|
|
|
$this->filesystem = $this->prophesize(Filesystem::class);
|
|
|
|
$this->options = new GeoLite2Options([
|
|
|
|
'temp_dir' => __DIR__ . '/../../../test-resources',
|
2019-07-23 23:04:01 +03:00
|
|
|
'db_location' => 'db_location',
|
2018-11-11 23:08:23 +03:00
|
|
|
'download_from' => '',
|
|
|
|
]);
|
|
|
|
|
|
|
|
$this->dbUpdater = new DbUpdater($this->httpClient->reveal(), $this->filesystem->reveal(), $this->options);
|
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
|
|
|
public function anExceptionIsThrownIfFreshDbCannotBeDownloaded(): void
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
|
|
|
$request = $this->httpClient->request(Argument::cetera())->willThrow(ClientException::class);
|
|
|
|
|
|
|
|
$this->expectException(RuntimeException::class);
|
2018-11-17 20:40:47 +03:00
|
|
|
$this->expectExceptionCode(0);
|
2018-11-11 23:08:23 +03:00
|
|
|
$this->expectExceptionMessage(
|
|
|
|
'An error occurred while trying to download a fresh copy of the GeoLite2 database'
|
|
|
|
);
|
|
|
|
$request->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$this->dbUpdater->downloadFreshCopy();
|
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
|
|
|
public function anExceptionIsThrownIfFreshDbCannotBeExtracted(): void
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
|
|
|
$this->options->tempDir = '__invalid__';
|
|
|
|
|
|
|
|
$request = $this->httpClient->request(Argument::cetera())->willReturn(new Response());
|
|
|
|
|
|
|
|
$this->expectException(RuntimeException::class);
|
2018-11-17 20:40:47 +03:00
|
|
|
$this->expectExceptionCode(0);
|
2018-11-11 23:08:23 +03:00
|
|
|
$this->expectExceptionMessage(
|
|
|
|
'An error occurred while trying to extract the GeoLite2 database from __invalid__/GeoLite2-City.tar.gz'
|
|
|
|
);
|
|
|
|
$request->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$this->dbUpdater->downloadFreshCopy();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideFilesystemExceptions
|
|
|
|
*/
|
2019-02-17 22:28:34 +03:00
|
|
|
public function anExceptionIsThrownIfFreshDbCannotBeCopiedToDestination(string $e): void
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
|
|
|
$request = $this->httpClient->request(Argument::cetera())->willReturn(new Response());
|
|
|
|
$copy = $this->filesystem->copy(Argument::cetera())->willThrow($e);
|
|
|
|
|
|
|
|
$this->expectException(RuntimeException::class);
|
2018-11-17 20:40:47 +03:00
|
|
|
$this->expectExceptionCode(0);
|
2018-11-11 23:08:23 +03:00
|
|
|
$this->expectExceptionMessage('An error occurred while trying to copy GeoLite2 db file to destination');
|
|
|
|
$request->shouldBeCalledOnce();
|
|
|
|
$copy->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$this->dbUpdater->downloadFreshCopy();
|
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
public function provideFilesystemExceptions(): iterable
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
2019-02-17 22:28:34 +03:00
|
|
|
yield 'file not found' => [FilesystemException\FileNotFoundException::class];
|
|
|
|
yield 'IO error' => [FilesystemException\IOException::class];
|
2018-11-11 23:08:23 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
|
|
|
public function noExceptionsAreThrownIfEverythingWorksFine(): void
|
2018-11-11 23:08:23 +03:00
|
|
|
{
|
|
|
|
$request = $this->httpClient->request(Argument::cetera())->willReturn(new Response());
|
|
|
|
$copy = $this->filesystem->copy(Argument::cetera())->will(function () {
|
|
|
|
});
|
|
|
|
$remove = $this->filesystem->remove(Argument::cetera())->will(function () {
|
|
|
|
});
|
|
|
|
|
|
|
|
$this->dbUpdater->downloadFreshCopy();
|
|
|
|
|
|
|
|
$request->shouldHaveBeenCalledOnce();
|
|
|
|
$copy->shouldHaveBeenCalledOnce();
|
|
|
|
$remove->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
2019-07-23 23:04:01 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideExists
|
|
|
|
*/
|
|
|
|
public function databaseFileExistsChecksIfTheFilesExistsInTheFilesystem(bool $expected): void
|
|
|
|
{
|
|
|
|
$exists = $this->filesystem->exists('db_location')->willReturn($expected);
|
|
|
|
|
|
|
|
$result = $this->dbUpdater->databaseFileExists();
|
|
|
|
|
|
|
|
$this->assertEquals($expected, $result);
|
|
|
|
$exists->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideExists(): iterable
|
|
|
|
{
|
|
|
|
return [[true], [false]];
|
|
|
|
}
|
2018-11-11 23:08:23 +03:00
|
|
|
}
|