123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- <?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\Asset;
- use Symfony\Component\Asset\Context\ContextInterface;
- use Symfony\Component\Asset\Exception\InvalidArgumentException;
- use Symfony\Component\Asset\Exception\LogicException;
- use Symfony\Component\Asset\VersionStrategy\VersionStrategyInterface;
- /**
- * Package that adds a base URL to asset URLs in addition to a version.
- *
- * The package allows to use more than one base URLs in which case
- * it randomly chooses one for each asset; it also guarantees that
- * any given path will always use the same base URL to be nice with
- * HTTP caching mechanisms.
- *
- * When the request context is available, this package can choose the
- * best base URL to use based on the current request scheme:
- *
- * * For HTTP request, it chooses between all base URLs;
- * * For HTTPs requests, it chooses between HTTPs base URLs and relative protocol URLs
- * or falls back to any base URL if no secure ones are available.
- *
- * @author Fabien Potencier <fabien@symfony.com>
- */
- class UrlPackage extends Package
- {
- private $baseUrls = [];
- private $sslPackage;
- /**
- * @param string|string[] $baseUrls Base asset URLs
- */
- public function __construct($baseUrls, VersionStrategyInterface $versionStrategy, ContextInterface $context = null)
- {
- parent::__construct($versionStrategy, $context);
- if (!\is_array($baseUrls)) {
- $baseUrls = (array) $baseUrls;
- }
- if (!$baseUrls) {
- throw new LogicException('You must provide at least one base URL.');
- }
- foreach ($baseUrls as $baseUrl) {
- $this->baseUrls[] = rtrim($baseUrl, '/');
- }
- $sslUrls = $this->getSslUrls($baseUrls);
- if ($sslUrls && $baseUrls !== $sslUrls) {
- $this->sslPackage = new self($sslUrls, $versionStrategy);
- }
- }
- /**
- * {@inheritdoc}
- */
- public function getUrl(string $path)
- {
- if ($this->isAbsoluteUrl($path)) {
- return $path;
- }
- if (null !== $this->sslPackage && $this->getContext()->isSecure()) {
- return $this->sslPackage->getUrl($path);
- }
- $url = $this->getVersionStrategy()->applyVersion($path);
- if ($this->isAbsoluteUrl($url)) {
- return $url;
- }
- if ($url && '/' != $url[0]) {
- $url = '/'.$url;
- }
- return $this->getBaseUrl($path).$url;
- }
- /**
- * Returns the base URL for a path.
- *
- * @return string The base URL
- */
- public function getBaseUrl(string $path)
- {
- if (1 === \count($this->baseUrls)) {
- return $this->baseUrls[0];
- }
- return $this->baseUrls[$this->chooseBaseUrl($path)];
- }
- /**
- * Determines which base URL to use for the given path.
- *
- * Override this method to change the default distribution strategy.
- * This method should always return the same base URL index for a given path.
- *
- * @return int The base URL index for the given path
- */
- protected function chooseBaseUrl(string $path)
- {
- return (int) fmod(hexdec(substr(hash('sha256', $path), 0, 10)), \count($this->baseUrls));
- }
- private function getSslUrls(array $urls)
- {
- $sslUrls = [];
- foreach ($urls as $url) {
- if ('https://' === substr($url, 0, 8) || '//' === substr($url, 0, 2)) {
- $sslUrls[] = $url;
- } elseif (null === parse_url($url, \PHP_URL_SCHEME)) {
- throw new InvalidArgumentException(sprintf('"%s" is not a valid URL.', $url));
- }
- }
- return $sslUrls;
- }
- }
|