From 5601673dd96b886b76a25d977db00435e1bad26f Mon Sep 17 00:00:00 2001
From: Benjamin Franzke <bfr@qbus.de>
Date: Fri, 22 Nov 2019 07:26:43 +0100
Subject: [PATCH] [BUGFIX] Provide global Symfony container instance in upgrade
 wizards
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit

Upgrade wizars may contain legacy code that requires services
using makeInstance. As makeInstance needs to use the PSR-11 container
for services that require dependency injection, we need to provide
the global container intance during upgrade wizard execution.

Also ensure that only *one* (singleton) PackageManager instance is used:
ProviderConfigurationLoader used to create a second PackageManager
instance. It passed a constructor argument which caused GU::makeInstance
to bypass the symfony (PSR-11) container and to create a new instance
(on its own). This new instance was then cached as singleton instance and
all subsequent makeInstance calls returned this instance instead of the
global instance. Therefore we now ensure that new singleton instances
will not be cached if the singleton is known by the PSR-11 container.
This inconsistency caused the Typo3DbExtractionUpdate to fail because
some codepaths used one PackageManager instance (through DI) and some
codepaths another instance (through makeInstance).

Change-Id: I947310d1c2e46f7ded6399ad48d94efa9854f347
Releases: master
Resolves: #89504
Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/62425
Tested-by: TYPO3com <noreply@typo3.com>
Tested-by: Benni Mack <benni@typo3.org>
Tested-by: Susanne Moog <look@susi.dev>
Reviewed-by: Benni Mack <benni@typo3.org>
Reviewed-by: Jörg Bösche <typo3@joergboesche.de>
Reviewed-by: Susanne Moog <look@susi.dev>
---
 typo3/sysext/core/Classes/Core/Bootstrap.php  |  5 +++++
 .../ProviderConfigurationLoader.php           |  6 +-----
 .../core/Classes/Utility/GeneralUtility.php   |  4 ++--
 .../ContentObjectRendererTest.php             |  3 +++
 .../Classes/Controller/AbstractController.php | 10 +++++++--
 .../Classes/Controller/UpgradeController.php  | 21 ++++++++++++-------
 .../Classes/Service/LateBootService.php       |  7 +++++--
 7 files changed, 38 insertions(+), 18 deletions(-)

diff --git a/typo3/sysext/core/Classes/Core/Bootstrap.php b/typo3/sysext/core/Classes/Core/Bootstrap.php
index cf18dd731464..f8969689daec 100644
--- a/typo3/sysext/core/Classes/Core/Bootstrap.php
+++ b/typo3/sysext/core/Classes/Core/Bootstrap.php
@@ -132,6 +132,11 @@ class Bootstrap
         // makeInstance() method creates classes using the container from now on.
         GeneralUtility::setContainer($container);
 
+        // Reset singleton instances in order for GeneralUtility::makeInstance() to use
+        // ContainerInterface->get() for early services from now on.
+        GeneralUtility::removeSingletonInstance(LogManager::class, $logManager);
+        GeneralUtility::removeSingletonInstance(PackageManager::class, $packageManager);
+
         if ($failsafe) {
             $bootState->done = true;
             return $container;
diff --git a/typo3/sysext/core/Classes/ExpressionLanguage/ProviderConfigurationLoader.php b/typo3/sysext/core/Classes/ExpressionLanguage/ProviderConfigurationLoader.php
index e90ab08a50c8..51f9e2071df3 100644
--- a/typo3/sysext/core/Classes/ExpressionLanguage/ProviderConfigurationLoader.php
+++ b/typo3/sysext/core/Classes/ExpressionLanguage/ProviderConfigurationLoader.php
@@ -5,7 +5,6 @@ namespace TYPO3\CMS\Core\ExpressionLanguage;
 
 use TYPO3\CMS\Core\Cache\CacheManager;
 use TYPO3\CMS\Core\Package\PackageManager;
-use TYPO3\CMS\Core\Service\DependencyOrderingService;
 use TYPO3\CMS\Core\Utility\GeneralUtility;
 
 /**
@@ -22,10 +21,7 @@ class ProviderConfigurationLoader
      */
     public function getExpressionLanguageProviders(): array
     {
-        $packageManager = GeneralUtility::makeInstance(
-            PackageManager::class,
-            GeneralUtility::makeInstance(DependencyOrderingService::class)
-        );
+        $packageManager = GeneralUtility::makeInstance(PackageManager::class);
         $cache = GeneralUtility::makeInstance(CacheManager::class)->getCache('core');
 
         if ($cache->has($this->cacheIdentifier)) {
diff --git a/typo3/sysext/core/Classes/Utility/GeneralUtility.php b/typo3/sysext/core/Classes/Utility/GeneralUtility.php
index 7d4c71be1efb..dd4fd6491211 100644
--- a/typo3/sysext/core/Classes/Utility/GeneralUtility.php
+++ b/typo3/sysext/core/Classes/Utility/GeneralUtility.php
@@ -3459,8 +3459,8 @@ class GeneralUtility
 
         // Create new instance and call constructor with parameters
         $instance = new $finalClassName(...$constructorArguments);
-        // Register new singleton instance
-        if ($instance instanceof SingletonInterface) {
+        // Register new singleton instance, but only if it is not a known PSR-11 container service
+        if ($instance instanceof SingletonInterface && !(self::$container !== null && self::$container->has($className))) {
             self::$singletonInstances[$finalClassName] = $instance;
         }
         if ($instance instanceof LoggerAwareInterface) {
diff --git a/typo3/sysext/frontend/Tests/Unit/ContentObject/ContentObjectRendererTest.php b/typo3/sysext/frontend/Tests/Unit/ContentObject/ContentObjectRendererTest.php
index defd00bcb064..627e6c44499a 100644
--- a/typo3/sysext/frontend/Tests/Unit/ContentObject/ContentObjectRendererTest.php
+++ b/typo3/sysext/frontend/Tests/Unit/ContentObject/ContentObjectRendererTest.php
@@ -37,6 +37,7 @@ use TYPO3\CMS\Core\Resource\Exception\InvalidPathException;
 use TYPO3\CMS\Core\Resource\File;
 use TYPO3\CMS\Core\Resource\ResourceFactory;
 use TYPO3\CMS\Core\Resource\ResourceStorage;
+use TYPO3\CMS\Core\Service\DependencyOrderingService;
 use TYPO3\CMS\Core\Site\Entity\Site;
 use TYPO3\CMS\Core\Site\Entity\SiteLanguage;
 use TYPO3\CMS\Core\TimeTracker\TimeTracker;
@@ -1710,6 +1711,8 @@ class ContentObjectRendererTest extends UnitTestCase
      */
     public function getDataWithTypeSiteWithBaseVariants(): void
     {
+        $packageManager = new PackageManager(new DependencyOrderingService);
+        GeneralUtility::setSingletonInstance(PackageManager::class, $packageManager);
         $cacheManagerProphecy = $this->prophesize(CacheManager::class);
         $cacheManagerProphecy->getCache('core')->willReturn(new NullFrontend('core'));
         GeneralUtility::setSingletonInstance(CacheManager::class, $cacheManagerProphecy->reveal());
diff --git a/typo3/sysext/install/Classes/Controller/AbstractController.php b/typo3/sysext/install/Classes/Controller/AbstractController.php
index 85c4fca6aa16..b72a63726179 100644
--- a/typo3/sysext/install/Classes/Controller/AbstractController.php
+++ b/typo3/sysext/install/Classes/Controller/AbstractController.php
@@ -59,10 +59,16 @@ class AbstractController
      * Those actions can potentially fatal if some old extension is loaded that triggers
      * a fatal in ext_localconf or ext_tables code! Use only if really needed.
      *
+     * @param bool $resetContainer
      * @return ContainerInterface
      */
-    protected function loadExtLocalconfDatabaseAndExtTables(): ContainerInterface
+    public function loadExtLocalconfDatabaseAndExtTables(bool $resetContainer = true): ContainerInterface
     {
-        return GeneralUtility::makeInstance(LateBootService::class)->loadExtLocalconfDatabaseAndExtTables();
+        return GeneralUtility::makeInstance(LateBootService::class)->loadExtLocalconfDatabaseAndExtTables($resetContainer);
+    }
+
+    public function resetGlobalContainer(): void
+    {
+        GeneralUtility::makeInstance(LateBootService::class)->makeCurrent(null, []);
     }
 }
diff --git a/typo3/sysext/install/Classes/Controller/UpgradeController.php b/typo3/sysext/install/Classes/Controller/UpgradeController.php
index 8c263954e883..1cce33f25780 100644
--- a/typo3/sysext/install/Classes/Controller/UpgradeController.php
+++ b/typo3/sysext/install/Classes/Controller/UpgradeController.php
@@ -916,9 +916,10 @@ class UpgradeController extends AbstractController
     public function upgradeWizardsBlockingDatabaseAddsAction(): ResponseInterface
     {
         // ext_localconf, db and ext_tables must be loaded for the updates :(
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $adds = $upgradeWizardsService->getBlockingDatabaseAdds();
+        $this->resetGlobalContainer();
         $needsUpdate = false;
         if (!empty($adds)) {
             $needsUpdate = true;
@@ -938,9 +939,10 @@ class UpgradeController extends AbstractController
     public function upgradeWizardsBlockingDatabaseExecuteAction(): ResponseInterface
     {
         // ext_localconf, db and ext_tables must be loaded for the updates :(
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $upgradeWizardsService->addMissingTablesAndFields();
+        $this->resetGlobalContainer();
         $messages = new FlashMessageQueue('install');
         $messages->enqueue(new FlashMessage(
             '',
@@ -994,10 +996,11 @@ class UpgradeController extends AbstractController
      */
     public function upgradeWizardsDoneUpgradesAction(): ResponseInterface
     {
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $wizardsDone = $upgradeWizardsService->listOfWizardsDone();
         $rowUpdatersDone = $upgradeWizardsService->listOfRowUpdatersDone();
+        $this->resetGlobalContainer();
         $messages = new FlashMessageQueue('install');
         if (empty($wizardsDone) && empty($rowUpdatersDone)) {
             $messages->enqueue(new FlashMessage(
@@ -1022,10 +1025,11 @@ class UpgradeController extends AbstractController
     public function upgradeWizardsExecuteAction(ServerRequestInterface $request): ResponseInterface
     {
         // ext_localconf, db and ext_tables must be loaded for the updates :(
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $identifier = $request->getParsedBody()['install']['identifier'];
         $messages = $upgradeWizardsService->executeWizard($identifier);
+        $this->resetGlobalContainer();
         return new JsonResponse([
             'success' => true,
             'status' => $messages,
@@ -1041,10 +1045,11 @@ class UpgradeController extends AbstractController
     public function upgradeWizardsInputAction(ServerRequestInterface $request): ResponseInterface
     {
         // ext_localconf, db and ext_tables must be loaded for the updates :(
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $identifier = $request->getParsedBody()['install']['identifier'];
         $result = $upgradeWizardsService->getWizardUserInput($identifier);
+        $this->resetGlobalContainer();
         return new JsonResponse([
             'success' => true,
             'status' => [],
@@ -1060,9 +1065,10 @@ class UpgradeController extends AbstractController
     public function upgradeWizardsListAction(): ResponseInterface
     {
         // ext_localconf, db and ext_tables must be loaded for the updates :(
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $upgradeWizardsService = new UpgradeWizardsService();
         $wizards = $upgradeWizardsService->getUpgradeWizardsList();
+        $this->resetGlobalContainer();
         return new JsonResponse([
             'success' => true,
             'status' => [],
@@ -1078,10 +1084,11 @@ class UpgradeController extends AbstractController
      */
     public function upgradeWizardsMarkUndoneAction(ServerRequestInterface $request): ResponseInterface
     {
-        $this->loadExtLocalconfDatabaseAndExtTables();
+        $this->loadExtLocalconfDatabaseAndExtTables(false);
         $wizardToBeMarkedAsUndoneIdentifier = $request->getParsedBody()['install']['identifier'];
         $upgradeWizardsService = new UpgradeWizardsService();
         $result = $upgradeWizardsService->markWizardUndone($wizardToBeMarkedAsUndoneIdentifier);
+        $this->resetGlobalContainer();
         $messages = new FlashMessageQueue('install');
         if ($result) {
             $messages->enqueue(new FlashMessage(
diff --git a/typo3/sysext/install/Classes/Service/LateBootService.php b/typo3/sysext/install/Classes/Service/LateBootService.php
index fe7223d0c45b..291d486eafac 100644
--- a/typo3/sysext/install/Classes/Service/LateBootService.php
+++ b/typo3/sysext/install/Classes/Service/LateBootService.php
@@ -114,9 +114,10 @@ class LateBootService
     /**
      * Bootstrap a non-failsafe container and load ext_localconf
      *
+     * @param bool $resetContainer
      * @return ContainerInterface
      */
-    public function loadExtLocalconfDatabaseAndExtTables(): ContainerInterface
+    public function loadExtLocalconfDatabaseAndExtTables(bool $resetContainer = true): ContainerInterface
     {
         $container = $this->getContainer();
 
@@ -134,7 +135,9 @@ class LateBootService
         Bootstrap::loadBaseTca(false);
         Bootstrap::loadExtTables(false);
 
-        $this->makeCurrent(null, $backup);
+        if ($resetContainer) {
+            $this->makeCurrent(null, $backup);
+        }
 
         return $container;
     }
-- 
GitLab