From ca166d8c7a3a9c2060f0543e248c08f358fcac4d Mon Sep 17 00:00:00 2001
From: Benni Mack <benni@typo3.org>
Date: Sat, 13 Jul 2019 19:03:41 +0200
Subject: [PATCH] [TASK] Clean up TSFE constructor initialization

Due to e50b1c1acdd5da514a35f837d9b853692bcfa16d
the TypoScriptFrontendController requires four objects
as constructor arguments while keeping maximum
compatibility by fetching fallback information from the
current PSR-7 object or setting data from the legacy arguments.

The constructor is cleaned up with special initialize methods
for readability.

Resolves: #88747
Related: #88717
Releases: master
Change-Id: I734d1184780b8bfdf0e9638aca75c6a4fc0f7e2c
Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/61289
Tested-by: TYPO3com <noreply@typo3.com>
Tested-by: Susanne Moog <look@susi.dev>
Tested-by: Georg Ringer <georg.ringer@gmail.com>
Reviewed-by: Susanne Moog <look@susi.dev>
Reviewed-by: Oliver Klee <typo3-coding@oliverklee.de>
Reviewed-by: Georg Ringer <georg.ringer@gmail.com>
---
 ...ngedRequestWorkflowForFrontendRequests.rst |   2 +-
 .../TypoScriptFrontendController.php          | 183 ++++++++++++++----
 2 files changed, 148 insertions(+), 37 deletions(-)

diff --git a/typo3/sysext/core/Documentation/Changelog/master/Breaking-88540-ChangedRequestWorkflowForFrontendRequests.rst b/typo3/sysext/core/Documentation/Changelog/master/Breaking-88540-ChangedRequestWorkflowForFrontendRequests.rst
index 94abba36d190..111d1f15caec 100644
--- a/typo3/sysext/core/Documentation/Changelog/master/Breaking-88540-ChangedRequestWorkflowForFrontendRequests.rst
+++ b/typo3/sysext/core/Documentation/Changelog/master/Breaking-88540-ChangedRequestWorkflowForFrontendRequests.rst
@@ -55,7 +55,7 @@ The new request workflow looks like this (simplified):
 3. Resolving Site configuration and Language from URL if possible
 4. Resolving logged-in Backend User Authentication for previewing hidden pages or languages
 5. Authentication of Website Users ("Frontend Users")
-6. Executing various static routes and redirct functionality
+6. Executing various static routes and redirect functionality
 7. Resolving Target Page ID and URL parameters based on Routing, Validation of Page Arguments based on "cHash"
 8. Setting up global $TSFE object, injecting previously resolved settings into TSFE.
 9. Resolving the Rootline for the page
diff --git a/typo3/sysext/frontend/Classes/Controller/TypoScriptFrontendController.php b/typo3/sysext/frontend/Classes/Controller/TypoScriptFrontendController.php
index 4dcc4eb57ee8..e7a061ae50c2 100644
--- a/typo3/sysext/frontend/Classes/Controller/TypoScriptFrontendController.php
+++ b/typo3/sysext/frontend/Classes/Controller/TypoScriptFrontendController.php
@@ -660,20 +660,59 @@ class TypoScriptFrontendController implements LoggerAwareInterface
     protected $context;
 
     /**
-     * Class constructor
-     * Takes a number of GET/POST input variable as arguments and stores them internally.
-     * The processing of these variables goes on later in this class.
+     * Since TYPO3 v10.0, TSFE is composed out of
+     *  - Context
+     *  - Site
+     *  - SiteLanguage
+     *  - PageArguments (containing ID, Type, cHash and MP arguments)
+     *
+     * With TYPO3 v11, they will become mandatory and the method arguments will become strongly typed.
+     * For TYPO3 v10 this is built in a way to ensure maximum compatibility.
+     *
      * Also sets a unique string (->uniqueString) for this script instance; A md5 hash of the microtime()
      *
-     * @param Context|array $context the Context object to work on, previously defined to set TYPO3_CONF_VARS
+     * @param Context|array|null $context the Context object to work on, previously defined to set TYPO3_CONF_VARS
      * @param mixed|SiteInterface $siteOrId The resolved site to work on, previously this was the value of GeneralUtility::_GP('id')
-     * @param int|SiteLanguage $siteLanguageOrType The resolved language to work on, previously the value of GeneralUtility::_GP('type')
-     * @param bool|string|PageArguments $pageArguments The PageArguments object containing ID, type and GET parameters, previously unused or the value of GeneralUtility::_GP('no_cache')
-     * @param string|FrontendUserAuthentication $cHashOrFrontendUser FrontendUserAuthentication object, previously the value of GeneralUtility::_GP('cHash'), use the PageArguments object instead, will be removed in TYPO3 v11.0
-     * @param string $_2 previously was used to define the jumpURL, use the PageArguments object instead, will be removed in TYPO3 v11.0
-     * @param string $MP The value of GeneralUtility::_GP('MP'), use the PageArguments object instead, will be removed in TYPO3 v11.0
+     * @param SiteLanguage|int|string $siteLanguageOrType The resolved language to work on, previously the value of GeneralUtility::_GP('type')
+     * @param bool|string|PageArguments|null $pageArguments The PageArguments object containing ID, type and GET parameters, previously unused or the value of GeneralUtility::_GP('no_cache')
+     * @param string|FrontendUserAuthentication|null $cHashOrFrontendUser FrontendUserAuthentication object, previously the value of GeneralUtility::_GP('cHash'), use the PageArguments object instead, will be removed in TYPO3 v11.0
+     * @param string|null $_2 previously was used to define the jumpURL, use the PageArguments object instead, will be removed in TYPO3 v11.0
+     * @param string|null $MP The value of GeneralUtility::_GP('MP'), use the PageArguments object instead, will be removed in TYPO3 v11.0
      */
     public function __construct($context = null, $siteOrId = null, $siteLanguageOrType = null, $pageArguments = null, $cHashOrFrontendUser = null, $_2 = null, $MP = null)
+    {
+        $this->initializeContextWithGlobalFallback($context);
+
+        // Fetch the request for fetching data (site/language/pageArguments) for compatibility reasons, not needed
+        // in TYPO3 v11.0 anymore.
+        /** @var ServerRequestInterface $request */
+        $request = $GLOBALS['TYPO3_REQUEST'] ?? ServerRequestFactory::fromGlobals();
+
+        $this->initializeSiteWithCompatibility($siteOrId, $request);
+        $this->initializeSiteLanguageWithCompatibility($siteLanguageOrType, $request);
+        $pageArguments = $this->buildPageArgumentsWithFallback($pageArguments, $request);
+        $pageArguments = $this->initializeFrontendUserOrUpdateCHashArgument($cHashOrFrontendUser, $pageArguments);
+        $pageArguments = $this->initializeLegacyMountPointArgument($MP, $pageArguments);
+
+        $this->setPageArguments($pageArguments);
+
+        $this->domainStartPage = $this->site->getRootPageId();
+        $this->uniqueString = md5(microtime());
+        $this->initPageRenderer();
+        $this->initCaches();
+    }
+
+    /**
+     * Various initialize methods used for fallback, which can be simplified in TYPO3 v11.0
+     */
+
+    /**
+     * Used to set $this->context. The first argument was $GLOBALS[TYPO3_CONF_VARS] (array) until TYPO3 v8,
+     * so no type hint possible.
+     *
+     * @param Context|array|null $context
+     */
+    private function initializeContextWithGlobalFallback($context): void
     {
         if ($context instanceof Context) {
             $this->context = $context;
@@ -682,7 +721,18 @@ class TypoScriptFrontendController implements LoggerAwareInterface
             trigger_error('TypoScriptFrontendController requires a context object as first constructor argument in TYPO3 v11.0, now falling back to the global Context. This fallback layer will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
             $this->context = GeneralUtility::makeInstance(Context::class);
         }
-        $request = $GLOBALS['TYPO3_REQUEST'] ?? ServerRequestFactory::fromGlobals();
+    }
+
+    /**
+     * Second argument of the constructor. Until TYPO3 v10, this was the Page ID (int/string) but since TYPO3 v10.0
+     * this can also be a SiteInterface object, which will be mandatory in TYPO3 v11.0. If no Site object is given,
+     * this is fetched from the given request object.
+     *
+     * @param SiteInterface|int|string $siteOrId
+     * @param ServerRequestInterface $request
+     */
+    private function initializeSiteWithCompatibility($siteOrId, ServerRequestInterface $request): void
+    {
         if ($siteOrId instanceof SiteInterface) {
             $this->site = $siteOrId;
         } else {
@@ -694,9 +744,22 @@ class TypoScriptFrontendController implements LoggerAwareInterface
                 throw new \InvalidArgumentException('TypoScriptFrontendController must be constructed with a valid Site object or a resolved site in the current request as fallback. None given.', 1561583122);
             }
         }
+    }
+
+    /**
+     * Until TYPO3 v10.0, the third argument of the constructor was given from GET/POST "type" to define the page type
+     * Since TYPO3 v10.0, this argument is requested to be of type SiteLanguage, which will be mandatory in TYPO3 v11.0.
+     * If no SiteLanguage object is given, this is fetched from the given request object.
+     *
+     * @param SiteLanguage|int|string $siteLanguageOrType
+     * @param ServerRequestInterface $request
+     */
+    private function initializeSiteLanguageWithCompatibility($siteLanguageOrType, ServerRequestInterface $request): void
+    {
         if ($siteLanguageOrType instanceof SiteLanguage) {
             $this->language = $siteLanguageOrType;
         } else {
+            trigger_error('TypoScriptFrontendController should evaluate the parameter "type" by the PageArguments object, not by a separate constructor argument. This functionality will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
             $this->type = $siteLanguageOrType;
             if ($request->getAttribute('language') instanceof SiteLanguage) {
                 $this->language = $request->getAttribute('language');
@@ -704,38 +767,86 @@ class TypoScriptFrontendController implements LoggerAwareInterface
                 throw new \InvalidArgumentException('TypoScriptFrontendController must be constructed with a valid SiteLanguage object or a resolved site in the current request as fallback. None given.', 1561583127);
             }
         }
+    }
 
-        if (!$pageArguments instanceof PageArguments) {
-            $pageArguments = $request->getAttribute('routing');
-            if (!$pageArguments instanceof PageArguments) {
-                trigger_error('TypoScriptFrontendController must be constructed with a valid PageArguments object or a resolved page argument in the current request as fallback. None given.', E_USER_DEPRECATED);
-                $queryParams = $request->getQueryParams();
-                $pageId = $queryParams['id'] ?? $request->getParsedBody()['id'] ?? 0;
-                $pageType = $queryParams['type'] ?? $request->getParsedBody()['type'] ?? 0;
-                $pageArguments = new PageArguments((int)$pageId, (string)$pageType, [], $queryParams);
-            }
+    /**
+     * Since TYPO3 v10.0, the fourth constructor argument should be of type PageArguments. However, until TYPO3 v8,
+     * this was the GET/POST parameter "no_cache". If no PageArguments object is given, the given request is checked
+     * for the PageArguments.
+     *
+     * @param bool|string|PageArguments|null $pageArguments
+     * @param ServerRequestInterface $request
+     * @return PageArguments
+     */
+    private function buildPageArgumentsWithFallback($pageArguments, ServerRequestInterface $request): PageArguments
+    {
+        if ($pageArguments instanceof PageArguments) {
+            return $pageArguments;
         }
-        $this->setPageArguments($pageArguments);
+        if ($request->getAttribute('routing') instanceof PageArguments) {
+            return $request->getAttribute('routing');
+        }
+        trigger_error('TypoScriptFrontendController must be constructed with a valid PageArguments object or a resolved page argument in the current request as fallback. None given.', E_USER_DEPRECATED);
+        $queryParams = $request->getQueryParams();
+        $pageId = $this->id ?: ($queryParams['id'] ?? $request->getParsedBody()['id'] ?? 0);
+        $pageType = $this->type ?: ($queryParams['type'] ?? $request->getParsedBody()['type'] ?? 0);
+        return new PageArguments((int)$pageId, (string)$pageType, [], $queryParams);
+    }
 
-        if ($cHashOrFrontendUser !== null) {
-            if ($cHashOrFrontendUser instanceof FrontendUserAuthentication) {
-                $this->fe_user = $cHashOrFrontendUser;
-            } else {
-                trigger_error('TypoScriptFrontendController should evaluate the parameter "cHash" by the PageArguments object, not by a separate constructor argument. This functionality will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
-                $this->cHash = $cHashOrFrontendUser;
-            }
+    /**
+     * Since TYPO3 v10.0, the fifth constructor argument is expected to to be of Type FrontendUserAuthentication.
+     * However, up until TYPO3 v9.5 this argument was used to define the "cHash" GET/POST parameter. In order to
+     * ensure maximum compatibility, a deprecation is triggered if an old argument is still used, and PageArguments
+     * are updated accordingly, and returned.
+     *
+     * @param string|FrontendUserAuthentication|null $cHashOrFrontendUser
+     * @param PageArguments $pageArguments
+     * @return PageArguments
+     */
+    private function initializeFrontendUserOrUpdateCHashArgument($cHashOrFrontendUser, PageArguments $pageArguments): PageArguments
+    {
+        if ($cHashOrFrontendUser === null) {
+            return $pageArguments;
         }
-        if ($MP !== null) {
-            trigger_error('TypoScriptFrontendController should evaluate the MountPoint Parameter "MP" by the PageArguments object, not by a separate constructor argument. This functionality will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
-            if ($GLOBALS['TYPO3_CONF_VARS']['FE']['enable_mount_pids']) {
-                $this->MP = (string)$MP;
-            }
+        if ($cHashOrFrontendUser instanceof FrontendUserAuthentication) {
+            $this->fe_user = $cHashOrFrontendUser;
+            return $pageArguments;
         }
+        trigger_error('TypoScriptFrontendController should evaluate the parameter "cHash" by the PageArguments object, not by a separate constructor argument. This functionality will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
+        return new PageArguments(
+            $pageArguments->getPageId(),
+            $pageArguments->getPageType(),
+            $pageArguments->getRouteArguments(),
+            array_replace_recursive($pageArguments->getStaticArguments(), ['cHash' => $cHashOrFrontendUser]),
+            $pageArguments->getDynamicArguments()
+        );
+    }
 
-        $this->domainStartPage = $this->site->getRootPageId();
-        $this->uniqueString = md5(microtime());
-        $this->initPageRenderer();
-        $this->initCaches();
+    /**
+     * Since TYPO3 v10.0 the seventh constructor argument is not needed anymore, as all data is already provided by
+     * the given PageArguments object. However, if a specific MP parameter is given anyways, the PageArguments object
+     * is updated and returned.
+     *
+     * @param string|null $MP
+     * @param PageArguments $pageArguments
+     * @return PageArguments
+     */
+    private function initializeLegacyMountPointArgument(?string $MP, PageArguments $pageArguments): PageArguments
+    {
+        if ($MP === null) {
+            return $pageArguments;
+        }
+        trigger_error('TypoScriptFrontendController should evaluate the MountPoint Parameter "MP" by the PageArguments object, not by a separate constructor argument. This functionality will be removed in TYPO3 v11.0', E_USER_DEPRECATED);
+        if (!$GLOBALS['TYPO3_CONF_VARS']['FE']['enable_mount_pids']) {
+            return $pageArguments;
+        }
+        return new PageArguments(
+            $pageArguments->getPageId(),
+            $pageArguments->getPageType(),
+            $pageArguments->getRouteArguments(),
+            array_replace_recursive($pageArguments->getStaticArguments(), ['MP' => $MP]),
+            $pageArguments->getDynamicArguments()
+        );
     }
 
     /**
-- 
GitLab