mirror of
https://github.com/shlinkio/shlink.git
synced 2024-12-23 23:50:09 -06:00
165 lines
6.5 KiB
PHP
165 lines
6.5 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace ShlinkioTest\Shlink\Core\Service;
|
|
|
|
use Cake\Chronos\Chronos;
|
|
use Doctrine\ORM\EntityManagerInterface;
|
|
use PHPUnit\Framework\TestCase;
|
|
use Prophecy\Argument;
|
|
use Prophecy\PhpUnit\ProphecyTrait;
|
|
use Prophecy\Prophecy\ObjectProphecy;
|
|
use Psr\EventDispatcher\EventDispatcherInterface;
|
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
|
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
|
|
use Shlinkio\Shlink\Core\Model\ShortUrlMeta;
|
|
use Shlinkio\Shlink\Core\Repository\ShortUrlRepository;
|
|
use Shlinkio\Shlink\Core\Service\ShortUrl\ShortCodeUniquenessHelperInterface;
|
|
use Shlinkio\Shlink\Core\Service\UrlShortener;
|
|
use Shlinkio\Shlink\Core\ShortUrl\Helper\ShortUrlTitleResolutionHelperInterface;
|
|
use Shlinkio\Shlink\Core\ShortUrl\Resolver\SimpleShortUrlRelationResolver;
|
|
|
|
class UrlShortenerTest extends TestCase
|
|
{
|
|
use ProphecyTrait;
|
|
|
|
private UrlShortener $urlShortener;
|
|
private ObjectProphecy $em;
|
|
private ObjectProphecy $titleResolutionHelper;
|
|
private ObjectProphecy $shortCodeHelper;
|
|
private ObjectProphecy $eventDispatcher;
|
|
|
|
protected function setUp(): void
|
|
{
|
|
$this->titleResolutionHelper = $this->prophesize(ShortUrlTitleResolutionHelperInterface::class);
|
|
$this->titleResolutionHelper->processTitleAndValidateUrl(Argument::cetera())->willReturnArgument();
|
|
|
|
$this->em = $this->prophesize(EntityManagerInterface::class);
|
|
$this->em->persist(Argument::any())->will(function ($arguments): void {
|
|
/** @var ShortUrl $shortUrl */
|
|
[$shortUrl] = $arguments;
|
|
$shortUrl->setId('10');
|
|
});
|
|
$this->em->wrapInTransaction(Argument::type('callable'))->will(function (array $args) {
|
|
/** @var callable $callback */
|
|
[$callback] = $args;
|
|
|
|
return $callback();
|
|
});
|
|
$repo = $this->prophesize(ShortUrlRepository::class);
|
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
|
|
|
|
$this->shortCodeHelper = $this->prophesize(ShortCodeUniquenessHelperInterface::class);
|
|
$this->shortCodeHelper->ensureShortCodeUniqueness(Argument::cetera())->willReturn(true);
|
|
|
|
$this->eventDispatcher = $this->prophesize(EventDispatcherInterface::class);
|
|
|
|
$this->urlShortener = new UrlShortener(
|
|
$this->titleResolutionHelper->reveal(),
|
|
$this->em->reveal(),
|
|
new SimpleShortUrlRelationResolver(),
|
|
$this->shortCodeHelper->reveal(),
|
|
$this->eventDispatcher->reveal(),
|
|
);
|
|
}
|
|
|
|
/** @test */
|
|
public function urlIsProperlyShortened(): void
|
|
{
|
|
$longUrl = 'http://foobar.com/12345/hello?foo=bar';
|
|
$meta = ShortUrlMeta::fromRawData(['longUrl' => $longUrl]);
|
|
$shortUrl = $this->urlShortener->shorten($meta);
|
|
|
|
self::assertEquals($longUrl, $shortUrl->getLongUrl());
|
|
$this->titleResolutionHelper->processTitleAndValidateUrl($meta)->shouldHaveBeenCalledOnce();
|
|
}
|
|
|
|
/** @test */
|
|
public function exceptionIsThrownWhenNonUniqueSlugIsProvided(): void
|
|
{
|
|
$ensureUniqueness = $this->shortCodeHelper->ensureShortCodeUniqueness(Argument::cetera())->willReturn(false);
|
|
|
|
$ensureUniqueness->shouldBeCalledOnce();
|
|
$this->expectException(NonUniqueSlugException::class);
|
|
|
|
$this->urlShortener->shorten(ShortUrlMeta::fromRawData(
|
|
['customSlug' => 'custom-slug', 'longUrl' => 'http://foobar.com/12345/hello?foo=bar'],
|
|
));
|
|
}
|
|
|
|
/**
|
|
* @test
|
|
* @dataProvider provideExistingShortUrls
|
|
*/
|
|
public function existingShortUrlIsReturnedWhenRequested(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($meta);
|
|
|
|
$findExisting->shouldHaveBeenCalledOnce();
|
|
$getRepo->shouldHaveBeenCalledOnce();
|
|
$this->em->persist(Argument::cetera())->shouldNotHaveBeenCalled();
|
|
$this->titleResolutionHelper->processTitleAndValidateUrl(Argument::cetera())->shouldNotHaveBeenCalled();
|
|
self::assertSame($expected, $result);
|
|
}
|
|
|
|
public function provideExistingShortUrls(): iterable
|
|
{
|
|
$url = 'http://foo.com';
|
|
|
|
yield [ShortUrlMeta::fromRawData(['findIfExists' => true, 'longUrl' => $url]), ShortUrl::withLongUrl(
|
|
$url,
|
|
)];
|
|
yield [ShortUrlMeta::fromRawData(
|
|
['findIfExists' => true, 'customSlug' => 'foo', 'longUrl' => $url],
|
|
), ShortUrl::withLongUrl($url)];
|
|
yield [
|
|
ShortUrlMeta::fromRawData(['findIfExists' => true, 'longUrl' => $url, 'tags' => ['foo', 'bar']]),
|
|
ShortUrl::fromMeta(ShortUrlMeta::fromRawData(['longUrl' => $url, 'tags' => ['foo', 'bar']])),
|
|
];
|
|
yield [
|
|
ShortUrlMeta::fromRawData(['findIfExists' => true, 'maxVisits' => 3, 'longUrl' => $url]),
|
|
ShortUrl::fromMeta(ShortUrlMeta::fromRawData(['maxVisits' => 3, 'longUrl' => $url])),
|
|
];
|
|
yield [
|
|
ShortUrlMeta::fromRawData(
|
|
['findIfExists' => true, 'validSince' => Chronos::parse('2017-01-01'), 'longUrl' => $url],
|
|
),
|
|
ShortUrl::fromMeta(
|
|
ShortUrlMeta::fromRawData(['validSince' => Chronos::parse('2017-01-01'), 'longUrl' => $url]),
|
|
),
|
|
];
|
|
yield [
|
|
ShortUrlMeta::fromRawData(
|
|
['findIfExists' => true, 'validUntil' => Chronos::parse('2017-01-01'), 'longUrl' => $url],
|
|
),
|
|
ShortUrl::fromMeta(
|
|
ShortUrlMeta::fromRawData(['validUntil' => Chronos::parse('2017-01-01'), 'longUrl' => $url]),
|
|
),
|
|
];
|
|
yield [
|
|
ShortUrlMeta::fromRawData(['findIfExists' => true, 'domain' => 'example.com', 'longUrl' => $url]),
|
|
ShortUrl::fromMeta(ShortUrlMeta::fromRawData(['domain' => 'example.com', 'longUrl' => $url])),
|
|
];
|
|
yield [
|
|
ShortUrlMeta::fromRawData([
|
|
'findIfExists' => true,
|
|
'validUntil' => Chronos::parse('2017-01-01'),
|
|
'maxVisits' => 4,
|
|
'longUrl' => $url,
|
|
'tags' => ['baz', 'foo', 'bar'],
|
|
]),
|
|
ShortUrl::fromMeta(ShortUrlMeta::fromRawData([
|
|
'validUntil' => Chronos::parse('2017-01-01'),
|
|
'maxVisits' => 4,
|
|
'longUrl' => $url,
|
|
'tags' => ['foo', 'bar', 'baz'],
|
|
])),
|
|
];
|
|
}
|
|
}
|