2020-05-01 12:57:46 +03:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioTest\Shlink\Core\Visit;
|
|
|
|
|
|
|
|
use Doctrine\ORM\EntityManagerInterface;
|
2021-02-08 21:46:51 +03:00
|
|
|
use Laminas\Stdlib\ArrayUtils;
|
2023-02-09 11:32:38 +03:00
|
|
|
use PHPUnit\Framework\Assert;
|
2023-02-09 22:42:18 +03:00
|
|
|
use PHPUnit\Framework\Attributes\DataProvider;
|
2023-06-18 11:41:24 +03:00
|
|
|
use PHPUnit\Framework\Attributes\DataProviderExternal;
|
2023-02-09 22:42:18 +03:00
|
|
|
use PHPUnit\Framework\Attributes\Test;
|
2022-10-23 22:05:13 +03:00
|
|
|
use PHPUnit\Framework\MockObject\MockObject;
|
2020-05-01 12:57:46 +03:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2022-09-23 20:03:32 +03:00
|
|
|
use Shlinkio\Shlink\Core\Domain\Entity\Domain;
|
2022-04-23 12:02:51 +03:00
|
|
|
use Shlinkio\Shlink\Core\Domain\Repository\DomainRepository;
|
|
|
|
use Shlinkio\Shlink\Core\Exception\DomainNotFoundException;
|
2021-02-08 21:46:51 +03:00
|
|
|
use Shlinkio\Shlink\Core\Exception\ShortUrlNotFoundException;
|
|
|
|
use Shlinkio\Shlink\Core\Exception\TagNotFoundException;
|
2022-09-23 20:03:32 +03:00
|
|
|
use Shlinkio\Shlink\Core\ShortUrl\Entity\ShortUrl;
|
2022-09-23 19:05:17 +03:00
|
|
|
use Shlinkio\Shlink\Core\ShortUrl\Model\ShortUrlIdentifier;
|
2024-02-17 12:21:36 +03:00
|
|
|
use Shlinkio\Shlink\Core\ShortUrl\Repository\ShortUrlRepository;
|
2022-09-23 20:03:32 +03:00
|
|
|
use Shlinkio\Shlink\Core\Tag\Entity\Tag;
|
2022-09-23 19:24:14 +03:00
|
|
|
use Shlinkio\Shlink\Core\Tag\Repository\TagRepository;
|
2024-04-01 11:22:51 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Entity\OrphanVisitsCount;
|
2024-03-31 11:33:31 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Entity\ShortUrlVisitsCount;
|
2022-09-23 20:03:32 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Entity\Visit;
|
2024-02-10 19:51:42 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Model\OrphanVisitsParams;
|
2022-09-23 19:05:17 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Model\Visitor;
|
|
|
|
use Shlinkio\Shlink\Core\Visit\Model\VisitsParams;
|
2020-05-01 12:57:46 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Model\VisitsStats;
|
2024-02-10 15:57:16 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\OrphanVisitsCountFiltering;
|
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\OrphanVisitsListFiltering;
|
2021-05-22 21:16:32 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\VisitsCountFiltering;
|
2021-05-22 21:32:30 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\VisitsListFiltering;
|
2024-04-01 11:22:51 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Repository\OrphanVisitsCountRepository;
|
2024-03-31 11:33:31 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Repository\ShortUrlVisitsCountRepository;
|
2022-09-23 19:24:14 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Repository\VisitRepository;
|
2020-05-01 12:57:46 +03:00
|
|
|
use Shlinkio\Shlink\Core\Visit\VisitsStatsHelper;
|
2021-02-08 21:46:51 +03:00
|
|
|
use Shlinkio\Shlink\Rest\Entity\ApiKey;
|
2023-06-18 11:41:24 +03:00
|
|
|
use ShlinkioTest\Shlink\Core\Util\ApiKeyDataProviders;
|
2020-05-01 12:57:46 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
use function array_map;
|
2021-02-10 00:40:40 +03:00
|
|
|
use function count;
|
2020-05-01 12:57:46 +03:00
|
|
|
use function range;
|
|
|
|
|
|
|
|
class VisitsStatsHelperTest extends TestCase
|
|
|
|
{
|
|
|
|
private VisitsStatsHelper $helper;
|
2022-10-24 20:53:13 +03:00
|
|
|
private MockObject & EntityManagerInterface $em;
|
2020-05-01 12:57:46 +03:00
|
|
|
|
2022-09-11 13:02:49 +03:00
|
|
|
protected function setUp(): void
|
2020-05-01 12:57:46 +03:00
|
|
|
{
|
2022-10-23 22:05:13 +03:00
|
|
|
$this->em = $this->createMock(EntityManagerInterface::class);
|
|
|
|
$this->helper = new VisitsStatsHelper($this->em);
|
2020-05-01 12:57:46 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test, DataProvider('provideCounts')]
|
2023-05-31 10:11:20 +03:00
|
|
|
public function returnsExpectedVisitsStats(int $expectedCount, ?ApiKey $apiKey): void
|
2020-05-01 12:57:46 +03:00
|
|
|
{
|
2023-02-09 11:32:38 +03:00
|
|
|
$callCount = 0;
|
2024-03-31 11:33:31 +03:00
|
|
|
$visitsCountRepo = $this->createMock(ShortUrlVisitsCountRepository::class);
|
|
|
|
$visitsCountRepo->expects($this->exactly(2))->method('countNonOrphanVisits')->willReturnCallback(
|
2023-05-31 10:11:20 +03:00
|
|
|
function (VisitsCountFiltering $options) use ($expectedCount, $apiKey, &$callCount) {
|
2023-02-09 11:32:38 +03:00
|
|
|
Assert::assertEquals($callCount !== 0, $options->excludeBots);
|
2023-05-31 10:11:20 +03:00
|
|
|
Assert::assertEquals($apiKey, $options->apiKey);
|
2023-02-09 11:32:38 +03:00
|
|
|
$callCount++;
|
|
|
|
|
|
|
|
return $expectedCount * 3;
|
|
|
|
},
|
|
|
|
);
|
2024-03-31 11:33:31 +03:00
|
|
|
|
2024-04-01 11:22:51 +03:00
|
|
|
$orphanVisitsCountRepo = $this->createMock(OrphanVisitsCountRepository::class);
|
|
|
|
$orphanVisitsCountRepo->expects($this->exactly(2))->method('countOrphanVisits')->with(
|
2023-02-09 11:32:38 +03:00
|
|
|
$this->isInstanceOf(VisitsCountFiltering::class),
|
2022-10-23 22:05:13 +03:00
|
|
|
)->willReturn($expectedCount);
|
2024-03-31 11:33:31 +03:00
|
|
|
|
|
|
|
$this->em->expects($this->exactly(2))->method('getRepository')->willReturnMap([
|
2024-04-01 11:22:51 +03:00
|
|
|
[OrphanVisitsCount::class, $orphanVisitsCountRepo],
|
2024-03-31 11:33:31 +03:00
|
|
|
[ShortUrlVisitsCount::class, $visitsCountRepo],
|
|
|
|
]);
|
2020-05-01 12:57:46 +03:00
|
|
|
|
2023-05-31 10:11:20 +03:00
|
|
|
$stats = $this->helper->getVisitsStats($apiKey);
|
2020-05-01 12:57:46 +03:00
|
|
|
|
2021-02-09 00:44:58 +03:00
|
|
|
self::assertEquals(new VisitsStats($expectedCount * 3, $expectedCount), $stats);
|
2020-05-01 12:57:46 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 11:32:38 +03:00
|
|
|
public static function provideCounts(): iterable
|
2020-05-01 12:57:46 +03:00
|
|
|
{
|
2023-05-31 10:11:20 +03:00
|
|
|
return [
|
2023-11-29 14:34:13 +03:00
|
|
|
...array_map(fn (int $value) => [$value, null], range(0, 50, 5)),
|
|
|
|
...array_map(fn (int $value) => [$value, ApiKey::create()], range(0, 18, 3)),
|
2023-05-31 10:11:20 +03:00
|
|
|
];
|
2020-05-01 12:57:46 +03:00
|
|
|
}
|
2021-02-08 21:46:51 +03:00
|
|
|
|
2023-06-18 11:41:24 +03:00
|
|
|
#[Test, DataProviderExternal(ApiKeyDataProviders::class, 'adminApiKeysProvider')]
|
2021-02-08 21:46:51 +03:00
|
|
|
public function infoReturnsVisitsForCertainShortCode(?ApiKey $apiKey): void
|
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
2021-05-23 09:41:42 +03:00
|
|
|
$identifier = ShortUrlIdentifier::fromShortCodeAndDomain($shortCode);
|
2021-05-23 13:31:10 +03:00
|
|
|
$spec = $apiKey?->spec();
|
2021-05-23 09:41:42 +03:00
|
|
|
|
2024-02-17 12:21:36 +03:00
|
|
|
$repo = $this->createMock(ShortUrlRepository::class);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo->expects($this->once())->method('shortCodeIsInUse')->with($identifier, $spec)->willReturn(true);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(
|
|
|
|
static fn () => Visit::forValidShortUrl(ShortUrl::createFake(), Visitor::emptyInstance()),
|
|
|
|
range(0, 1),
|
|
|
|
);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo2 = $this->createMock(VisitRepository::class);
|
|
|
|
$repo2->method('findVisitsByShortCode')->with(
|
|
|
|
$identifier,
|
|
|
|
$this->isInstanceOf(VisitsListFiltering::class),
|
|
|
|
)->willReturn($list);
|
|
|
|
$repo2->method('countVisitsByShortCode')->with(
|
|
|
|
$identifier,
|
2022-10-24 00:07:50 +03:00
|
|
|
$this->isInstanceOf(VisitsCountFiltering::class),
|
2022-10-23 22:05:13 +03:00
|
|
|
)->willReturn(1);
|
|
|
|
|
|
|
|
$this->em->expects($this->exactly(2))->method('getRepository')->willReturnMap([
|
|
|
|
[ShortUrl::class, $repo],
|
|
|
|
[Visit::class, $repo2],
|
|
|
|
]);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
2021-05-23 09:41:42 +03:00
|
|
|
$paginator = $this->helper->visitsForShortUrl($identifier, new VisitsParams(), $apiKey);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2021-02-08 21:46:51 +03:00
|
|
|
public function throwsExceptionWhenRequestingVisitsForInvalidShortCode(): void
|
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
2021-05-23 09:41:42 +03:00
|
|
|
$identifier = ShortUrlIdentifier::fromShortCodeAndDomain($shortCode);
|
|
|
|
|
2024-02-17 12:21:36 +03:00
|
|
|
$repo = $this->createMock(ShortUrlRepository::class);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo->expects($this->once())->method('shortCodeIsInUse')->with($identifier, null)->willReturn(false);
|
|
|
|
$this->em->expects($this->once())->method('getRepository')->with(ShortUrl::class)->willReturn($repo);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
|
|
|
$this->expectException(ShortUrlNotFoundException::class);
|
|
|
|
|
2021-05-23 09:41:42 +03:00
|
|
|
$this->helper->visitsForShortUrl($identifier, new VisitsParams());
|
2021-02-08 21:46:51 +03:00
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2021-02-08 21:46:51 +03:00
|
|
|
public function throwsExceptionWhenRequestingVisitsForInvalidTag(): void
|
|
|
|
{
|
|
|
|
$tag = 'foo';
|
2021-03-14 11:59:35 +03:00
|
|
|
$apiKey = ApiKey::create();
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(TagRepository::class);
|
|
|
|
$repo->expects($this->once())->method('tagExists')->with($tag, $apiKey)->willReturn(false);
|
|
|
|
$this->em->expects($this->once())->method('getRepository')->with(Tag::class)->willReturn($repo);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
|
|
|
$this->expectException(TagNotFoundException::class);
|
|
|
|
|
|
|
|
$this->helper->visitsForTag($tag, new VisitsParams(), $apiKey);
|
|
|
|
}
|
|
|
|
|
2023-06-18 11:41:24 +03:00
|
|
|
#[Test, DataProviderExternal(ApiKeyDataProviders::class, 'adminApiKeysProvider')]
|
2021-02-08 21:46:51 +03:00
|
|
|
public function visitsForTagAreReturnedAsExpected(?ApiKey $apiKey): void
|
|
|
|
{
|
|
|
|
$tag = 'foo';
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(TagRepository::class);
|
|
|
|
$repo->expects($this->once())->method('tagExists')->with($tag, $apiKey)->willReturn(true);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(
|
|
|
|
static fn () => Visit::forValidShortUrl(ShortUrl::createFake(), Visitor::emptyInstance()),
|
|
|
|
range(0, 1),
|
|
|
|
);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo2 = $this->createMock(VisitRepository::class);
|
|
|
|
$repo2->method('findVisitsByTag')->with($tag, $this->isInstanceOf(VisitsListFiltering::class))->willReturn(
|
|
|
|
$list,
|
|
|
|
);
|
|
|
|
$repo2->method('countVisitsByTag')->with($tag, $this->isInstanceOf(VisitsCountFiltering::class))->willReturn(1);
|
|
|
|
|
|
|
|
$this->em->expects($this->exactly(2))->method('getRepository')->willReturnMap([
|
|
|
|
[Tag::class, $repo],
|
|
|
|
[Visit::class, $repo2],
|
|
|
|
]);
|
2021-02-08 21:46:51 +03:00
|
|
|
|
|
|
|
$paginator = $this->helper->visitsForTag($tag, new VisitsParams(), $apiKey);
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
2021-02-10 00:40:40 +03:00
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2022-04-23 12:02:51 +03:00
|
|
|
public function throwsExceptionWhenRequestingVisitsForInvalidDomain(): void
|
|
|
|
{
|
|
|
|
$domain = 'foo.com';
|
|
|
|
$apiKey = ApiKey::create();
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(DomainRepository::class);
|
|
|
|
$repo->expects($this->once())->method('domainExists')->with($domain, $apiKey)->willReturn(false);
|
|
|
|
$this->em->expects($this->once())->method('getRepository')->with(Domain::class)->willReturn($repo);
|
2022-04-23 12:02:51 +03:00
|
|
|
|
|
|
|
$this->expectException(DomainNotFoundException::class);
|
|
|
|
|
|
|
|
$this->helper->visitsForDomain($domain, new VisitsParams(), $apiKey);
|
|
|
|
}
|
|
|
|
|
2023-06-18 11:41:24 +03:00
|
|
|
#[Test, DataProviderExternal(ApiKeyDataProviders::class, 'adminApiKeysProvider')]
|
2022-04-23 12:02:51 +03:00
|
|
|
public function visitsForNonDefaultDomainAreReturnedAsExpected(?ApiKey $apiKey): void
|
|
|
|
{
|
|
|
|
$domain = 'foo.com';
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(DomainRepository::class);
|
|
|
|
$repo->expects($this->once())->method('domainExists')->with($domain, $apiKey)->willReturn(true);
|
2022-04-23 12:02:51 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(
|
|
|
|
static fn () => Visit::forValidShortUrl(ShortUrl::createFake(), Visitor::emptyInstance()),
|
|
|
|
range(0, 1),
|
|
|
|
);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo2 = $this->createMock(VisitRepository::class);
|
|
|
|
$repo2->method('findVisitsByDomain')->with(
|
|
|
|
$domain,
|
|
|
|
$this->isInstanceOf(VisitsListFiltering::class),
|
|
|
|
)->willReturn($list);
|
|
|
|
$repo2->method('countVisitsByDomain')->with(
|
|
|
|
$domain,
|
|
|
|
$this->isInstanceOf(VisitsCountFiltering::class),
|
|
|
|
)->willReturn(1);
|
|
|
|
|
|
|
|
$this->em->expects($this->exactly(2))->method('getRepository')->willReturnMap([
|
|
|
|
[Domain::class, $repo],
|
|
|
|
[Visit::class, $repo2],
|
|
|
|
]);
|
2022-04-23 12:02:51 +03:00
|
|
|
|
|
|
|
$paginator = $this->helper->visitsForDomain($domain, new VisitsParams(), $apiKey);
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
|
|
|
|
2023-06-18 11:41:24 +03:00
|
|
|
#[Test, DataProviderExternal(ApiKeyDataProviders::class, 'adminApiKeysProvider')]
|
2022-04-23 12:02:51 +03:00
|
|
|
public function visitsForDefaultDomainAreReturnedAsExpected(?ApiKey $apiKey): void
|
|
|
|
{
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(DomainRepository::class);
|
|
|
|
$repo->expects($this->never())->method('domainExists');
|
2022-04-23 12:02:51 +03:00
|
|
|
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(
|
|
|
|
static fn () => Visit::forValidShortUrl(ShortUrl::createFake(), Visitor::emptyInstance()),
|
|
|
|
range(0, 1),
|
|
|
|
);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo2 = $this->createMock(VisitRepository::class);
|
|
|
|
$repo2->method('findVisitsByDomain')->with(
|
|
|
|
'DEFAULT',
|
|
|
|
$this->isInstanceOf(VisitsListFiltering::class),
|
|
|
|
)->willReturn($list);
|
|
|
|
$repo2->method('countVisitsByDomain')->with(
|
|
|
|
'DEFAULT',
|
|
|
|
$this->isInstanceOf(VisitsCountFiltering::class),
|
|
|
|
)->willReturn(1);
|
|
|
|
|
|
|
|
$this->em->expects($this->exactly(2))->method('getRepository')->willReturnMap([
|
|
|
|
[Domain::class, $repo],
|
|
|
|
[Visit::class, $repo2],
|
|
|
|
]);
|
2022-04-23 12:02:51 +03:00
|
|
|
|
|
|
|
$paginator = $this->helper->visitsForDomain('DEFAULT', new VisitsParams(), $apiKey);
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2021-02-10 00:40:40 +03:00
|
|
|
public function orphanVisitsAreReturnedAsExpected(): void
|
|
|
|
{
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(static fn () => Visit::forBasePath(Visitor::emptyInstance()), range(0, 3));
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(VisitRepository::class);
|
|
|
|
$repo->expects($this->once())->method('countOrphanVisits')->with(
|
2024-02-10 15:57:16 +03:00
|
|
|
$this->isInstanceOf(OrphanVisitsCountFiltering::class),
|
2022-10-23 22:05:13 +03:00
|
|
|
)->willReturn(count($list));
|
|
|
|
$repo->expects($this->once())->method('findOrphanVisits')->with(
|
2024-02-10 15:57:16 +03:00
|
|
|
$this->isInstanceOf(OrphanVisitsListFiltering::class),
|
2022-10-23 22:05:13 +03:00
|
|
|
)->willReturn($list);
|
|
|
|
$this->em->expects($this->once())->method('getRepository')->with(Visit::class)->willReturn($repo);
|
2021-02-10 00:40:40 +03:00
|
|
|
|
2024-02-10 19:51:42 +03:00
|
|
|
$paginator = $this->helper->orphanVisits(new OrphanVisitsParams());
|
2021-02-10 00:40:40 +03:00
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
2022-01-16 14:29:36 +03:00
|
|
|
|
2023-02-09 22:42:18 +03:00
|
|
|
#[Test]
|
2022-01-16 14:29:36 +03:00
|
|
|
public function nonOrphanVisitsAreReturnedAsExpected(): void
|
|
|
|
{
|
2023-11-29 14:34:13 +03:00
|
|
|
$list = array_map(
|
|
|
|
static fn () => Visit::forValidShortUrl(ShortUrl::createFake(), Visitor::emptyInstance()),
|
|
|
|
range(0, 3),
|
|
|
|
);
|
2022-10-23 22:05:13 +03:00
|
|
|
$repo = $this->createMock(VisitRepository::class);
|
|
|
|
$repo->expects($this->once())->method('countNonOrphanVisits')->with(
|
|
|
|
$this->isInstanceOf(VisitsCountFiltering::class),
|
|
|
|
)->willReturn(count($list));
|
|
|
|
$repo->expects($this->once())->method('findNonOrphanVisits')->with(
|
|
|
|
$this->isInstanceOf(VisitsListFiltering::class),
|
|
|
|
)->willReturn($list);
|
|
|
|
$this->em->expects($this->once())->method('getRepository')->with(Visit::class)->willReturn($repo);
|
2022-01-16 14:29:36 +03:00
|
|
|
|
|
|
|
$paginator = $this->helper->nonOrphanVisits(new VisitsParams());
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
}
|
2020-05-01 12:57:46 +03:00
|
|
|
}
|