2020-05-01 11:57:46 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
|
|
|
|
|
|
|
namespace ShlinkioTest\Shlink\Core\Visit;
|
|
|
|
|
|
|
|
use Doctrine\ORM\EntityManagerInterface;
|
2021-02-08 19:46:51 +01:00
|
|
|
use Laminas\Stdlib\ArrayUtils;
|
2020-05-01 11:57:46 +02:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2021-02-08 19:46:51 +01:00
|
|
|
use Prophecy\Argument;
|
2020-11-02 11:50:19 +01:00
|
|
|
use Prophecy\PhpUnit\ProphecyTrait;
|
2020-05-01 11:57:46 +02:00
|
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
2021-02-08 19:46:51 +01:00
|
|
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
|
|
|
use Shlinkio\Shlink\Core\Entity\Tag;
|
2020-05-01 11:57:46 +02:00
|
|
|
use Shlinkio\Shlink\Core\Entity\Visit;
|
2021-02-08 19:46:51 +01:00
|
|
|
use Shlinkio\Shlink\Core\Exception\ShortUrlNotFoundException;
|
|
|
|
use Shlinkio\Shlink\Core\Exception\TagNotFoundException;
|
|
|
|
use Shlinkio\Shlink\Core\Model\ShortUrlIdentifier;
|
|
|
|
use Shlinkio\Shlink\Core\Model\Visitor;
|
|
|
|
use Shlinkio\Shlink\Core\Model\VisitsParams;
|
|
|
|
use Shlinkio\Shlink\Core\Repository\ShortUrlRepositoryInterface;
|
|
|
|
use Shlinkio\Shlink\Core\Repository\TagRepository;
|
2020-05-01 11:57:46 +02:00
|
|
|
use Shlinkio\Shlink\Core\Repository\VisitRepository;
|
|
|
|
use Shlinkio\Shlink\Core\Visit\Model\VisitsStats;
|
2021-05-22 20:16:32 +02:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\VisitsCountFiltering;
|
2021-05-22 20:32:30 +02:00
|
|
|
use Shlinkio\Shlink\Core\Visit\Persistence\VisitsListFiltering;
|
2020-05-01 11:57:46 +02:00
|
|
|
use Shlinkio\Shlink\Core\Visit\VisitsStatsHelper;
|
2021-02-08 19:46:51 +01:00
|
|
|
use Shlinkio\Shlink\Rest\Entity\ApiKey;
|
|
|
|
use ShlinkioTest\Shlink\Core\Util\ApiKeyHelpersTrait;
|
2020-05-01 11:57:46 +02:00
|
|
|
|
2021-02-09 22:40:40 +01:00
|
|
|
use function count;
|
2020-05-01 11:57:46 +02:00
|
|
|
use function Functional\map;
|
|
|
|
use function range;
|
|
|
|
|
|
|
|
class VisitsStatsHelperTest extends TestCase
|
|
|
|
{
|
2021-02-08 19:46:51 +01:00
|
|
|
use ApiKeyHelpersTrait;
|
2020-11-02 11:50:19 +01:00
|
|
|
use ProphecyTrait;
|
|
|
|
|
2020-05-01 11:57:46 +02:00
|
|
|
private VisitsStatsHelper $helper;
|
|
|
|
private ObjectProphecy $em;
|
|
|
|
|
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
$this->em = $this->prophesize(EntityManagerInterface::class);
|
|
|
|
$this->helper = new VisitsStatsHelper($this->em->reveal());
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideCounts
|
|
|
|
*/
|
|
|
|
public function returnsExpectedVisitsStats(int $expectedCount): void
|
|
|
|
{
|
|
|
|
$repo = $this->prophesize(VisitRepository::class);
|
2021-02-08 22:44:58 +01:00
|
|
|
$count = $repo->countVisits(null)->willReturn($expectedCount * 3);
|
2021-05-22 20:16:32 +02:00
|
|
|
$countOrphan = $repo->countOrphanVisits(Argument::type(VisitsCountFiltering::class))->willReturn(
|
|
|
|
$expectedCount,
|
|
|
|
);
|
2020-05-01 11:57:46 +02:00
|
|
|
$getRepo = $this->em->getRepository(Visit::class)->willReturn($repo->reveal());
|
|
|
|
|
|
|
|
$stats = $this->helper->getVisitsStats();
|
|
|
|
|
2021-02-08 22:44:58 +01:00
|
|
|
self::assertEquals(new VisitsStats($expectedCount * 3, $expectedCount), $stats);
|
2020-05-01 11:57:46 +02:00
|
|
|
$count->shouldHaveBeenCalledOnce();
|
2021-02-08 22:44:58 +01:00
|
|
|
$countOrphan->shouldHaveBeenCalledOnce();
|
2020-05-01 11:57:46 +02:00
|
|
|
$getRepo->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function provideCounts(): iterable
|
|
|
|
{
|
|
|
|
return map(range(0, 50, 5), fn (int $value) => [$value]);
|
|
|
|
}
|
2021-02-08 19:46:51 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideAdminApiKeys
|
|
|
|
*/
|
|
|
|
public function infoReturnsVisitsForCertainShortCode(?ApiKey $apiKey): void
|
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
2021-05-23 08:41:42 +02:00
|
|
|
$identifier = ShortUrlIdentifier::fromShortCodeAndDomain($shortCode);
|
2021-05-23 12:31:10 +02:00
|
|
|
$spec = $apiKey?->spec();
|
2021-05-23 08:41:42 +02:00
|
|
|
|
2021-02-08 19:46:51 +01:00
|
|
|
$repo = $this->prophesize(ShortUrlRepositoryInterface::class);
|
2021-05-23 08:41:42 +02:00
|
|
|
$count = $repo->shortCodeIsInUse($identifier, $spec)->willReturn(
|
2021-05-02 10:33:27 +02:00
|
|
|
true,
|
|
|
|
);
|
2021-02-08 19:46:51 +01:00
|
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal())->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$list = map(range(0, 1), fn () => Visit::forValidShortUrl(ShortUrl::createEmpty(), Visitor::emptyInstance()));
|
|
|
|
$repo2 = $this->prophesize(VisitRepository::class);
|
2021-05-23 08:41:42 +02:00
|
|
|
$repo2->findVisitsByShortCode($identifier, Argument::type(VisitsListFiltering::class))->willReturn($list);
|
|
|
|
$repo2->countVisitsByShortCode($identifier, Argument::type(VisitsCountFiltering::class))->willReturn(1);
|
2021-02-08 19:46:51 +01:00
|
|
|
$this->em->getRepository(Visit::class)->willReturn($repo2->reveal())->shouldBeCalledOnce();
|
|
|
|
|
2021-05-23 08:41:42 +02:00
|
|
|
$paginator = $this->helper->visitsForShortUrl($identifier, new VisitsParams(), $apiKey);
|
2021-02-08 19:46:51 +01:00
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
$count->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @test */
|
|
|
|
public function throwsExceptionWhenRequestingVisitsForInvalidShortCode(): void
|
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
2021-05-23 08:41:42 +02:00
|
|
|
$identifier = ShortUrlIdentifier::fromShortCodeAndDomain($shortCode);
|
|
|
|
|
2021-02-08 19:46:51 +01:00
|
|
|
$repo = $this->prophesize(ShortUrlRepositoryInterface::class);
|
2021-05-23 08:41:42 +02:00
|
|
|
$count = $repo->shortCodeIsInUse($identifier, null)->willReturn(
|
2021-05-02 10:33:27 +02:00
|
|
|
false,
|
|
|
|
);
|
2021-02-08 19:46:51 +01:00
|
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal())->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$this->expectException(ShortUrlNotFoundException::class);
|
|
|
|
$count->shouldBeCalledOnce();
|
|
|
|
|
2021-05-23 08:41:42 +02:00
|
|
|
$this->helper->visitsForShortUrl($identifier, new VisitsParams());
|
2021-02-08 19:46:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/** @test */
|
|
|
|
public function throwsExceptionWhenRequestingVisitsForInvalidTag(): void
|
|
|
|
{
|
|
|
|
$tag = 'foo';
|
2021-03-14 09:59:35 +01:00
|
|
|
$apiKey = ApiKey::create();
|
2021-02-08 19:46:51 +01:00
|
|
|
$repo = $this->prophesize(TagRepository::class);
|
|
|
|
$tagExists = $repo->tagExists($tag, $apiKey)->willReturn(false);
|
|
|
|
$getRepo = $this->em->getRepository(Tag::class)->willReturn($repo->reveal());
|
|
|
|
|
|
|
|
$this->expectException(TagNotFoundException::class);
|
|
|
|
$tagExists->shouldBeCalledOnce();
|
|
|
|
$getRepo->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$this->helper->visitsForTag($tag, new VisitsParams(), $apiKey);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @test
|
|
|
|
* @dataProvider provideAdminApiKeys
|
|
|
|
*/
|
|
|
|
public function visitsForTagAreReturnedAsExpected(?ApiKey $apiKey): void
|
|
|
|
{
|
|
|
|
$tag = 'foo';
|
|
|
|
$repo = $this->prophesize(TagRepository::class);
|
|
|
|
$tagExists = $repo->tagExists($tag, $apiKey)->willReturn(true);
|
|
|
|
$getRepo = $this->em->getRepository(Tag::class)->willReturn($repo->reveal());
|
|
|
|
|
|
|
|
$list = map(range(0, 1), fn () => Visit::forValidShortUrl(ShortUrl::createEmpty(), Visitor::emptyInstance()));
|
|
|
|
$repo2 = $this->prophesize(VisitRepository::class);
|
2021-05-22 20:32:30 +02:00
|
|
|
$repo2->findVisitsByTag($tag, Argument::type(VisitsListFiltering::class))->willReturn($list);
|
2021-05-22 20:16:32 +02:00
|
|
|
$repo2->countVisitsByTag($tag, Argument::type(VisitsCountFiltering::class))->willReturn(1);
|
2021-02-08 19:46:51 +01:00
|
|
|
$this->em->getRepository(Visit::class)->willReturn($repo2->reveal())->shouldBeCalledOnce();
|
|
|
|
|
|
|
|
$paginator = $this->helper->visitsForTag($tag, new VisitsParams(), $apiKey);
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
$tagExists->shouldHaveBeenCalledOnce();
|
|
|
|
$getRepo->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
2021-02-09 22:40:40 +01:00
|
|
|
|
|
|
|
/** @test */
|
|
|
|
public function orphanVisitsAreReturnedAsExpected(): void
|
|
|
|
{
|
|
|
|
$list = map(range(0, 3), fn () => Visit::forBasePath(Visitor::emptyInstance()));
|
|
|
|
$repo = $this->prophesize(VisitRepository::class);
|
2021-05-22 20:16:32 +02:00
|
|
|
$countVisits = $repo->countOrphanVisits(Argument::type(VisitsCountFiltering::class))->willReturn(count($list));
|
2021-05-22 20:32:30 +02:00
|
|
|
$listVisits = $repo->findOrphanVisits(Argument::type(VisitsListFiltering::class))->willReturn($list);
|
2021-02-09 22:40:40 +01:00
|
|
|
$getRepo = $this->em->getRepository(Visit::class)->willReturn($repo->reveal());
|
|
|
|
|
|
|
|
$paginator = $this->helper->orphanVisits(new VisitsParams());
|
|
|
|
|
|
|
|
self::assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentPageResults()));
|
|
|
|
$listVisits->shouldHaveBeenCalledOnce();
|
|
|
|
$countVisits->shouldHaveBeenCalledOnce();
|
|
|
|
$getRepo->shouldHaveBeenCalledOnce();
|
|
|
|
}
|
2020-05-01 11:57:46 +02:00
|
|
|
}
|