diff --git a/typo3/sysext/redirects/Classes/Http/Middleware/RedirectHandler.php b/typo3/sysext/redirects/Classes/Http/Middleware/RedirectHandler.php index e77043b8bc5b77c68dac3d4ce03156a4f6c668a1..87d595b69625d6ab7be71cdd115f355d761714bd 100644 --- a/typo3/sysext/redirects/Classes/Http/Middleware/RedirectHandler.php +++ b/typo3/sysext/redirects/Classes/Http/Middleware/RedirectHandler.php @@ -66,7 +66,7 @@ class RedirectHandler implements MiddlewareInterface, LoggerAwareInterface // If the matched redirect is found, resolve it, and check further if (is_array($matchedRedirect)) { - $url = $this->redirectService->getTargetUrl($matchedRedirect, $request->getQueryParams(), $request->getAttribute('frontend.user'), $request->getAttribute('site')); + $url = $this->redirectService->getTargetUrl($matchedRedirect, $request->getQueryParams(), $request->getAttribute('frontend.user'), $request->getUri(), $request->getAttribute('site')); if ($url instanceof UriInterface) { $this->logger->debug('Redirecting', ['record' => $matchedRedirect, 'uri' => $url]); $response = $this->buildRedirectResponse($url, $matchedRedirect); diff --git a/typo3/sysext/redirects/Classes/Service/RedirectService.php b/typo3/sysext/redirects/Classes/Service/RedirectService.php index 89fabbb4febb2e6f8826be5f699e7faed8113d2a..7a450343a57c293164db0aacd31dba37f6beaca1 100644 --- a/typo3/sysext/redirects/Classes/Service/RedirectService.php +++ b/typo3/sysext/redirects/Classes/Service/RedirectService.php @@ -193,10 +193,11 @@ class RedirectService implements LoggerAwareInterface * @param array $matchedRedirect * @param array $queryParams * @param FrontendUserAuthentication $frontendUserAuthentication + * @param UriInterface $uri * @param SiteInterface|null $site * @return UriInterface|null */ - public function getTargetUrl(array $matchedRedirect, array $queryParams, FrontendUserAuthentication $frontendUserAuthentication, ?SiteInterface $site = null): ?UriInterface + public function getTargetUrl(array $matchedRedirect, array $queryParams, FrontendUserAuthentication $frontendUserAuthentication, UriInterface $uri, ?SiteInterface $site = null): ?UriInterface { $this->logger->debug('Found a redirect to process', $matchedRedirect); $linkParameterParts = GeneralUtility::makeInstance(TypoLinkCodecService::class)->decode((string)$matchedRedirect['target']); @@ -211,6 +212,10 @@ class RedirectService implements LoggerAwareInterface } // Do this for files, folders, external URLs if (!empty($linkDetails['url'])) { + if ($matchedRedirect['is_regexp']) { + $linkDetails = $this->replaceRegExpCaptureGroup($matchedRedirect, $uri, $linkDetails); + } + $url = new Uri($linkDetails['url']); if ($matchedRedirect['force_https']) { $url = $url->withScheme('https'); @@ -337,4 +342,21 @@ class RedirectService implements LoggerAwareInterface } return $controller; } + + /** + * @param array $matchedRedirect + * @param UriInterface $uri + * @param array $linkDetails + * @return array + */ + protected function replaceRegExpCaptureGroup(array $matchedRedirect, UriInterface $uri, array $linkDetails): array + { + $matchResult = @preg_match($matchedRedirect['source_path'], $uri->getPath(), $matches); + if ($matchResult > 0) { + foreach ($matches as $key => $val) { + $linkDetails['url'] = str_replace('$' . $key, $val, $linkDetails['url']); + } + } + return $linkDetails; + } } diff --git a/typo3/sysext/redirects/Tests/Unit/Service/RedirectServiceTest.php b/typo3/sysext/redirects/Tests/Unit/Service/RedirectServiceTest.php index f50a2bdc33f2e545f15a2f35f43ceca4e775b83d..7b8c4be5f96115ba991737c9382acca33c3bc467 100644 --- a/typo3/sysext/redirects/Tests/Unit/Service/RedirectServiceTest.php +++ b/typo3/sysext/redirects/Tests/Unit/Service/RedirectServiceTest.php @@ -442,7 +442,7 @@ class RedirectServiceTest extends UnitTestCase { $this->linkServiceProphecy->resolve(Argument::any())->willThrow(new InvalidPathException('', 1516531195)); - $result = $this->redirectService->getTargetUrl(['target' => 'invalid'], [], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $result = $this->redirectService->getTargetUrl(['target' => 'invalid'], [], new FrontendUserAuthentication(), new Uri(), new Site('dummy', 13, [])); self::assertNull($result); } @@ -463,7 +463,8 @@ class RedirectServiceTest extends UnitTestCase ]; $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); - $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $source = new Uri('https://example.com'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); $uri = new Uri('https://example.com/'); self::assertEquals($uri, $result); @@ -487,7 +488,8 @@ class RedirectServiceTest extends UnitTestCase ]; $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); - $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $source = new Uri('https://example.com'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); $uri = new Uri('https://example.com/file.txt'); self::assertEquals($uri, $result); @@ -512,7 +514,8 @@ class RedirectServiceTest extends UnitTestCase ]; $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); - $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $source = new Uri('https://example.com/'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); $uri = new Uri('https://example.com/folder/'); self::assertEquals($uri, $result); @@ -534,7 +537,8 @@ class RedirectServiceTest extends UnitTestCase ]; $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); - $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $source = new Uri('https://example.com'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); $uri = new Uri('https://example.com'); self::assertEquals($uri, $result); @@ -556,7 +560,8 @@ class RedirectServiceTest extends UnitTestCase ]; $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); - $result = $this->redirectService->getTargetUrl($redirectTargetMatch, ['bar' => 3, 'baz' => 4], new FrontendUserAuthentication(), new Site('dummy', 13, [])); + $source = new Uri('https://example.com/?bar=2&baz=4&foo=1'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, ['bar' => 3, 'baz' => 4], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); $uri = new Uri('https://example.com/?bar=2&baz=4&foo=1'); self::assertEquals($uri, $result); @@ -601,8 +606,33 @@ class RedirectServiceTest extends UnitTestCase $redirectService->expects(self::once())->method('getUriFromCustomLinkDetails') ->with($redirectTargetMatch, $frontendUserAuthentication, $site, $linkDetails, $queryParams) ->willReturn($uri); - $result = $redirectService->getTargetUrl($redirectTargetMatch, [], $frontendUserAuthentication, $site); + $result = $redirectService->getTargetUrl($redirectTargetMatch, [], $frontendUserAuthentication, $uri, $site); self::assertEquals($uri, $result); } + + /** + * @test + */ + public function getTargetUrlReplaceRegExpCaptureGroup() + { + $redirectTargetMatch = [ + 'source_path' => '#^/foo/(.*)#', + 'target' => 'https://anotherdomain.com/$1', + 'force_https' => '0', + 'keep_query_parameters' => '1', + 'is_regexp' => 1 + ]; + $linkDetails = [ + 'type' => LinkService::TYPE_URL, + 'url' => 'https://anotherdomain.com/$1' + ]; + $this->linkServiceProphecy->resolve($redirectTargetMatch['target'])->willReturn($linkDetails); + + $source = new Uri('https://example.com/foo/bar'); + $result = $this->redirectService->getTargetUrl($redirectTargetMatch, [], new FrontendUserAuthentication(), $source, new Site('dummy', 13, [])); + + $uri = new Uri('https://anotherdomain.com/bar'); + self::assertEquals($uri, $result); + } }