Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[HttpKernel] Add a class to generate fragment URLs
- Loading branch information
Showing
6 changed files
with
125 additions
and
53 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
85 changes: 85 additions & 0 deletions
85
src/Symfony/Component/HttpKernel/Fragment/FragmentUriGenerator.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,85 @@ | ||
<?php | ||
|
||
/* | ||
* This file is part of the Symfony package. | ||
* | ||
* (c) Fabien Potencier <fabien@symfony.com> | ||
* | ||
* For the full copyright and license information, please view the LICENSE | ||
* file that was distributed with this source code. | ||
*/ | ||
|
||
namespace Symfony\Component\HttpKernel\Fragment; | ||
|
||
use Symfony\Component\HttpFoundation\Request; | ||
use Symfony\Component\HttpKernel\UriSigner; | ||
|
||
/** | ||
* Generates a fragment URI. | ||
* | ||
* @author Kévin Dunglas <kevin@dunglas.fr> | ||
* @author Fabien Potencier <fabien@symfony.com> | ||
*/ | ||
final class FragmentUriGenerator implements FragmentUriGeneratorInterface | ||
{ | ||
private $fragmentPath; | ||
private $signer; | ||
|
||
public function __construct(string $fragmentPath, UriSigner $signer = null) | ||
{ | ||
$this->fragmentPath = $fragmentPath; | ||
$this->signer = $signer; | ||
} | ||
|
||
/** | ||
* {@inheritDoc} | ||
*/ | ||
public function generate($controller, Request $request, bool $absolute = false, bool $strict = true, bool $sign = true): string | ||
{ | ||
if ($sign && null === $this->signer) { | ||
throw new \LogicException('You must use a URI when using the ESI rendering strategy or set a URL signer.'); | ||
} | ||
|
||
if ($strict) { | ||
$this->checkNonScalar($controller->attributes); | ||
} | ||
|
||
// We need to forward the current _format and _locale values as we don't have | ||
// a proper routing pattern to do the job for us. | ||
// This makes things inconsistent if you switch from rendering a controller | ||
// to rendering a route if the route pattern does not contain the special | ||
// _format and _locale placeholders. | ||
if (!isset($controller->attributes['_format'])) { | ||
$controller->attributes['_format'] = $request->getRequestFormat(); | ||
} | ||
if (!isset($controller->attributes['_locale'])) { | ||
$controller->attributes['_locale'] = $request->getLocale(); | ||
} | ||
|
||
$controller->attributes['_controller'] = $controller->controller; | ||
$controller->query['_path'] = http_build_query($controller->attributes, '', '&'); | ||
$path = $this->fragmentPath.'?'.http_build_query($controller->query, '', '&'); | ||
|
||
// we need to sign the absolute URI, but want to return the path only. | ||
$fragmentUri = $sign || $absolute ? $request->getUriForPath($path) : $request->getBaseUrl().$path; | ||
|
||
if (!$sign) { | ||
return $fragmentUri; | ||
} | ||
|
||
$fragmentUri = $this->signer->sign($fragmentUri); | ||
|
||
return $absolute ? $fragmentUri : substr($fragmentUri, \strlen($request->getSchemeAndHttpHost())); | ||
} | ||
|
||
private function checkNonScalar(array $values): void | ||
{ | ||
foreach ($values as $key => $value) { | ||
if (\is_array($value)) { | ||
$this->checkNonScalar($value); | ||
} elseif (!is_scalar($value) && null !== $value) { | ||
throw new \LogicException(sprintf('Controller attributes cannot contain non-scalar/non-null values (value for key "%s" is not a scalar or null).', $key)); | ||
} | ||
} | ||
} | ||
} |
34 changes: 34 additions & 0 deletions
34
src/Symfony/Component/HttpKernel/Fragment/FragmentUriGeneratorInterface.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
<?php | ||
|
||
/* | ||
* This file is part of the Symfony package. | ||
* | ||
* (c) Fabien Potencier <fabien@symfony.com> | ||
* | ||
* For the full copyright and license information, please view the LICENSE | ||
* file that was distributed with this source code. | ||
*/ | ||
|
||
namespace Symfony\Component\HttpKernel\Fragment; | ||
|
||
use Symfony\Component\HttpFoundation\Request; | ||
use Symfony\Component\HttpKernel\Controller\ControllerReference; | ||
|
||
/** | ||
* Interface implemented by rendering strategies able to generate an URL for a fragment. | ||
* | ||
* @author Kévin Dunglas <kevin@dunglas.fr> | ||
*/ | ||
interface FragmentUriGeneratorInterface | ||
{ | ||
/** | ||
* Generates a fragment URI for a given controller. | ||
* | ||
* @param string|ControllerReference $controller The name of a controller as a string or a ControllerReference instance | ||
* @param bool $absolute Whether to generate an absolute URL or not | ||
* @param bool $strict Whether to allow non-scalar attributes or not | ||
* | ||
* @return string A fragment URI | ||
*/ | ||
public function generate($controller, Request $request, bool $absolute = false, bool $strict = true, bool $sign = true): string; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters