Merge pull request #1496 from acelaya-forks/feature/centralize-multi-segment

Feature/centralize multi segment
This commit is contained in:
Alejandro Celaya 2022-08-06 09:54:09 +02:00 committed by GitHub
commit b9180be685
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
12 changed files with 131 additions and 46 deletions

View file

@ -4,6 +4,23 @@ All notable changes to this project will be documented in this file.
The format is based on [Keep a Changelog](https://keepachangelog.com), and this project adheres to [Semantic Versioning](https://semver.org). The format is based on [Keep a Changelog](https://keepachangelog.com), and this project adheres to [Semantic Versioning](https://semver.org).
## [Unreleased]
### Added
* *Nothing*
### Changed
* [#1495](https://github.com/shlinkio/shlink/issues/1495) Centralized how routes are configured to support multi-segment slugs.
### Deprecated
* *Nothing*
### Removed
* *Nothing*
### Fixed
* *Nothing*
## [3.2.0] - 2022-08-05 ## [3.2.0] - 2022-08-05
### Added ### Added
* [#854](https://github.com/shlinkio/shlink/issues/854) Added support for multi-segment custom slugs. * [#854](https://github.com/shlinkio/shlink/issues/854) Added support for multi-segment custom slugs.

View file

@ -43,7 +43,7 @@
"php-middleware/request-id": "^4.1", "php-middleware/request-id": "^4.1",
"pugx/shortid-php": "^1.0", "pugx/shortid-php": "^1.0",
"ramsey/uuid": "^4.3", "ramsey/uuid": "^4.3",
"shlinkio/shlink-common": "^4.5", "shlinkio/shlink-common": "dev-main#6163a03 as 5.0",
"shlinkio/shlink-config": "^1.6", "shlinkio/shlink-config": "^1.6",
"shlinkio/shlink-event-dispatcher": "^2.4", "shlinkio/shlink-event-dispatcher": "^2.4",
"shlinkio/shlink-importer": "^3.0", "shlinkio/shlink-importer": "^3.0",

View file

@ -7,23 +7,19 @@ namespace Shlinkio\Shlink;
use Fig\Http\Message\RequestMethodInterface; use Fig\Http\Message\RequestMethodInterface;
use RKA\Middleware\IpAddress; use RKA\Middleware\IpAddress;
use Shlinkio\Shlink\Core\Action as CoreAction; use Shlinkio\Shlink\Core\Action as CoreAction;
use Shlinkio\Shlink\Core\Config\EnvVars;
use Shlinkio\Shlink\Rest\Action; use Shlinkio\Shlink\Rest\Action;
use Shlinkio\Shlink\Rest\ConfigProvider; use Shlinkio\Shlink\Rest\ConfigProvider;
use Shlinkio\Shlink\Rest\Middleware; use Shlinkio\Shlink\Rest\Middleware;
use Shlinkio\Shlink\Rest\Middleware\Mercure\NotConfiguredMercureErrorHandler; use Shlinkio\Shlink\Rest\Middleware\Mercure\NotConfiguredMercureErrorHandler;
use function sprintf;
// The order of the routes defined here matters. Changing it might cause path conflicts
return (static function (): array { return (static function (): array {
$contentNegotiationMiddleware = Middleware\ShortUrl\CreateShortUrlContentNegotiationMiddleware::class; $contentNegotiationMiddleware = Middleware\ShortUrl\CreateShortUrlContentNegotiationMiddleware::class;
$dropDomainMiddleware = Middleware\ShortUrl\DropDefaultDomainFromRequestMiddleware::class; $dropDomainMiddleware = Middleware\ShortUrl\DropDefaultDomainFromRequestMiddleware::class;
$overrideDomainMiddleware = Middleware\ShortUrl\OverrideDomainMiddleware::class; $overrideDomainMiddleware = Middleware\ShortUrl\OverrideDomainMiddleware::class;
$multiSegment = (bool) EnvVars::MULTI_SEGMENT_SLUGS_ENABLED->loadFromEnv(false);
return [ return [
// The order of the routes defined here matters. Changing it might cause path conflicts
'routes' => [ 'routes' => [
// Rest // Rest
...ConfigProvider::applyRoutesPrefix([ ...ConfigProvider::applyRoutesPrefix([
@ -64,7 +60,7 @@ return (static function (): array {
Action\Domain\DomainRedirectsAction::getRouteDef(), Action\Domain\DomainRedirectsAction::getRouteDef(),
Action\MercureInfoAction::getRouteDef([NotConfiguredMercureErrorHandler::class]), Action\MercureInfoAction::getRouteDef([NotConfiguredMercureErrorHandler::class]),
], $multiSegment), ]),
// Non-rest // Non-rest
[ [
@ -77,7 +73,7 @@ return (static function (): array {
], ],
[ [
'name' => CoreAction\PixelAction::class, 'name' => CoreAction\PixelAction::class,
'path' => sprintf('/{shortCode%s}/track', $multiSegment ? ':.+' : ''), 'path' => '/{shortCode}/track',
'middleware' => [ 'middleware' => [
IpAddress::class, IpAddress::class,
CoreAction\PixelAction::class, CoreAction\PixelAction::class,
@ -86,7 +82,7 @@ return (static function (): array {
], ],
[ [
'name' => CoreAction\QrCodeAction::class, 'name' => CoreAction\QrCodeAction::class,
'path' => sprintf('/{shortCode%s}/qr-code', $multiSegment ? ':.+' : ''), 'path' => '/{shortCode}/qr-code',
'middleware' => [ 'middleware' => [
CoreAction\QrCodeAction::class, CoreAction\QrCodeAction::class,
], ],
@ -94,7 +90,7 @@ return (static function (): array {
], ],
[ [
'name' => CoreAction\RedirectAction::class, 'name' => CoreAction\RedirectAction::class,
'path' => sprintf('/{shortCode%s}', $multiSegment ? ':.+' : ''), 'path' => '/{shortCode}',
'middleware' => [ 'middleware' => [
IpAddress::class, IpAddress::class,
CoreAction\RedirectAction::class, CoreAction\RedirectAction::class,

View file

@ -12,6 +12,7 @@ return [
'hostname' => sprintf('localhost:%s', $isSwoole ? '8080' : '8000'), 'hostname' => sprintf('localhost:%s', $isSwoole ? '8080' : '8000'),
], ],
'auto_resolve_titles' => true, 'auto_resolve_titles' => true,
// 'multi_segment_slugs_enabled' => true,
], ],
]; ];

View file

@ -47,4 +47,5 @@ return (new ConfigAggregator\ConfigAggregator([
new ConfigAggregator\PhpFileProvider('config/autoload/routes.config.php'), new ConfigAggregator\PhpFileProvider('config/autoload/routes.config.php'),
], 'data/cache/app_config.php', [ ], 'data/cache/app_config.php', [
Core\Config\BasePathPrefixer::class, Core\Config\BasePathPrefixer::class,
Core\Config\MultiSegmentSlugProcessor::class,
]))->getMergedConfig(); ]))->getMergedConfig();

View 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;
}
}

View file

@ -84,13 +84,13 @@ class ShortUrlRepository extends EntitySpecificationRepository implements ShortU
->where('1=1'); ->where('1=1');
$dateRange = $filtering->dateRange(); $dateRange = $filtering->dateRange();
if ($dateRange?->startDate() !== null) { if ($dateRange?->startDate !== null) {
$qb->andWhere($qb->expr()->gte('s.dateCreated', ':startDate')); $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->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(); $searchTerm = $filtering->searchTerm();

View file

@ -245,11 +245,11 @@ class VisitRepository extends EntitySpecificationRepository implements VisitRepo
{ {
$conn = $this->getEntityManager()->getConnection(); $conn = $this->getEntityManager()->getConnection();
if ($dateRange?->startDate() !== null) { if ($dateRange?->startDate !== null) {
$qb->andWhere($qb->expr()->gte('v.date', $conn->quote($dateRange->startDate()->toDateTimeString()))); $qb->andWhere($qb->expr()->gte('v.date', $conn->quote($dateRange->startDate->toDateTimeString())));
} }
if ($dateRange?->endDate() !== null) { if ($dateRange?->endDate !== null) {
$qb->andWhere($qb->expr()->lte('v.date', $conn->quote($dateRange->endDate()->toDateTimeString()))); $qb->andWhere($qb->expr()->lte('v.date', $conn->quote($dateRange->endDate->toDateTimeString())));
} }
} }

View file

@ -20,12 +20,12 @@ class InDateRange extends BaseSpecification
{ {
$criteria = []; $criteria = [];
if ($this->dateRange?->startDate() !== null) { if ($this->dateRange?->startDate !== null) {
$criteria[] = Spec::gte($this->field, $this->dateRange->startDate()->toDateTimeString()); $criteria[] = Spec::gte($this->field, $this->dateRange->startDate->toDateTimeString());
} }
if ($this->dateRange?->endDate() !== null) { if ($this->dateRange?->endDate !== null) {
$criteria[] = Spec::lte($this->field, $this->dateRange->endDate()->toDateTimeString()); $criteria[] = Spec::lte($this->field, $this->dateRange->endDate->toDateTimeString());
} }
return Spec::andX(...$criteria); return Spec::andX(...$criteria);

View 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'],
],
];
}
}

View file

@ -8,7 +8,6 @@ use function Functional\first;
use function Functional\map; use function Functional\map;
use function Shlinkio\Shlink\Config\loadConfigFromGlob; use function Shlinkio\Shlink\Config\loadConfigFromGlob;
use function sprintf; use function sprintf;
use function str_replace;
class ConfigProvider class ConfigProvider
{ {
@ -21,16 +20,12 @@ class ConfigProvider
return loadConfigFromGlob(__DIR__ . '/../config/{,*.}config.php'); 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); $healthRoute = self::buildUnversionedHealthRouteFromExistingRoutes($routes);
$prefixedRoutes = map($routes, static function (array $route) use ($multiSegmentEnabled) { $prefixedRoutes = map($routes, static function (array $route) {
['path' => $path] = $route; ['path' => $path] = $route;
if ($multiSegmentEnabled) {
$path = str_replace('{shortCode}', '{shortCode:.+}', $path);
}
$route['path'] = sprintf('%s%s', self::ROUTES_PREFIX, $path); $route['path'] = sprintf('%s%s', self::ROUTES_PREFIX, $path);
return $route; return $route;
}); });

View file

@ -33,9 +33,9 @@ class ConfigProviderTest extends TestCase
* @test * @test
* @dataProvider provideRoutesConfig * @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 public function provideRoutesConfig(): iterable
@ -47,7 +47,6 @@ class ConfigProviderTest extends TestCase
['path' => '/baz/foo'], ['path' => '/baz/foo'],
['path' => '/health'], ['path' => '/health'],
], ],
false,
[ [
['path' => '/rest/v{version:1|2}/foo'], ['path' => '/rest/v{version:1|2}/foo'],
['path' => '/rest/v{version:1|2}/bar'], ['path' => '/rest/v{version:1|2}/bar'],
@ -62,25 +61,11 @@ class ConfigProviderTest extends TestCase
['path' => '/bar'], ['path' => '/bar'],
['path' => '/baz/foo'], ['path' => '/baz/foo'],
], ],
false,
[ [
['path' => '/rest/v{version:1|2}/foo'], ['path' => '/rest/v{version:1|2}/foo'],
['path' => '/rest/v{version:1|2}/bar'], ['path' => '/rest/v{version:1|2}/bar'],
['path' => '/rest/v{version:1|2}/baz/foo'], ['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'],
],
];
} }
} }