mirror of
https://github.com/shlinkio/shlink.git
synced 2026-03-06 07:13:11 +08:00
Centralized how routes are configured to support multi-segment slugs
This commit is contained in:
30
module/Core/src/Config/MultiSegmentSlugProcessor.php
Normal file
30
module/Core/src/Config/MultiSegmentSlugProcessor.php
Normal file
@@ -0,0 +1,30 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Shlinkio\Shlink\Core\Config;
|
||||
|
||||
use function Functional\map;
|
||||
use function str_replace;
|
||||
|
||||
class MultiSegmentSlugProcessor
|
||||
{
|
||||
private const SINGLE_SHORT_CODE_PATTERN = '{shortCode}';
|
||||
private const MULTI_SHORT_CODE_PATTERN = '{shortCode:.+}';
|
||||
|
||||
public function __invoke(array $config): array
|
||||
{
|
||||
$multiSegmentEnabled = $config['url_shortener']['multi_segment_slugs_enabled'] ?? false;
|
||||
if (! $multiSegmentEnabled) {
|
||||
return $config;
|
||||
}
|
||||
|
||||
$config['routes'] = map($config['routes'] ?? [], static function (array $route): array {
|
||||
['path' => $path] = $route;
|
||||
$route['path'] = str_replace(self::SINGLE_SHORT_CODE_PATTERN, self::MULTI_SHORT_CODE_PATTERN, $path);
|
||||
return $route;
|
||||
});
|
||||
|
||||
return $config;
|
||||
}
|
||||
}
|
||||
@@ -84,13 +84,13 @@ class ShortUrlRepository extends EntitySpecificationRepository implements ShortU
|
||||
->where('1=1');
|
||||
|
||||
$dateRange = $filtering->dateRange();
|
||||
if ($dateRange?->startDate() !== null) {
|
||||
if ($dateRange?->startDate !== null) {
|
||||
$qb->andWhere($qb->expr()->gte('s.dateCreated', ':startDate'));
|
||||
$qb->setParameter('startDate', $dateRange->startDate(), ChronosDateTimeType::CHRONOS_DATETIME);
|
||||
$qb->setParameter('startDate', $dateRange->startDate, ChronosDateTimeType::CHRONOS_DATETIME);
|
||||
}
|
||||
if ($dateRange?->endDate() !== null) {
|
||||
if ($dateRange?->endDate !== null) {
|
||||
$qb->andWhere($qb->expr()->lte('s.dateCreated', ':endDate'));
|
||||
$qb->setParameter('endDate', $dateRange->endDate(), ChronosDateTimeType::CHRONOS_DATETIME);
|
||||
$qb->setParameter('endDate', $dateRange->endDate, ChronosDateTimeType::CHRONOS_DATETIME);
|
||||
}
|
||||
|
||||
$searchTerm = $filtering->searchTerm();
|
||||
|
||||
@@ -245,11 +245,11 @@ class VisitRepository extends EntitySpecificationRepository implements VisitRepo
|
||||
{
|
||||
$conn = $this->getEntityManager()->getConnection();
|
||||
|
||||
if ($dateRange?->startDate() !== null) {
|
||||
$qb->andWhere($qb->expr()->gte('v.date', $conn->quote($dateRange->startDate()->toDateTimeString())));
|
||||
if ($dateRange?->startDate !== null) {
|
||||
$qb->andWhere($qb->expr()->gte('v.date', $conn->quote($dateRange->startDate->toDateTimeString())));
|
||||
}
|
||||
if ($dateRange?->endDate() !== null) {
|
||||
$qb->andWhere($qb->expr()->lte('v.date', $conn->quote($dateRange->endDate()->toDateTimeString())));
|
||||
if ($dateRange?->endDate !== null) {
|
||||
$qb->andWhere($qb->expr()->lte('v.date', $conn->quote($dateRange->endDate->toDateTimeString())));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -20,12 +20,12 @@ class InDateRange extends BaseSpecification
|
||||
{
|
||||
$criteria = [];
|
||||
|
||||
if ($this->dateRange?->startDate() !== null) {
|
||||
$criteria[] = Spec::gte($this->field, $this->dateRange->startDate()->toDateTimeString());
|
||||
if ($this->dateRange?->startDate !== null) {
|
||||
$criteria[] = Spec::gte($this->field, $this->dateRange->startDate->toDateTimeString());
|
||||
}
|
||||
|
||||
if ($this->dateRange?->endDate() !== null) {
|
||||
$criteria[] = Spec::lte($this->field, $this->dateRange->endDate()->toDateTimeString());
|
||||
if ($this->dateRange?->endDate !== null) {
|
||||
$criteria[] = Spec::lte($this->field, $this->dateRange->endDate->toDateTimeString());
|
||||
}
|
||||
|
||||
return Spec::andX(...$criteria);
|
||||
|
||||
60
module/Core/test/Config/MultiSegmentSlugProcessorTest.php
Normal file
60
module/Core/test/Config/MultiSegmentSlugProcessorTest.php
Normal file
@@ -0,0 +1,60 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace ShlinkioTest\Shlink\Core\Config;
|
||||
|
||||
use PHPUnit\Framework\TestCase;
|
||||
use Shlinkio\Shlink\Core\Config\MultiSegmentSlugProcessor;
|
||||
|
||||
class MultiSegmentSlugProcessorTest extends TestCase
|
||||
{
|
||||
private MultiSegmentSlugProcessor $processor;
|
||||
|
||||
protected function setUp(): void
|
||||
{
|
||||
$this->processor = new MultiSegmentSlugProcessor();
|
||||
}
|
||||
|
||||
/**
|
||||
* @test
|
||||
* @dataProvider provideConfigs
|
||||
*/
|
||||
public function parsesRoutesAsExpected(array $config, array $expectedRoutes): void
|
||||
{
|
||||
self::assertEquals($expectedRoutes, ($this->processor)($config)['routes'] ?? []);
|
||||
}
|
||||
|
||||
public function provideConfigs(): iterable
|
||||
{
|
||||
yield [[], []];
|
||||
yield [['url_shortener' => []], []];
|
||||
yield [['url_shortener' => ['multi_segment_slugs_enabled' => false]], []];
|
||||
yield [
|
||||
[
|
||||
'url_shortener' => ['multi_segment_slugs_enabled' => false],
|
||||
'routes' => $routes = [
|
||||
['path' => '/foo'],
|
||||
['path' => '/bar/{shortCode}'],
|
||||
['path' => '/baz/{shortCode}/foo'],
|
||||
],
|
||||
],
|
||||
$routes,
|
||||
];
|
||||
yield [
|
||||
[
|
||||
'url_shortener' => ['multi_segment_slugs_enabled' => true],
|
||||
'routes' => [
|
||||
['path' => '/foo'],
|
||||
['path' => '/bar/{shortCode}'],
|
||||
['path' => '/baz/{shortCode}/foo'],
|
||||
],
|
||||
],
|
||||
[
|
||||
['path' => '/foo'],
|
||||
['path' => '/bar/{shortCode:.+}'],
|
||||
['path' => '/baz/{shortCode:.+}/foo'],
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
||||
@@ -8,7 +8,6 @@ use function Functional\first;
|
||||
use function Functional\map;
|
||||
use function Shlinkio\Shlink\Config\loadConfigFromGlob;
|
||||
use function sprintf;
|
||||
use function str_replace;
|
||||
|
||||
class ConfigProvider
|
||||
{
|
||||
@@ -21,16 +20,12 @@ class ConfigProvider
|
||||
return loadConfigFromGlob(__DIR__ . '/../config/{,*.}config.php');
|
||||
}
|
||||
|
||||
public static function applyRoutesPrefix(array $routes, bool $multiSegmentEnabled): array
|
||||
public static function applyRoutesPrefix(array $routes): array
|
||||
{
|
||||
$healthRoute = self::buildUnversionedHealthRouteFromExistingRoutes($routes);
|
||||
$prefixedRoutes = map($routes, static function (array $route) use ($multiSegmentEnabled) {
|
||||
$prefixedRoutes = map($routes, static function (array $route) {
|
||||
['path' => $path] = $route;
|
||||
if ($multiSegmentEnabled) {
|
||||
$path = str_replace('{shortCode}', '{shortCode:.+}', $path);
|
||||
}
|
||||
$route['path'] = sprintf('%s%s', self::ROUTES_PREFIX, $path);
|
||||
|
||||
return $route;
|
||||
});
|
||||
|
||||
|
||||
@@ -33,9 +33,9 @@ class ConfigProviderTest extends TestCase
|
||||
* @test
|
||||
* @dataProvider provideRoutesConfig
|
||||
*/
|
||||
public function routesAreProperlyPrefixed(array $routes, bool $multiSegmentEnabled, array $expected): void
|
||||
public function routesAreProperlyPrefixed(array $routes, array $expected): void
|
||||
{
|
||||
self::assertEquals($expected, ConfigProvider::applyRoutesPrefix($routes, $multiSegmentEnabled));
|
||||
self::assertEquals($expected, ConfigProvider::applyRoutesPrefix($routes));
|
||||
}
|
||||
|
||||
public function provideRoutesConfig(): iterable
|
||||
@@ -47,7 +47,6 @@ class ConfigProviderTest extends TestCase
|
||||
['path' => '/baz/foo'],
|
||||
['path' => '/health'],
|
||||
],
|
||||
false,
|
||||
[
|
||||
['path' => '/rest/v{version:1|2}/foo'],
|
||||
['path' => '/rest/v{version:1|2}/bar'],
|
||||
@@ -62,25 +61,11 @@ class ConfigProviderTest extends TestCase
|
||||
['path' => '/bar'],
|
||||
['path' => '/baz/foo'],
|
||||
],
|
||||
false,
|
||||
[
|
||||
['path' => '/rest/v{version:1|2}/foo'],
|
||||
['path' => '/rest/v{version:1|2}/bar'],
|
||||
['path' => '/rest/v{version:1|2}/baz/foo'],
|
||||
],
|
||||
];
|
||||
yield 'multi-segment enabled' => [
|
||||
[
|
||||
['path' => '/foo'],
|
||||
['path' => '/bar/{shortCode}'],
|
||||
['path' => '/baz/{shortCode}/foo'],
|
||||
],
|
||||
true,
|
||||
[
|
||||
['path' => '/rest/v{version:1|2}/foo'],
|
||||
['path' => '/rest/v{version:1|2}/bar/{shortCode:.+}'],
|
||||
['path' => '/rest/v{version:1|2}/baz/{shortCode:.+}/foo'],
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user