shlink/module/Core/test/Service/UrlShortenerTest.php

171 lines
6.2 KiB
PHP
Raw Normal View History

2016-04-17 06:42:52 -05:00
<?php
2019-10-05 10:26:10 -05:00
2017-10-12 03:13:20 -05:00
declare(strict_types=1);
2016-07-19 11:01:39 -05:00
namespace ShlinkioTest\Shlink\Core\Service;
2016-04-17 06:42:52 -05:00
use Cake\Chronos\Chronos;
use Doctrine\Common\Collections\ArrayCollection;
2016-04-17 06:42:52 -05:00
use Doctrine\ORM\EntityManagerInterface;
2017-03-24 14:34:18 -05:00
use PHPUnit\Framework\TestCase;
2016-04-17 06:42:52 -05:00
use Prophecy\Argument;
2020-11-02 04:50:19 -06:00
use Prophecy\PhpUnit\ProphecyTrait;
2016-04-17 06:42:52 -05:00
use Prophecy\Prophecy\ObjectProphecy;
2016-07-19 11:01:39 -05:00
use Shlinkio\Shlink\Core\Entity\ShortUrl;
use Shlinkio\Shlink\Core\Entity\Tag;
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
use Shlinkio\Shlink\Core\Model\ShortUrlMeta;
use Shlinkio\Shlink\Core\Repository\ShortUrlRepository;
use Shlinkio\Shlink\Core\Service\ShortUrl\ShortCodeHelperInterface;
2016-07-19 11:01:39 -05:00
use Shlinkio\Shlink\Core\Service\UrlShortener;
use Shlinkio\Shlink\Core\ShortUrl\Resolver\SimpleShortUrlRelationResolver;
2019-11-16 03:19:25 -06:00
use Shlinkio\Shlink\Core\Util\UrlValidatorInterface;
2016-04-17 06:42:52 -05:00
class UrlShortenerTest extends TestCase
{
2020-11-02 04:50:19 -06:00
use ProphecyTrait;
private UrlShortener $urlShortener;
private ObjectProphecy $em;
private ObjectProphecy $urlValidator;
private ObjectProphecy $shortCodeHelper;
2016-04-17 06:42:52 -05:00
public function setUp(): void
2016-04-17 06:42:52 -05:00
{
2019-11-16 03:19:25 -06:00
$this->urlValidator = $this->prophesize(UrlValidatorInterface::class);
$this->urlValidator->validateUrl('http://foobar.com/12345/hello?foo=bar', null)->will(
function (): void {
},
);
2016-04-17 06:42:52 -05:00
$this->em = $this->prophesize(EntityManagerInterface::class);
2020-01-01 13:48:31 -06:00
$this->em->persist(Argument::any())->will(function ($arguments): void {
2016-04-17 06:42:52 -05:00
/** @var ShortUrl $shortUrl */
[$shortUrl] = $arguments;
$shortUrl->setId('10');
2016-04-17 06:42:52 -05:00
});
$this->em->transactional(Argument::type('callable'))->will(function (array $args) {
/** @var callable $callback */
[$callback] = $args;
return $callback();
});
$repo = $this->prophesize(ShortUrlRepository::class);
$repo->shortCodeIsInUse(Argument::cetera())->willReturn(false);
2016-04-17 06:42:52 -05:00
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
$this->shortCodeHelper = $this->prophesize(ShortCodeHelperInterface::class);
$this->shortCodeHelper->ensureShortCodeUniqueness(Argument::cetera())->willReturn(true);
$this->urlShortener = new UrlShortener(
$this->urlValidator->reveal(),
$this->em->reveal(),
new SimpleShortUrlRelationResolver(),
$this->shortCodeHelper->reveal(),
);
2016-04-17 06:42:52 -05:00
}
2019-02-17 13:28:34 -06:00
/** @test */
public function urlIsProperlyShortened(): void
2016-04-17 06:42:52 -05:00
{
$shortUrl = $this->urlShortener->shorten(
'http://foobar.com/12345/hello?foo=bar',
[],
2020-01-01 13:48:31 -06:00
ShortUrlMeta::createEmpty(),
);
2020-10-03 17:35:14 -05:00
self::assertEquals('http://foobar.com/12345/hello?foo=bar', $shortUrl->getLongUrl());
2016-04-17 06:42:52 -05:00
}
2019-10-11 04:28:53 -05:00
/** @test */
public function exceptionIsThrownWhenNonUniqueSlugIsProvided(): void
2019-10-11 04:28:53 -05:00
{
$ensureUniqueness = $this->shortCodeHelper->ensureShortCodeUniqueness(Argument::cetera())->willReturn(false);
2019-10-11 04:28:53 -05:00
$ensureUniqueness->shouldBeCalledOnce();
$this->expectException(NonUniqueSlugException::class);
$this->urlShortener->shorten(
'http://foobar.com/12345/hello?foo=bar',
2019-10-11 04:28:53 -05:00
[],
ShortUrlMeta::fromRawData(['customSlug' => 'custom-slug']),
2019-10-11 04:28:53 -05:00
);
}
2016-04-17 06:42:52 -05:00
/**
* @test
2019-02-17 13:28:34 -06:00
* @dataProvider provideExistingShortUrls
2016-04-17 06:42:52 -05:00
*/
public function existingShortUrlIsReturnedWhenRequested(
string $url,
array $tags,
ShortUrlMeta $meta,
ShortUrl $expected
): void {
$repo = $this->prophesize(ShortUrlRepository::class);
$findExisting = $repo->findOneMatching(Argument::cetera())->willReturn($expected);
$getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
$result = $this->urlShortener->shorten($url, $tags, $meta);
$findExisting->shouldHaveBeenCalledOnce();
$getRepo->shouldHaveBeenCalledOnce();
$this->em->persist(Argument::cetera())->shouldNotHaveBeenCalled();
$this->urlValidator->validateUrl(Argument::cetera())->shouldNotHaveBeenCalled();
2020-10-03 17:35:14 -05:00
self::assertSame($expected, $result);
}
2019-02-17 13:28:34 -06:00
public function provideExistingShortUrls(): iterable
{
$url = 'http://foo.com';
yield [$url, [], ShortUrlMeta::fromRawData(['findIfExists' => true]), new ShortUrl($url)];
yield [$url, [], ShortUrlMeta::fromRawData(
2020-01-01 13:48:31 -06:00
['findIfExists' => true, 'customSlug' => 'foo'],
2019-02-17 13:28:34 -06:00
), new ShortUrl($url)];
yield [
$url,
['foo', 'bar'],
ShortUrlMeta::fromRawData(['findIfExists' => true]),
2019-02-17 13:28:34 -06:00
(new ShortUrl($url))->setTags(new ArrayCollection([new Tag('bar'), new Tag('foo')])),
];
yield [
$url,
[],
ShortUrlMeta::fromRawData(['findIfExists' => true, 'maxVisits' => 3]),
new ShortUrl($url, ShortUrlMeta::fromRawData(['maxVisits' => 3])),
2019-02-17 13:28:34 -06:00
];
yield [
$url,
[],
ShortUrlMeta::fromRawData(['findIfExists' => true, 'validSince' => Chronos::parse('2017-01-01')]),
new ShortUrl($url, ShortUrlMeta::fromRawData(['validSince' => Chronos::parse('2017-01-01')])),
2019-02-17 13:28:34 -06:00
];
yield [
$url,
[],
ShortUrlMeta::fromRawData(['findIfExists' => true, 'validUntil' => Chronos::parse('2017-01-01')]),
new ShortUrl($url, ShortUrlMeta::fromRawData(['validUntil' => Chronos::parse('2017-01-01')])),
2019-02-17 13:28:34 -06:00
];
2019-10-11 04:28:53 -05:00
yield [
$url,
[],
ShortUrlMeta::fromRawData(['findIfExists' => true, 'domain' => 'example.com']),
new ShortUrl($url, ShortUrlMeta::fromRawData(['domain' => 'example.com'])),
2019-10-11 04:28:53 -05:00
];
2019-02-17 13:28:34 -06:00
yield [
$url,
['baz', 'foo', 'bar'],
ShortUrlMeta::fromRawData([
2019-02-17 13:28:34 -06:00
'findIfExists' => true,
'validUntil' => Chronos::parse('2017-01-01'),
'maxVisits' => 4,
]),
(new ShortUrl($url, ShortUrlMeta::fromRawData([
2019-02-17 13:28:34 -06:00
'validUntil' => Chronos::parse('2017-01-01'),
'maxVisits' => 4,
])))->setTags(new ArrayCollection([new Tag('foo'), new Tag('bar'), new Tag('baz')])),
];
}
2016-04-17 06:42:52 -05:00
}