2016-05-01 18:54:56 +03:00
|
|
|
<?php
|
2019-10-05 18:26:10 +03:00
|
|
|
|
2017-10-12 11:13:20 +03:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
2016-07-19 19:01:39 +03:00
|
|
|
namespace ShlinkioTest\Shlink\Core\Service;
|
2016-05-01 18:54:56 +03:00
|
|
|
|
|
|
|
use Doctrine\ORM\EntityManager;
|
|
|
|
use Doctrine\ORM\EntityRepository;
|
2018-11-28 22:39:08 +03:00
|
|
|
use PHPUnit\Framework\Assert;
|
2017-03-24 22:34:18 +03:00
|
|
|
use PHPUnit\Framework\TestCase;
|
2016-05-01 18:54:56 +03:00
|
|
|
use Prophecy\Argument;
|
2016-07-30 23:55:28 +03:00
|
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
2019-07-13 13:04:21 +03:00
|
|
|
use Psr\EventDispatcher\EventDispatcherInterface;
|
2018-11-27 23:09:27 +03:00
|
|
|
use Shlinkio\Shlink\Common\Util\DateRange;
|
2016-07-19 19:01:39 +03:00
|
|
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
2016-07-30 23:55:28 +03:00
|
|
|
use Shlinkio\Shlink\Core\Entity\Visit;
|
2019-07-13 13:04:21 +03:00
|
|
|
use Shlinkio\Shlink\Core\EventDispatcher\ShortUrlVisited;
|
2018-10-18 21:19:22 +03:00
|
|
|
use Shlinkio\Shlink\Core\Model\Visitor;
|
2018-11-27 23:09:27 +03:00
|
|
|
use Shlinkio\Shlink\Core\Model\VisitsParams;
|
2016-07-30 23:55:28 +03:00
|
|
|
use Shlinkio\Shlink\Core\Repository\VisitRepository;
|
2016-07-19 19:01:39 +03:00
|
|
|
use Shlinkio\Shlink\Core\Service\VisitsTracker;
|
2018-11-28 22:39:08 +03:00
|
|
|
use Zend\Stdlib\ArrayUtils;
|
2016-05-01 18:54:56 +03:00
|
|
|
|
|
|
|
class VisitsTrackerTest extends TestCase
|
|
|
|
{
|
2019-12-30 00:48:40 +03:00
|
|
|
private VisitsTracker $visitsTracker;
|
|
|
|
private ObjectProphecy $em;
|
|
|
|
private ObjectProphecy $eventDispatcher;
|
2016-07-30 23:55:28 +03:00
|
|
|
|
2019-02-16 12:53:45 +03:00
|
|
|
public function setUp(): void
|
2016-07-30 23:55:28 +03:00
|
|
|
{
|
|
|
|
$this->em = $this->prophesize(EntityManager::class);
|
2019-07-13 13:04:21 +03:00
|
|
|
$this->eventDispatcher = $this->prophesize(EventDispatcherInterface::class);
|
|
|
|
|
|
|
|
$this->visitsTracker = new VisitsTracker($this->em->reveal(), $this->eventDispatcher->reveal());
|
2016-07-30 23:55:28 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-07-13 13:04:21 +03:00
|
|
|
public function trackPersistsVisit(): void
|
2016-05-01 18:54:56 +03:00
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
|
|
|
$repo = $this->prophesize(EntityRepository::class);
|
2018-10-28 18:00:54 +03:00
|
|
|
$repo->findOneBy(['shortCode' => $shortCode])->willReturn(new ShortUrl(''));
|
2016-05-01 18:54:56 +03:00
|
|
|
|
2018-11-11 15:18:21 +03:00
|
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal())->shouldBeCalledOnce();
|
2019-12-30 01:16:55 +03:00
|
|
|
$this->em->persist(Argument::that(fn (Visit $visit) => $visit->setId('1')))->shouldBeCalledOnce();
|
2019-11-20 22:03:06 +03:00
|
|
|
$this->em->flush()->shouldBeCalledOnce();
|
2016-07-30 23:55:28 +03:00
|
|
|
|
2018-10-18 21:19:22 +03:00
|
|
|
$this->visitsTracker->track($shortCode, Visitor::emptyInstance());
|
2019-07-13 13:04:21 +03:00
|
|
|
|
|
|
|
$this->eventDispatcher->dispatch(Argument::type(ShortUrlVisited::class))->shouldHaveBeenCalled();
|
2016-07-30 23:55:28 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-07-13 13:04:21 +03:00
|
|
|
public function trackedIpAddressGetsObfuscated(): void
|
2016-08-09 10:13:39 +03:00
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
|
|
|
$repo = $this->prophesize(EntityRepository::class);
|
2018-10-28 18:00:54 +03:00
|
|
|
$repo->findOneBy(['shortCode' => $shortCode])->willReturn(new ShortUrl(''));
|
2016-08-09 10:13:39 +03:00
|
|
|
|
2018-11-11 15:18:21 +03:00
|
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal())->shouldBeCalledOnce();
|
2018-11-28 22:39:08 +03:00
|
|
|
$this->em->persist(Argument::any())->will(function ($args) {
|
2016-08-09 10:13:39 +03:00
|
|
|
/** @var Visit $visit */
|
|
|
|
$visit = $args[0];
|
2018-11-28 22:39:08 +03:00
|
|
|
Assert::assertEquals('4.3.2.0', $visit->getRemoteAddr());
|
2019-07-13 13:04:21 +03:00
|
|
|
$visit->setId('1');
|
|
|
|
return $visit;
|
2019-11-20 22:03:06 +03:00
|
|
|
})->shouldBeCalledOnce();
|
|
|
|
$this->em->flush()->shouldBeCalledOnce();
|
2016-08-09 10:13:39 +03:00
|
|
|
|
2018-10-18 21:19:22 +03:00
|
|
|
$this->visitsTracker->track($shortCode, new Visitor('', '', '4.3.2.1'));
|
2019-07-13 13:04:21 +03:00
|
|
|
|
|
|
|
$this->eventDispatcher->dispatch(Argument::type(ShortUrlVisited::class))->shouldHaveBeenCalled();
|
2016-08-09 10:13:39 +03:00
|
|
|
}
|
|
|
|
|
2019-02-17 22:28:34 +03:00
|
|
|
/** @test */
|
2019-07-13 13:04:21 +03:00
|
|
|
public function infoReturnsVisitsForCertainShortCode(): void
|
2016-07-30 23:55:28 +03:00
|
|
|
{
|
|
|
|
$shortCode = '123ABC';
|
|
|
|
$repo = $this->prophesize(EntityRepository::class);
|
2018-11-28 22:39:08 +03:00
|
|
|
$count = $repo->count(['shortCode' => $shortCode])->willReturn(1);
|
2018-11-11 15:18:21 +03:00
|
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal())->shouldBeCalledOnce();
|
2016-07-30 23:55:28 +03:00
|
|
|
|
|
|
|
$list = [
|
2018-10-28 18:20:10 +03:00
|
|
|
new Visit(new ShortUrl(''), Visitor::emptyInstance()),
|
|
|
|
new Visit(new ShortUrl(''), Visitor::emptyInstance()),
|
2016-07-30 23:55:28 +03:00
|
|
|
];
|
|
|
|
$repo2 = $this->prophesize(VisitRepository::class);
|
2018-11-28 22:39:08 +03:00
|
|
|
$repo2->findVisitsByShortCode($shortCode, Argument::type(DateRange::class), 1, 0)->willReturn($list);
|
|
|
|
$repo2->countVisitsByShortCode($shortCode, Argument::type(DateRange::class))->willReturn(1);
|
2018-11-11 15:18:21 +03:00
|
|
|
$this->em->getRepository(Visit::class)->willReturn($repo2->reveal())->shouldBeCalledOnce();
|
2016-05-01 18:54:56 +03:00
|
|
|
|
2018-11-28 22:39:08 +03:00
|
|
|
$paginator = $this->visitsTracker->info($shortCode, new VisitsParams());
|
|
|
|
|
|
|
|
$this->assertEquals($list, ArrayUtils::iteratorToArray($paginator->getCurrentItems()));
|
|
|
|
$count->shouldHaveBeenCalledOnce();
|
2016-05-01 18:54:56 +03:00
|
|
|
}
|
|
|
|
}
|