mirror of
https://github.com/shlinkio/shlink.git
synced 2024-11-23 21:27:44 +03:00
Added option to provide custom slug when creating a short url
This commit is contained in:
parent
0232f68b91
commit
1f7a94794d
5 changed files with 118 additions and 7 deletions
|
@ -14,6 +14,7 @@
|
||||||
"require": {
|
"require": {
|
||||||
"php": "^7.0",
|
"php": "^7.0",
|
||||||
"acelaya/ze-content-based-error-handler": "^2.0",
|
"acelaya/ze-content-based-error-handler": "^2.0",
|
||||||
|
"cocur/slugify": "^3.0",
|
||||||
"doctrine/annotations": "^1.4 <1.5",
|
"doctrine/annotations": "^1.4 <1.5",
|
||||||
"doctrine/cache": "^1.6 <1.7",
|
"doctrine/cache": "^1.6 <1.7",
|
||||||
"doctrine/collections": "^1.4 <1.5",
|
"doctrine/collections": "^1.4 <1.5",
|
||||||
|
|
14
module/Core/src/Exception/NonUniqueSlugException.php
Normal file
14
module/Core/src/Exception/NonUniqueSlugException.php
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
<?php
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace Shlinkio\Shlink\Core\Exception;
|
||||||
|
|
||||||
|
use Shlinkio\Shlink\Common\Exception\InvalidArgumentException;
|
||||||
|
|
||||||
|
class NonUniqueSlugException extends InvalidArgumentException
|
||||||
|
{
|
||||||
|
public static function fromSlug(string $slug): self
|
||||||
|
{
|
||||||
|
return new self(sprintf('Provided slug "%s" is not unique.', $slug));
|
||||||
|
}
|
||||||
|
}
|
|
@ -3,6 +3,8 @@ declare(strict_types=1);
|
||||||
|
|
||||||
namespace Shlinkio\Shlink\Core\Service;
|
namespace Shlinkio\Shlink\Core\Service;
|
||||||
|
|
||||||
|
use Cocur\Slugify\Slugify;
|
||||||
|
use Cocur\Slugify\SlugifyInterface;
|
||||||
use Doctrine\Common\Cache\Cache;
|
use Doctrine\Common\Cache\Cache;
|
||||||
use Doctrine\ORM\EntityManagerInterface;
|
use Doctrine\ORM\EntityManagerInterface;
|
||||||
use Doctrine\ORM\ORMException;
|
use Doctrine\ORM\ORMException;
|
||||||
|
@ -14,6 +16,7 @@ use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
||||||
use Shlinkio\Shlink\Core\Exception\EntityDoesNotExistException;
|
use Shlinkio\Shlink\Core\Exception\EntityDoesNotExistException;
|
||||||
use Shlinkio\Shlink\Core\Exception\InvalidShortCodeException;
|
use Shlinkio\Shlink\Core\Exception\InvalidShortCodeException;
|
||||||
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
|
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
|
||||||
|
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
|
||||||
use Shlinkio\Shlink\Core\Repository\ShortUrlRepository;
|
use Shlinkio\Shlink\Core\Repository\ShortUrlRepository;
|
||||||
use Shlinkio\Shlink\Core\Util\TagManagerTrait;
|
use Shlinkio\Shlink\Core\Util\TagManagerTrait;
|
||||||
|
|
||||||
|
@ -39,17 +42,23 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
* @var Cache
|
* @var Cache
|
||||||
*/
|
*/
|
||||||
private $cache;
|
private $cache;
|
||||||
|
/**
|
||||||
|
* @var SlugifyInterface
|
||||||
|
*/
|
||||||
|
private $slugger;
|
||||||
|
|
||||||
public function __construct(
|
public function __construct(
|
||||||
ClientInterface $httpClient,
|
ClientInterface $httpClient,
|
||||||
EntityManagerInterface $em,
|
EntityManagerInterface $em,
|
||||||
Cache $cache,
|
Cache $cache,
|
||||||
$chars = self::DEFAULT_CHARS
|
$chars = self::DEFAULT_CHARS,
|
||||||
|
SlugifyInterface $slugger = null
|
||||||
) {
|
) {
|
||||||
$this->httpClient = $httpClient;
|
$this->httpClient = $httpClient;
|
||||||
$this->em = $em;
|
$this->em = $em;
|
||||||
$this->chars = empty($chars) ? self::DEFAULT_CHARS : $chars;
|
$this->chars = empty($chars) ? self::DEFAULT_CHARS : $chars;
|
||||||
$this->cache = $cache;
|
$this->cache = $cache;
|
||||||
|
$this->slugger = $slugger ?: new Slugify();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -59,7 +68,9 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
* @param string[] $tags
|
* @param string[] $tags
|
||||||
* @param \DateTime|null $validSince
|
* @param \DateTime|null $validSince
|
||||||
* @param \DateTime|null $validUntil
|
* @param \DateTime|null $validUntil
|
||||||
|
* @param string|null $customSlug
|
||||||
* @return string
|
* @return string
|
||||||
|
* @throws NonUniqueSlugException
|
||||||
* @throws InvalidUrlException
|
* @throws InvalidUrlException
|
||||||
* @throws RuntimeException
|
* @throws RuntimeException
|
||||||
*/
|
*/
|
||||||
|
@ -67,7 +78,8 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
UriInterface $url,
|
UriInterface $url,
|
||||||
array $tags = [],
|
array $tags = [],
|
||||||
\DateTime $validSince = null,
|
\DateTime $validSince = null,
|
||||||
\DateTime $validUntil = null
|
\DateTime $validUntil = null,
|
||||||
|
string $customSlug = null
|
||||||
): string {
|
): string {
|
||||||
// If the url already exists in the database, just return its short code
|
// If the url already exists in the database, just return its short code
|
||||||
$shortUrl = $this->em->getRepository(ShortUrl::class)->findOneBy([
|
$shortUrl = $this->em->getRepository(ShortUrl::class)->findOneBy([
|
||||||
|
@ -79,6 +91,8 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
|
|
||||||
// Check that the URL exists
|
// Check that the URL exists
|
||||||
$this->checkUrlExists($url);
|
$this->checkUrlExists($url);
|
||||||
|
$customSlug = $this->processCustomSlug($customSlug);
|
||||||
|
|
||||||
|
|
||||||
// Transactionally insert the short url, then generate the short code and finally update the short code
|
// Transactionally insert the short url, then generate the short code and finally update the short code
|
||||||
try {
|
try {
|
||||||
|
@ -93,7 +107,7 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
$this->em->flush();
|
$this->em->flush();
|
||||||
|
|
||||||
// Generate the short code and persist it
|
// Generate the short code and persist it
|
||||||
$shortCode = $this->convertAutoincrementIdToShortCode($shortUrl->getId());
|
$shortCode = $customSlug ?? $this->convertAutoincrementIdToShortCode($shortUrl->getId());
|
||||||
$shortUrl->setShortCode($shortCode)
|
$shortUrl->setShortCode($shortCode)
|
||||||
->setTags($this->tagNamesToEntities($this->em, $tags));
|
->setTags($this->tagNamesToEntities($this->em, $tags));
|
||||||
$this->em->flush();
|
$this->em->flush();
|
||||||
|
@ -116,7 +130,7 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
* @param UriInterface $url
|
* @param UriInterface $url
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
protected function checkUrlExists(UriInterface $url)
|
private function checkUrlExists(UriInterface $url)
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$this->httpClient->request('GET', $url, ['allow_redirects' => [
|
$this->httpClient->request('GET', $url, ['allow_redirects' => [
|
||||||
|
@ -133,7 +147,7 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
protected function convertAutoincrementIdToShortCode($id)
|
private function convertAutoincrementIdToShortCode($id)
|
||||||
{
|
{
|
||||||
$id = ((int) $id) + 200000; // Increment the Id so that the generated shortcode is not too short
|
$id = ((int) $id) + 200000; // Increment the Id so that the generated shortcode is not too short
|
||||||
$length = strlen($this->chars);
|
$length = strlen($this->chars);
|
||||||
|
@ -148,6 +162,22 @@ class UrlShortener implements UrlShortenerInterface
|
||||||
return $this->chars[(int) $id] . $code;
|
return $this->chars[(int) $id] . $code;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function processCustomSlug($customSlug)
|
||||||
|
{
|
||||||
|
if ($customSlug === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If a custom slug was provided, check it is unique
|
||||||
|
$customSlug = $this->slugger->slugify($customSlug);
|
||||||
|
$shortUrl = $this->em->getRepository(ShortUrl::class)->findOneBy(['shortCode' => $customSlug]);
|
||||||
|
if ($shortUrl !== null) {
|
||||||
|
throw NonUniqueSlugException::fromSlug($customSlug);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $customSlug;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tries to find the mapped URL for provided short code. Returns null if not found
|
* Tries to find the mapped URL for provided short code. Returns null if not found
|
||||||
*
|
*
|
||||||
|
|
|
@ -8,6 +8,7 @@ use Shlinkio\Shlink\Common\Exception\RuntimeException;
|
||||||
use Shlinkio\Shlink\Core\Exception\EntityDoesNotExistException;
|
use Shlinkio\Shlink\Core\Exception\EntityDoesNotExistException;
|
||||||
use Shlinkio\Shlink\Core\Exception\InvalidShortCodeException;
|
use Shlinkio\Shlink\Core\Exception\InvalidShortCodeException;
|
||||||
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
|
use Shlinkio\Shlink\Core\Exception\InvalidUrlException;
|
||||||
|
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
|
||||||
|
|
||||||
interface UrlShortenerInterface
|
interface UrlShortenerInterface
|
||||||
{
|
{
|
||||||
|
@ -18,7 +19,9 @@ interface UrlShortenerInterface
|
||||||
* @param string[] $tags
|
* @param string[] $tags
|
||||||
* @param \DateTime|null $validSince
|
* @param \DateTime|null $validSince
|
||||||
* @param \DateTime|null $validUntil
|
* @param \DateTime|null $validUntil
|
||||||
|
* @param string|null $customSlug
|
||||||
* @return string
|
* @return string
|
||||||
|
* @throws NonUniqueSlugException
|
||||||
* @throws InvalidUrlException
|
* @throws InvalidUrlException
|
||||||
* @throws RuntimeException
|
* @throws RuntimeException
|
||||||
*/
|
*/
|
||||||
|
@ -26,7 +29,8 @@ interface UrlShortenerInterface
|
||||||
UriInterface $url,
|
UriInterface $url,
|
||||||
array $tags = [],
|
array $tags = [],
|
||||||
\DateTime $validSince = null,
|
\DateTime $validSince = null,
|
||||||
\DateTime $validUntil = null
|
\DateTime $validUntil = null,
|
||||||
|
string $customSlug = null
|
||||||
): string;
|
): string;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -3,6 +3,7 @@ declare(strict_types=1);
|
||||||
|
|
||||||
namespace ShlinkioTest\Shlink\Core\Service;
|
namespace ShlinkioTest\Shlink\Core\Service;
|
||||||
|
|
||||||
|
use Cocur\Slugify\SlugifyInterface;
|
||||||
use Doctrine\Common\Cache\ArrayCache;
|
use Doctrine\Common\Cache\ArrayCache;
|
||||||
use Doctrine\Common\Cache\Cache;
|
use Doctrine\Common\Cache\Cache;
|
||||||
use Doctrine\Common\Persistence\ObjectRepository;
|
use Doctrine\Common\Persistence\ObjectRepository;
|
||||||
|
@ -14,8 +15,10 @@ use GuzzleHttp\Exception\ClientException;
|
||||||
use GuzzleHttp\Psr7\Request;
|
use GuzzleHttp\Psr7\Request;
|
||||||
use PHPUnit\Framework\TestCase;
|
use PHPUnit\Framework\TestCase;
|
||||||
use Prophecy\Argument;
|
use Prophecy\Argument;
|
||||||
|
use Prophecy\Prophecy\MethodProphecy;
|
||||||
use Prophecy\Prophecy\ObjectProphecy;
|
use Prophecy\Prophecy\ObjectProphecy;
|
||||||
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
use Shlinkio\Shlink\Core\Entity\ShortUrl;
|
||||||
|
use Shlinkio\Shlink\Core\Exception\NonUniqueSlugException;
|
||||||
use Shlinkio\Shlink\Core\Repository\ShortUrlRepositoryInterface;
|
use Shlinkio\Shlink\Core\Repository\ShortUrlRepositoryInterface;
|
||||||
use Shlinkio\Shlink\Core\Service\UrlShortener;
|
use Shlinkio\Shlink\Core\Service\UrlShortener;
|
||||||
use Zend\Diactoros\Uri;
|
use Zend\Diactoros\Uri;
|
||||||
|
@ -38,6 +41,10 @@ class UrlShortenerTest extends TestCase
|
||||||
* @var Cache
|
* @var Cache
|
||||||
*/
|
*/
|
||||||
protected $cache;
|
protected $cache;
|
||||||
|
/**
|
||||||
|
* @var ObjectProphecy
|
||||||
|
*/
|
||||||
|
protected $slugger;
|
||||||
|
|
||||||
public function setUp()
|
public function setUp()
|
||||||
{
|
{
|
||||||
|
@ -60,8 +67,15 @@ class UrlShortenerTest extends TestCase
|
||||||
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
|
$this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
|
||||||
|
|
||||||
$this->cache = new ArrayCache();
|
$this->cache = new ArrayCache();
|
||||||
|
$this->slugger = $this->prophesize(SlugifyInterface::class);
|
||||||
|
|
||||||
$this->urlShortener = new UrlShortener($this->httpClient->reveal(), $this->em->reveal(), $this->cache);
|
$this->urlShortener = new UrlShortener(
|
||||||
|
$this->httpClient->reveal(),
|
||||||
|
$this->em->reveal(),
|
||||||
|
$this->cache,
|
||||||
|
UrlShortener::DEFAULT_CHARS,
|
||||||
|
$this->slugger->reveal()
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -117,6 +131,54 @@ class UrlShortenerTest extends TestCase
|
||||||
$this->assertEquals($shortUrl->getShortCode(), $shortCode);
|
$this->assertEquals($shortUrl->getShortCode(), $shortCode);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
*/
|
||||||
|
public function whenCustomSlugIsProvidedItIsUsed()
|
||||||
|
{
|
||||||
|
/** @var MethodProphecy $slugify */
|
||||||
|
$slugify = $this->slugger->slugify('custom-slug')->willReturnArgument();
|
||||||
|
|
||||||
|
$this->urlShortener->urlToShortCode(
|
||||||
|
new Uri('http://foobar.com/12345/hello?foo=bar'),
|
||||||
|
[],
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
'custom-slug'
|
||||||
|
);
|
||||||
|
|
||||||
|
$slugify->shouldHaveBeenCalledTimes(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @test
|
||||||
|
*/
|
||||||
|
public function exceptionIsThrownWhenNonUniqueSlugIsProvided()
|
||||||
|
{
|
||||||
|
/** @var MethodProphecy $slugify */
|
||||||
|
$slugify = $this->slugger->slugify('custom-slug')->willReturnArgument();
|
||||||
|
|
||||||
|
$repo = $this->prophesize(ShortUrlRepositoryInterface::class);
|
||||||
|
/** @var MethodProphecy $findBySlug */
|
||||||
|
$findBySlug = $repo->findOneBy(['shortCode' => 'custom-slug'])->willReturn(new ShortUrl());
|
||||||
|
$repo->findOneBy(Argument::cetera())->willReturn(null);
|
||||||
|
/** @var MethodProphecy $getRepo */
|
||||||
|
$getRepo = $this->em->getRepository(ShortUrl::class)->willReturn($repo->reveal());
|
||||||
|
|
||||||
|
$slugify->shouldBeCalledTimes(1);
|
||||||
|
$findBySlug->shouldBeCalledTimes(1);
|
||||||
|
$getRepo->shouldBeCalled();
|
||||||
|
$this->expectException(NonUniqueSlugException::class);
|
||||||
|
|
||||||
|
$this->urlShortener->urlToShortCode(
|
||||||
|
new Uri('http://foobar.com/12345/hello?foo=bar'),
|
||||||
|
[],
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
'custom-slug'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @test
|
* @test
|
||||||
*/
|
*/
|
||||||
|
|
Loading…
Reference in a new issue