‪TYPO3CMS  ‪main
Site.php
Go to the documentation of this file.
1 <?php
2 
3 declare(strict_types=1);
4 
5 /*
6  * This file is part of the TYPO3 CMS project.
7  *
8  * It is free software; you can redistribute it and/or modify it under
9  * the terms of the GNU General Public License, either version 2
10  * of the License, or any later version.
11  *
12  * For the full copyright and license information, please read the
13  * LICENSE.txt file that was distributed with this source code.
14  *
15  * The TYPO3 project - inspiring people to share!
16  */
17 
19 
20 use Psr\Http\Message\UriInterface;
21 use Symfony\Component\ExpressionLanguage\SyntaxError;
35 
41 class ‪Site implements ‪SiteInterface
42 {
43  protected const ‪ERRORHANDLER_TYPE_PAGE = 'Page';
44  protected const ‪ERRORHANDLER_TYPE_FLUID = 'Fluid';
45  protected const ‪ERRORHANDLER_TYPE_PHP = 'PHP';
46 
50  protected ‪$identifier;
51 
55  protected ‪$base;
56 
60  protected ‪$rootPageId;
61 
66  protected ‪$configuration;
67 
71  protected ‪$languages;
72 
76  protected $errorHandlers;
77 
78  protected ‪SiteSettings $settings;
79 
83  public function __construct(string ‪$identifier, int ‪$rootPageId, array ‪$configuration, ‪SiteSettings $settings = null)
84  {
87  if ($settings === null) {
88  $settings = new ‪SiteSettings(‪$configuration['settings'] ?? []);
89  }
90  $this->‪settings = $settings;
91  // Merge settings back in configuration for backwards-compatibility
92  ‪$configuration['settings'] = $this->‪settings->getAll();
94  $configuration['languages'] = !empty(‪$configuration['languages']) ? ‪$configuration['languages'] : [
95  0 => [
96  'languageId' => 0,
97  'title' => 'Default',
98  'navigationTitle' => '',
99  'flag' => 'us',
100  'locale' => 'en_US.UTF-8',
101  ],
102  ];
104  $configuration['base'] ?? '',
105  ‪$configuration['baseVariants'] ?? null
106  );
107  $this->‪base = new ‪Uri($this->‪sanitizeBaseUrl(‪$baseUrl));
108 
109  foreach ($configuration['languages'] as $languageConfiguration) {
110  $languageUid = (int)$languageConfiguration['languageId'];
111  // site language has defined its own base, this is the case most of the time.
112  if (!empty($languageConfiguration['base'])) {
114  $languageConfiguration['base'],
115  $languageConfiguration['baseVariants'] ?? null
116  );
117  ‪$base = new ‪Uri($this->‪sanitizeBaseUrl($base));
118  // no host given by the language-specific base, so lets prefix the main site base
119  if (‪$base->getScheme() === '' && ‪$base->getHost() === '') {
120  ‪$base = rtrim((string)$this->‪base, '/') . '/' . ltrim((string)‪$base, '/');
121  ‪$base = new ‪Uri($this->‪sanitizeBaseUrl($base));
122  }
123  } else {
124  // Language configuration does not have a base defined
125  // So the main site base is used (usually done for default languages)
126  ‪$base = new Uri($this->‪sanitizeBaseUrl(rtrim((string)$this->‪base, '/') . '/'));
127  }
128  if (!empty($languageConfiguration['flag'])) {
129  if ($languageConfiguration['flag'] === 'global') {
130  $languageConfiguration['flag'] = 'flags-multiple';
131  } elseif ($languageConfiguration['flag'] !== 'empty-empty') {
132  $languageConfiguration['flag'] = 'flags-' . $languageConfiguration['flag'];
133  }
134  }
135  $this->‪languages[$languageUid] = new SiteLanguage(
136  $languageUid,
137  $languageConfiguration['locale'],
138  ‪$base,
139  $languageConfiguration
140  );
141  }
142  foreach (‪$configuration['errorHandling'] ?? [] as $errorHandlingConfiguration) {
143  $code = $errorHandlingConfiguration['errorCode'];
144  unset($errorHandlingConfiguration['errorCode']);
145  $this->errorHandlers[(int)$code] = $errorHandlingConfiguration;
146  }
147  }
148 
152  protected function ‪resolveBaseWithVariants(string ‪$baseUrl, ?array $baseVariants): string
153  {
154  if (!empty($baseVariants)) {
155  $expressionLanguageResolver = GeneralUtility::makeInstance(
156  Resolver::class,
157  'site',
158  []
159  );
160  foreach ($baseVariants as $baseVariant) {
161  try {
162  if ((bool)$expressionLanguageResolver->evaluate($baseVariant['condition'])) {
163  ‪$baseUrl = $baseVariant['base'];
164  break;
165  }
166  } catch (SyntaxError $e) {
167  // silently fail and do not evaluate
168  // no logger here, as Site is currently cached and serialized
169  }
170  }
171  }
172  return ‪$baseUrl;
173  }
174 
179  public function ‪getIdentifier(): string
180  {
182  }
183 
187  public function ‪getBase(): UriInterface
188  {
190  }
191 
195  public function ‪getRootPageId(): int
196  {
197  return ‪$this->rootPageId;
198  }
199 
205  public function ‪getLanguages(): array
206  {
207  ‪$languages = [];
208  foreach ($this->‪languages as $languageId => $language) {
209  if ($language->enabled()) {
210  ‪$languages[$languageId] = $language;
211  }
212  }
213  return ‪$languages;
214  }
215 
221  public function getAllLanguages(): array
222  {
223  return ‪$this->languages;
224  }
225 
231  public function ‪getLanguageById(int $languageId): SiteLanguage
232  {
233  if (isset($this->‪languages[$languageId])) {
234  return $this->‪languages[$languageId];
235  }
236  throw new \InvalidArgumentException(
237  'Language ' . $languageId . ' does not exist on site ' . $this->‪identifier . '.',
238  1522960188
239  );
240  }
241 
242  public function ‪getDefaultLanguage(): ‪SiteLanguage
243  {
244  return reset($this->‪languages);
245  }
246 
250  public function ‪getAvailableLanguages(BackendUserAuthentication $user, bool $includeAllLanguagesFlag = false, int $pageId = null): array
251  {
253 
254  // Check if we need to add language "-1"
255  if ($includeAllLanguagesFlag && $user->checkLanguageAccess(-1)) {
256  ‪$availableLanguages[-1] = new ‪SiteLanguage(-1, '', $this->‪getBase(), [
257  'title' => $this->‪getLanguageService()->sL('LLL:EXT:core/Resources/Private/Language/locallang_mod_web_list.xlf:multipleLanguages'),
258  'flag' => 'flags-multiple',
259  ]);
260  }
261 
262  // Do not add the ones that are not allowed by the user
263  foreach ($this->‪languages as $language) {
264  if ($user->checkLanguageAccess($language->getLanguageId())) {
265  ‪$availableLanguages[$language->getLanguageId()] = $language;
266  }
267  }
268 
269  return ‪$availableLanguages;
270  }
271 
278  public function ‪getErrorHandler(int ‪$statusCode): PageErrorHandlerInterface
279  {
280  $errorHandlerConfiguration = $this->errorHandlers[‪$statusCode] ?? $this->errorHandlers[0] ?? null;
281  switch ($errorHandlerConfiguration['errorHandler'] ?? null) {
283  return GeneralUtility::makeInstance(FluidPageErrorHandler::class, ‪$statusCode, $errorHandlerConfiguration);
285  return GeneralUtility::makeInstance(PageContentErrorHandler::class, ‪$statusCode, $errorHandlerConfiguration);
287  $handler = GeneralUtility::makeInstance($errorHandlerConfiguration['errorPhpClassFQCN'], ‪$statusCode, $errorHandlerConfiguration);
288  // Check if the interface is implemented
289  if (!($handler instanceof PageErrorHandlerInterface)) {
290  throw new InvalidPageErrorHandlerException('The configured error handler "' . (string)$errorHandlerConfiguration['errorPhpClassFQCN'] . '" for status code ' . ‪$statusCode . ' must implement the PageErrorHandlerInterface.', 1527432330);
291  }
292  return $handler;
293  }
294  throw new ‪PageErrorHandlerNotConfiguredException('No error handler given for the status code "' . ‪$statusCode . '".', 1522495914);
295  }
296 
300  public function ‪getConfiguration(): array
301  {
303  }
304 
305  public function ‪getSettings(): SiteSettings
306  {
307  return $this->settings;
308  }
309 
316  public function ‪getAttribute(string $attributeName)
317  {
318  if (isset($this->‪configuration[$attributeName])) {
319  return $this->‪configuration[$attributeName];
320  }
321  throw new \InvalidArgumentException(
322  'Attribute ' . $attributeName . ' does not exist on site ' . $this->‪identifier . '.',
323  1522495954
324  );
325  }
326 
331  protected function ‪sanitizeBaseUrl(string $base): string
332  {
333  // no protocol ("//") and the first part is no "/" (path), means that this is a domain like
334  // "www.domain.com/subpage", and we want to ensure that this one then gets a "no-scheme agnostic" part
335  if (!empty(‪$base) && !str_contains(‪$base, '//') && ‪$base[0] !== '/') {
336  // either a scheme is added, or no scheme but with domain, or a path which is not absolute
337  // make the base prefixed with a slash, so it is recognized as path, not as domain
338  // treat as path
339  if (!str_contains(‪$base, '.')) {
340  ‪$base = '/' . ‪$base;
341  } else {
342  // treat as domain name
343  ‪$base = '//' . ‪$base;
344  }
345  }
346  return ‪$base;
347  }
348 
354  public function ‪getRouter(‪Context $context = null): ‪RouterInterface
355  {
356  return GeneralUtility::makeInstance(PageRouter::class, $this, $context);
357  }
358 
359  protected function ‪getLanguageService(): ‪LanguageService
360  {
361  return ‪$GLOBALS['LANG'];
362  }
363 }
‪TYPO3\CMS\Core\Site\Entity\Site\getLanguageService
‪getLanguageService()
Definition: Site.php:353
‪TYPO3\CMS\Core\Site\Entity\SiteInterface
Definition: SiteInterface.php:26
‪TYPO3\CMS\Core\Routing\RouterInterface
Definition: RouterInterface.php:28
‪TYPO3\CMS\Core\Site\Entity\Site\base
‪$this base
Definition: Site.php:101
‪TYPO3\CMS\Core\Site\Entity\Site\identifier
‪array< LanguageRef, $languages;protected array $errorHandlers;protected SiteSettings $settings;public function __construct(string $identifier, int $rootPageId, array $configuration, SiteSettings $settings=null) { $this-> identifier
Definition: Site.php:79
‪$languages
‪$languages
Definition: updateIsoDatabase.php:104
‪TYPO3\CMS\Core\Site\Entity\Site\getAttribute
‪mixed getAttribute(string $attributeName)
Definition: Site.php:310
‪TYPO3\CMS\Core\Error\PageErrorHandler\FluidPageErrorHandler
Definition: FluidPageErrorHandler.php:32
‪TYPO3\CMS\Core\Routing\PageRouter
Definition: PageRouter.php:70
‪TYPO3\CMS\Core\Site\Entity\Site\ERRORHANDLER_TYPE_PAGE
‪const ERRORHANDLER_TYPE_PAGE
Definition: Site.php:43
‪TYPO3\CMS\Core\Error\PageErrorHandler\PageErrorHandlerNotConfiguredException
Definition: PageErrorHandlerNotConfiguredException.php:26
‪TYPO3\CMS\Core\Site\Entity\Site\resolveBaseWithVariants
‪foreach($configuration['languages'] as $languageConfiguration) foreach($configuration['errorHandling'] ??[] as $errorHandlingConfiguration) resolveBaseWithVariants(string $baseUrl, ?array $baseVariants)
Definition: Site.php:146
‪TYPO3\CMS\Core\Context\Context
Definition: Context.php:54
‪TYPO3\CMS\Core\Site\Entity\Site\languages
‪array< LanguageRef, function getLanguages():array { $languages=[];foreach( $this->languages as $languageId=> $language) { if( $language->enabled()) { $languages[ $languageId]=$language;} } return $languages;} public array< LanguageRef, function getAllLanguages():array { return $this->languages;} public function getLanguageById(int $languageId):SiteLanguage { if(isset( $this->languages[ $languageId])) { return $this-> languages[$languageId]
Definition: Site.php:228
‪TYPO3\CMS\Core\Site\Entity\Site\configuration
‪$this configuration
Definition: Site.php:87
‪TYPO3\CMS\Core\Site\Entity\Site\settings
‪if($settings===null) $this settings
Definition: Site.php:84
‪TYPO3\CMS\Core\Http\Uri
Definition: Uri.php:30
‪TYPO3\CMS\Core\Error\PageErrorHandler\InvalidPageErrorHandlerException
Definition: InvalidPageErrorHandlerException.php:26
‪TYPO3\CMS\Core\Site\Entity\Site
Definition: Site.php:42
‪TYPO3\CMS\Core\Site\Entity\Site\getErrorHandler
‪getErrorHandler(int $statusCode)
Definition: Site.php:272
‪TYPO3\CMS\Core\Site\Entity\SiteLanguage
Definition: SiteLanguage.php:27
‪TYPO3\CMS\Core\Site\Entity\Site\getRootPageId
‪getRootPageId()
Definition: Site.php:189
‪TYPO3\CMS\Core\Site\Entity\Site\$identifier
‪string $identifier
Definition: Site.php:49
‪TYPO3\CMS\Redirects\Message\$statusCode
‪identifier readonly UriInterface readonly int $statusCode
Definition: RedirectWasHitMessage.php:34
‪TYPO3\CMS\Core\Site\Entity\Site\$rootPageId
‪int $rootPageId
Definition: Site.php:57
‪TYPO3\CMS\Core\Site\Entity\Site\ERRORHANDLER_TYPE_FLUID
‪const ERRORHANDLER_TYPE_FLUID
Definition: Site.php:44
‪TYPO3\CMS\Core\Site\Entity\Site\ERRORHANDLER_TYPE_PHP
‪const ERRORHANDLER_TYPE_PHP
Definition: Site.php:45
‪TYPO3\CMS\Core\Error\PageErrorHandler\PageContentErrorHandler
Definition: PageContentErrorHandler.php:39
‪TYPO3\CMS\Core\Authentication\BackendUserAuthentication\checkLanguageAccess
‪bool checkLanguageAccess($langValue)
Definition: BackendUserAuthentication.php:558
‪TYPO3\CMS\Core\Authentication\BackendUserAuthentication
Definition: BackendUserAuthentication.php:60
‪TYPO3\CMS\Core\Error\PageErrorHandler\PageErrorHandlerInterface
Definition: PageErrorHandlerInterface.php:29
‪TYPO3\CMS\Core\Site\Entity
Definition: NullSite.php:18
‪TYPO3\CMS\Core\Site\Entity\Site\$configuration
‪array $configuration
Definition: Site.php:62
‪TYPO3\CMS\Core\Site\Entity\Site\getConfiguration
‪getConfiguration()
Definition: Site.php:294
‪TYPO3\CMS\Core\ExpressionLanguage\Resolver
Definition: Resolver.php:32
‪$GLOBALS
‪$GLOBALS['TYPO3_CONF_VARS']['EXTCONF']['adminpanel']['modules']
Definition: ext_localconf.php:25
‪TYPO3\CMS\Core\Site\Entity\SiteInterface\getLanguages
‪SiteLanguage[] getLanguages()
‪TYPO3\CMS\Core\Site\Entity\Site\getDefaultLanguage
‪getDefaultLanguage()
Definition: Site.php:236
‪TYPO3\CMS\Core\Localization\LanguageService
Definition: LanguageService.php:46
‪TYPO3\CMS\Core\Site\Entity\Site\$availableLanguages
‪return $availableLanguages
Definition: Site.php:263
‪TYPO3\CMS\Core\Site\Entity\Site\$baseUrl
‪$baseUrl
Definition: Site.php:97
‪TYPO3\CMS\Core\Site\Entity\SiteInterface\getAvailableLanguages
‪SiteLanguage[] getAvailableLanguages(BackendUserAuthentication $user, bool $includeAllLanguagesFlag=false, int $pageId=null)
‪TYPO3\CMS\Core\Utility\GeneralUtility
Definition: GeneralUtility.php:51
‪TYPO3\CMS\Core\Site\Entity\Site\rootPageId
‪$this rootPageId
Definition: Site.php:80
‪TYPO3\CMS\Core\Site\Entity\SiteSettings
Definition: SiteSettings.php:28
‪TYPO3\CMS\Core\Site\Entity\Site\getSettings
‪getSettings()
Definition: Site.php:299
‪TYPO3\CMS\Core\Site\Entity\Site\$base
‪UriInterface $base
Definition: Site.php:53
‪TYPO3\CMS\Core\Site\Entity\Site\getIdentifier
‪getIdentifier()
Definition: Site.php:173
‪TYPO3\CMS\Core\Site\Entity\Site\getRouter
‪getRouter(Context $context=null)
Definition: Site.php:348
‪TYPO3\CMS\Core\Site\Entity\Site\getBase
‪getBase()
Definition: Site.php:181
‪TYPO3\CMS\Core\Site\Entity\Site\sanitizeBaseUrl
‪sanitizeBaseUrl(string $base)
Definition: Site.php:325
‪TYPO3\CMS\Core\Site\Entity\SiteInterface\getLanguageById
‪getLanguageById(int $languageId)