From ca19817b4ccf312498f59428d215dfb26fb02074 Mon Sep 17 00:00:00 2001 From: Benni Mack <benni@typo3.org> Date: Sat, 23 Nov 2019 23:36:33 +0100 Subject: [PATCH] [FEATURE] Migrate various Signals to PSR-14 events in system extensions The following new PSR-14 events are added: TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent TYPO3\CMS\Impexp\Event\BeforeImportEvent TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent They replace the following "old" signals with a deprecation layer: TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider::getPageRenderer TYPO3\CMS\Backend\Controller\EditDocumentController::preInitAfter TYPO3\CMS\Backend\Controller\EditDocumentController::initAfter TYPO3\CMS\Backend\Utility\BackendUtility::getPagesTSconfigPreInclude TYPO3\CMS\Beuser\Controller\BackendUserController::switchUser TYPO3\CMS\Impexp\Utility\ImportExportUtility::afterImportExportInitialisation TYPO3\CMS\Lang\Service\TranslationService::postProcessMirrorUrl TYPO3\CMS\Linkvalidator\LinkAnalyzer::beforeAnalyzeRecord TYPO3\CMS\Seo\Canonical\CanonicalGenerator::beforeGeneratingCanonical TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss Relates: #89733 Resolves: #89813 Releases: master Change-Id: I13f2454fd8f4efb5f4c5248d0b839634b77578db Reviewed-on: https://review.typo3.org/c/Packages/TYPO3.CMS/+/62422 Tested-by: TYPO3com <noreply@typo3.com> Tested-by: Andreas Fernandez <a.fernandez@scripting-base.de> Tested-by: Susanne Moog <look@susi.dev> Reviewed-by: Andreas Fernandez <a.fernandez@scripting-base.de> Reviewed-by: Susanne Moog <look@susi.dev> --- .../Authentication/Event/SwitchUserEvent.php | 59 ++++++++++++ .../Classes/Compatibility/SlotReplacement.php | 46 +++++++++ .../Controller/EditDocumentController.php | 47 +++------ .../AfterFormEnginePageInitializedEvent.php | 51 ++++++++++ .../BeforeFormEnginePageInitializedEvent.php | 51 ++++++++++ .../Classes/Controller/LoginController.php | 19 +++- ...ageLayoutOnLoginProviderSelectionEvent.php | 62 ++++++++++++ .../UsernamePasswordLoginProvider.php | 6 -- .../Classes/Utility/BackendUtility.php | 40 +------- .../backend/Configuration/Services.yaml | 19 ++++ .../Controller/EditDocumentControllerTest.php | 4 +- .../Tests/Unit/Utility/BackendUtilityTest.php | 9 +- .../Classes/Compatibility/SlotReplacement.php | 49 ++++++++++ .../Controller/BackendUserController.php | 32 ++++--- .../sysext/beuser/Configuration/Services.yaml | 8 ++ .../Event/ModifyLoadedPageTsConfigEvent.php | 59 ++++++++++++ .../Loader/PageTsConfigLoader.php | 17 +++- typo3/sysext/core/Configuration/Services.yaml | 3 + ...sInCoreExtensionMigratedToPSR-14Events.rst | 45 +++++++-- ...sForExistingSignalSlotsInCoreExtension.rst | 36 ++++++- .../Loader/PageTsConfigLoaderTest.php | 23 ++++- .../extbase/Classes/SignalSlot/Dispatcher.php | 52 ++++++++++ .../Classes/Compatibility/SlotReplacement.php | 49 ++++++++++ .../Classes/Event/BeforeImportEvent.php | 39 ++++++++ .../Classes/Utility/ImportExportUtility.php | 37 +++----- .../sysext/impexp/Configuration/Services.yaml | 11 +++ .../Classes/Compatibility/SlotReplacement.php | 54 +++++++++++ .../ModifyLanguagePackRemoteBaseUrlEvent.php | 55 +++++++++++ .../Classes/Service/LanguagePackService.php | 25 +++-- .../Php/ClassConstantMatcher.php | 24 +++++ .../install/Configuration/Services.yaml | 9 ++ .../Classes/Compatibility/SlotReplacement.php | 57 +++++++++++ .../Event/BeforeRecordIsAnalyzedEvent.php | 94 ++++++++++++++++++ .../linkvalidator/Classes/LinkAnalyzer.php | 47 ++------- .../linkvalidator/Configuration/Services.yaml | 7 ++ .../Tests/Functional/LinkAnalyzerTest.php | 10 +- .../Classes/Canonical/CanonicalGenerator.php | 34 ++----- .../Classes/Compatibility/SlotReplacement.php | 51 ++++++++++ .../Event/ModifyUrlForCanonicalTagEvent.php | 43 +++++++++ typo3/sysext/seo/Configuration/Services.yaml | 7 ++ .../Classes/Compatibility/SlotReplacement.php | 87 +++++++++++++++++ ...CompiledCacheableDataForWorkspaceEvent.php | 71 ++++++++++++++ .../AfterDataGeneratedForWorkspaceEvent.php | 71 ++++++++++++++ .../Classes/Event/GetVersionedDataEvent.php | 95 +++++++++++++++++++ .../Classes/Event/SortVersionedDataEvent.php | 87 +++++++++++++++++ .../Classes/Service/GridDataService.php | 86 +++++++++-------- .../workspaces/Configuration/Services.yaml | 22 +++++ .../Controller/Remote/RemoteServerTest.php | 2 +- 48 files changed, 1659 insertions(+), 252 deletions(-) create mode 100644 typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php create mode 100644 typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php create mode 100644 typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php create mode 100644 typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php create mode 100644 typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php create mode 100644 typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php create mode 100644 typo3/sysext/install/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php create mode 100644 typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php create mode 100644 typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php create mode 100644 typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php create mode 100644 typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php create mode 100644 typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php create mode 100644 typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php create mode 100644 typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php diff --git a/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php b/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php new file mode 100644 index 000000000000..366c9075b06d --- /dev/null +++ b/typo3/sysext/backend/Classes/Authentication/Event/SwitchUserEvent.php @@ -0,0 +1,59 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Authentication\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * This event is triggered when a "SU" (switch user) action has been triggered + */ +final class SwitchUserEvent +{ + /** + * @var string + */ + private $sessionId; + + /** + * @var array + */ + private $targetUser; + + /** + * @var array + */ + private $currentUser; + + public function __construct(string $sessionId, array $targetUser, array $currentUser) + { + $this->sessionId = $sessionId; + $this->targetUser = $targetUser; + $this->currentUser = $currentUser; + } + + public function getSessionId(): string + { + return $this->sessionId; + } + + public function getTargetUser(): array + { + return $this->targetUser; + } + + public function getCurrentUser(): array + { + return $this->currentUser; + } +} diff --git a/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php index a0df516f0eb3..088766c86935 100644 --- a/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php +++ b/typo3/sysext/backend/Classes/Compatibility/SlotReplacement.php @@ -17,6 +17,13 @@ namespace TYPO3\CMS\Backend\Compatibility; use TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent; use TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem; +use TYPO3\CMS\Backend\Controller\EditDocumentController; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; +use TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider; +use TYPO3\CMS\Backend\Utility\BackendUtility; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; /** @@ -51,4 +58,43 @@ class SlotReplacement [$event->getToolbarItem()] ); } + + public function onLoginProviderGetPageRenderer(ModifyPageLayoutOnLoginProviderSelectionEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + UsernamePasswordLoginProvider::class, + 'getPageRenderer', + [$event->getPageRenderer()] + ); + } + + public function onPreInitEditDocumentController(BeforeFormEnginePageInitializedEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + EditDocumentController::class, + 'preInitAfter', + [$event->getController(), 'request' => $event->getRequest()] + ); + } + + public function onInitEditDocumentController(AfterFormEnginePageInitializedEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + EditDocumentController::class, + 'initAfter', + [$event->getController(), 'request' => $event->getRequest()] + ); + } + + public function emitGetPagesTSconfigPreIncludeSignalBackendUtility(ModifyLoadedPageTsConfigEvent $event): void + { + $rootLine = $event->getRootLine(); + $page = end($rootLine); + $signalArguments = $this->signalSlotDispatcher->dispatch( + BackendUtility::class, + 'getPagesTSconfigPreInclude', + [$event->getTsConfig(), (int)$page['uid'], $rootLine, false] + ); + $event->setTsConfig($signalArguments[0]); + } } diff --git a/typo3/sysext/backend/Classes/Controller/EditDocumentController.php b/typo3/sysext/backend/Classes/Controller/EditDocumentController.php index a4ec19da5ae1..bea6ac250387 100644 --- a/typo3/sysext/backend/Classes/Controller/EditDocumentController.php +++ b/typo3/sysext/backend/Classes/Controller/EditDocumentController.php @@ -15,8 +15,11 @@ namespace TYPO3\CMS\Backend\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; use TYPO3\CMS\Backend\Form\Exception\AccessDeniedException; use TYPO3\CMS\Backend\Form\Exception\DatabaseRecordException; use TYPO3\CMS\Backend\Form\FormDataCompiler; @@ -51,7 +54,6 @@ use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\HttpUtility; use TYPO3\CMS\Core\Utility\MathUtility; use TYPO3\CMS\Core\Utility\PathUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; /** * Main backend controller almost always used if some database record is edited in the backend. @@ -347,11 +349,6 @@ class EditDocumentController */ protected $dontStoreDocumentRef = 0; - /** - * @var \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected $signalSlotDispatcher; - /** * Stores information needed to preview the currently saved record * @@ -381,10 +378,13 @@ class EditDocumentController protected $isPageInFreeTranslationMode = false; /** - * Constructor + * @var EventDispatcherInterface */ - public function __construct() + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->moduleTemplate = GeneralUtility::makeInstance(ModuleTemplate::class); $this->moduleTemplate->setUiBlock(true); // @todo Used by TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching @@ -489,8 +489,8 @@ class EditDocumentController $this->getBackendUser()->setTemporaryWorkspace($this->workspace); } - $this->emitFunctionAfterSignal('preInit', $request); - + $event = new BeforeFormEnginePageInitializedEvent($this, $request); + $this->eventDispatcher->dispatch($event); return null; } @@ -779,7 +779,8 @@ class EditDocumentController // Set context sensitive menu $this->moduleTemplate->getPageRenderer()->loadRequireJsModule('TYPO3/CMS/Backend/ContextMenu'); - $this->emitFunctionAfterSignal('init', $request); + $event = new AfterFormEnginePageInitializedEvent($this, $request); + $this->eventDispatcher->dispatch($event); } /** @@ -2452,30 +2453,6 @@ class EditDocumentController return new RedirectResponse($retUrl, 303); } - /** - * Emits a signal after a function was executed - * - * @param string $signalName - * @param ServerRequestInterface $request - */ - protected function emitFunctionAfterSignal($signalName, ServerRequestInterface $request): void - { - $this->getSignalSlotDispatcher()->dispatch(__CLASS__, $signalName . 'After', [$this, 'request' => $request]); - } - - /** - * Get the SignalSlot dispatcher - * - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected function getSignalSlotDispatcher() - { - if (!isset($this->signalSlotDispatcher)) { - $this->signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - } - return $this->signalSlotDispatcher; - } - /** * @return \TYPO3\CMS\Core\Authentication\BackendUserAuthentication */ diff --git a/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php b/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php new file mode 100644 index 000000000000..af5f41f86896 --- /dev/null +++ b/typo3/sysext/backend/Classes/Controller/Event/AfterFormEnginePageInitializedEvent.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Controller\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\EditDocumentController; + +/** + * Event to listen to after the form engine has been initialized (= all data has been persisted) + */ +final class AfterFormEnginePageInitializedEvent +{ + /** + * @var EditDocumentController + */ + private $controller; + + /** + * @var ServerRequestInterface + */ + private $request; + + public function __construct(EditDocumentController $controller, ServerRequestInterface $request) + { + $this->controller = $controller; + $this->request = $request; + } + + public function getController(): EditDocumentController + { + return $this->controller; + } + + public function getRequest(): ServerRequestInterface + { + return $this->request; + } +} diff --git a/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php b/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php new file mode 100644 index 000000000000..585fac7e505f --- /dev/null +++ b/typo3/sysext/backend/Classes/Controller/Event/BeforeFormEnginePageInitializedEvent.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Backend\Controller\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\ServerRequestInterface; +use TYPO3\CMS\Backend\Controller\EditDocumentController; + +/** + * Event to listen to before the form engine has been initialized (= before all data will be persisted) + */ +final class BeforeFormEnginePageInitializedEvent +{ + /** + * @var EditDocumentController + */ + private $controller; + + /** + * @var ServerRequestInterface + */ + private $request; + + public function __construct(EditDocumentController $controller, ServerRequestInterface $request) + { + $this->controller = $controller; + $this->request = $request; + } + + public function getController(): EditDocumentController + { + return $this->controller; + } + + public function getRequest(): ServerRequestInterface + { + return $this->request; + } +} diff --git a/typo3/sysext/backend/Classes/Controller/LoginController.php b/typo3/sysext/backend/Classes/Controller/LoginController.php index 73f3d8919cf9..39566ec6b2b0 100644 --- a/typo3/sysext/backend/Classes/Controller/LoginController.php +++ b/typo3/sysext/backend/Classes/Controller/LoginController.php @@ -15,11 +15,13 @@ namespace TYPO3\CMS\Backend\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Http\Message\ResponseInterface; use Psr\Http\Message\ServerRequestInterface; use Psr\Log\LoggerAwareInterface; use Psr\Log\LoggerAwareTrait; use TYPO3\CMS\Backend\Exception; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; use TYPO3\CMS\Backend\LoginProvider\LoginProviderInterface; use TYPO3\CMS\Backend\Routing\UriBuilder; use TYPO3\CMS\Backend\Template\DocumentTemplate; @@ -99,13 +101,17 @@ class LoginController implements LoggerAwareInterface * @var DocumentTemplate */ protected $documentTemplate; + + protected $eventDispatcher; + /** - * @var \TYPO3\CMS\Core\Information\Typo3Copyright + * @var Typo3Copyright */ private $copyright; - public function __construct(Typo3Copyright $copyright) + public function __construct(Typo3Copyright $copyright, EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->copyright = $copyright; } @@ -302,6 +308,15 @@ class LoginController implements LoggerAwareInterface /** @var LoginProviderInterface $loginProvider */ $loginProvider = GeneralUtility::makeInstance($this->loginProviders[$this->loginProviderIdentifier]['provider']); + + $this->eventDispatcher->dispatch( + new ModifyPageLayoutOnLoginProviderSelectionEvent( + $this, + $this->view, + $pageRenderer + ) + ); + $loginProvider->render($this->view, $pageRenderer, $this); $content = $this->documentTemplate->startPage('TYPO3 CMS Login: ' . $GLOBALS['TYPO3_CONF_VARS']['SYS']['sitename']); diff --git a/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php b/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php new file mode 100644 index 000000000000..3987f2568478 --- /dev/null +++ b/typo3/sysext/backend/Classes/LoginProvider/Event/ModifyPageLayoutOnLoginProviderSelectionEvent.php @@ -0,0 +1,62 @@ +<?php +namespace TYPO3\CMS\Backend\LoginProvider\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Backend\Controller\LoginController; +use TYPO3\CMS\Core\Page\PageRenderer; +use TYPO3\CMS\Fluid\View\StandaloneView; + +/** + * Allows to modify variables for the view depending on a special login provider set in the controller. + */ +final class ModifyPageLayoutOnLoginProviderSelectionEvent +{ + /** + * @var LoginController + */ + private $controller; + + /** + * @var StandaloneView + */ + private $view; + + /** + * @var PageRenderer + */ + private $pageRenderer; + + public function __construct(LoginController $controller, StandaloneView $view, PageRenderer $pageRenderer) + { + $this->controller = $controller; + $this->view = $view; + $this->pageRenderer = $pageRenderer; + } + + public function getController(): LoginController + { + return $this->controller; + } + + public function getView(): StandaloneView + { + return $this->view; + } + + public function getPageRenderer(): PageRenderer + { + return $this->pageRenderer; + } +} diff --git a/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php b/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php index e649f2da2ba9..c965b392df00 100644 --- a/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php +++ b/typo3/sysext/backend/Classes/LoginProvider/UsernamePasswordLoginProvider.php @@ -17,8 +17,6 @@ namespace TYPO3\CMS\Backend\LoginProvider; use TYPO3\CMS\Backend\Controller\LoginController; use TYPO3\CMS\Core\Page\PageRenderer; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\Object\ObjectManager; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Fluid\View\StandaloneView; /** @@ -27,8 +25,6 @@ use TYPO3\CMS\Fluid\View\StandaloneView; */ class UsernamePasswordLoginProvider implements LoginProviderInterface { - const SIGNAL_getPageRenderer = 'getPageRenderer'; - /** * @param StandaloneView $view * @param PageRenderer $pageRenderer @@ -37,8 +33,6 @@ class UsernamePasswordLoginProvider implements LoginProviderInterface */ public function render(StandaloneView $view, PageRenderer $pageRenderer, LoginController $loginController) { - GeneralUtility::makeInstance(ObjectManager::class)->get(Dispatcher::class)->dispatch(__CLASS__, self::SIGNAL_getPageRenderer, [$pageRenderer]); - $pageRenderer->loadRequireJsModule('TYPO3/CMS/Backend/UserPassLogin'); $view->setTemplatePathAndFilename(GeneralUtility::getFileAbsFileName('EXT:backend/Resources/Private/Templates/UserPassLoginForm.html')); diff --git a/typo3/sysext/backend/Classes/Utility/BackendUtility.php b/typo3/sysext/backend/Classes/Utility/BackendUtility.php index faccb2c1f971..62b8b467706d 100644 --- a/typo3/sysext/backend/Classes/Utility/BackendUtility.php +++ b/typo3/sysext/backend/Classes/Utility/BackendUtility.php @@ -14,12 +14,14 @@ namespace TYPO3\CMS\Backend\Utility; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Log\LoggerInterface; use TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching\ConditionMatcher; use TYPO3\CMS\Backend\Routing\UriBuilder; use TYPO3\CMS\Core\Authentication\BackendUserAuthentication; use TYPO3\CMS\Core\Cache\CacheManager; use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\CMS\Core\Configuration\Parser\PageTsConfigParser; use TYPO3\CMS\Core\Core\Environment; @@ -762,10 +764,9 @@ class BackendUtility $tsDataArray['uid_' . $v['uid']] = $v['TSconfig']; } - $tsDataArray = static::emitGetPagesTSconfigPreIncludeSignal($tsDataArray, $id, $rootLine); - $tsDataArray = TypoScriptParser::checkIncludeLines_array($tsDataArray); - - return $tsDataArray; + $eventDispatcher = GeneralUtility::getContainer()->get(EventDispatcherInterface::class); + $event = $eventDispatcher->dispatch(new ModifyLoadedPageTsConfigEvent($tsDataArray, $rootLine)); + return TypoScriptParser::checkIncludeLines_array($event->getTsConfig()); } /******************************************* @@ -3942,37 +3943,6 @@ class BackendUtility return !empty($GLOBALS['TCA'][$table]['ctrl']['security']['ignoreRootLevelRestriction']); } - /** - * Get the SignalSlot dispatcher - * - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected static function getSignalSlotDispatcher() - { - return GeneralUtility::makeInstance(\TYPO3\CMS\Extbase\SignalSlot\Dispatcher::class); - } - - /** - * Emits signal to modify the page TSconfig before include - * - * @param array $TSdataArray Current TSconfig data array - Can be modified by slots! - * @param int $id Page ID we are handling - * @param array $rootLine Rootline array of page - * @return array Modified Data array - */ - protected static function emitGetPagesTSconfigPreIncludeSignal( - array $TSdataArray, - $id, - array $rootLine - ) { - $signalArguments = static::getSignalSlotDispatcher()->dispatch( - __CLASS__, - 'getPagesTSconfigPreInclude', - [$TSdataArray, $id, $rootLine, false] - ); - return $signalArguments[0]; - } - /** * @param string $table * @return Connection diff --git a/typo3/sysext/backend/Configuration/Services.yaml b/typo3/sysext/backend/Configuration/Services.yaml index b9a64911f0fa..90f55c8a6256 100644 --- a/typo3/sysext/backend/Configuration/Services.yaml +++ b/typo3/sysext/backend/Configuration/Services.yaml @@ -19,6 +19,9 @@ services: TYPO3\CMS\Backend\History\RecordHistoryRollback: public: true + TYPO3\CMS\Backend\Controller\EditDocumentController: + tags: ['backend.controller'] + TYPO3\CMS\Backend\Controller\LoginController: tags: ['backend.controller'] @@ -33,6 +36,22 @@ services: identifier: 'legacy-slot' method: 'onSystemInformationToolbarEvent' event: TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onLoginProviderGetPageRenderer' + event: TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'emitGetPagesTSconfigPreIncludeSignalBackendUtility' + event: TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onPreInitEditDocumentController' + event: TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onInitEditDocumentController' + event: TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent # Category security checks for backend users TYPO3\CMS\Backend\Security\CategoryPermissionsAspect: diff --git a/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php b/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php index 8630752d5967..6c3bf1d7c7bc 100644 --- a/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php +++ b/typo3/sysext/backend/Tests/Unit/Controller/EditDocumentControllerTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Backend\Tests\Unit\Controller; */ use Prophecy\Argument; +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Controller\EditDocumentController; use TYPO3\CMS\Backend\Template\ModuleTemplate; use TYPO3\CMS\Core\Localization\LanguageService; @@ -52,10 +53,11 @@ class EditDocumentControllerTest extends UnitTestCase $GLOBALS['LANG'] = $this->prophesize(LanguageService::class)->reveal(); GeneralUtility::addInstance(ModuleTemplate::class, $moduleTemplate->reveal()); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); $mock = \Closure::bind(static function (EditDocumentController $editDocumentController) use (&$result, $typoScript) { return $editDocumentController->parseAdditionalGetParameters($result, $typoScript); }, null, EditDocumentController::class); - $mock(new EditDocumentController()); + $mock(new EditDocumentController($eventDispatcher->reveal())); self::assertSame($expectedParameters, $result); } diff --git a/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php b/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php index c11d1e2d5407..0e78b11f4d31 100644 --- a/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php +++ b/typo3/sysext/backend/Tests/Unit/Utility/BackendUtilityTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Backend\Tests\Unit\Utility; use Prophecy\Argument; use Prophecy\Prophecy\ObjectProphecy; +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching\ConditionMatcher; use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\LabelFromItemListMergedReturnsCorrectFieldsFixture; use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\ProcessedValueForGroupWithMultipleAllowedTablesFixture; @@ -24,6 +25,8 @@ use TYPO3\CMS\Backend\Tests\Unit\Utility\Fixtures\ProcessedValueForSelectWithMMR use TYPO3\CMS\Backend\Utility\BackendUtility; use TYPO3\CMS\Core\Cache\CacheManager; use TYPO3\CMS\Core\Cache\Frontend\FrontendInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; +use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\CMS\Core\Configuration\Parser\PageTsConfigParser; use TYPO3\CMS\Core\Database\Connection; use TYPO3\CMS\Core\Database\ConnectionPool; @@ -283,7 +286,7 @@ class BackendUtilityTest extends UnitTestCase $relationHandlerInstance = $relationHandlerProphet->reveal(); $relationHandlerInstance->tableArray['sys_category'] = [1, 2]; - list($queryBuilderProphet, $connectionPoolProphet) = $this->mockDatabaseConnection('sys_category'); + [$queryBuilderProphet, $connectionPoolProphet] = $this->mockDatabaseConnection('sys_category'); $statementProphet = $this->prophesize(\Doctrine\DBAL\Driver\Statement::class); $statementProphet->fetch()->shouldBeCalled()->willReturn( [ @@ -1035,6 +1038,10 @@ class BackendUtilityTest extends UnitTestCase { $expected = ['called.' => ['config']]; $pageId = 13; + $eventDispatcherProphecy = $this->prophesize(EventDispatcherInterface::class); + $eventDispatcherProphecy->dispatch(Argument::any())->willReturn(new ModifyLoadedPageTsConfigEvent([], [])); + $loader = new PageTsConfigLoader($eventDispatcherProphecy->reveal()); + GeneralUtility::addInstance(PageTsConfigLoader::class, $loader); $parserProphecy = $this->prophesize(PageTsConfigParser::class); $parserProphecy->parse(Argument::cetera())->willReturn($expected); GeneralUtility::addInstance(PageTsConfigParser::class, $parserProphecy->reveal()); diff --git a/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..54e864ea9865 --- /dev/null +++ b/typo3/sysext/beuser/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,49 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Beuser\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; +use TYPO3\CMS\Beuser\Controller\BackendUserController; +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; + +/** + * This class provides a replacement for all existing signals in EXT:beuser of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onSwitchUser(SwitchUserEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + BackendUserController::class, + 'switchUser', + [$event->getTargetUser()] + ); + } +} diff --git a/typo3/sysext/beuser/Classes/Controller/BackendUserController.php b/typo3/sysext/beuser/Classes/Controller/BackendUserController.php index fc7a32e537ef..61a4dab44c4d 100644 --- a/typo3/sysext/beuser/Classes/Controller/BackendUserController.php +++ b/typo3/sysext/beuser/Classes/Controller/BackendUserController.php @@ -14,10 +14,13 @@ namespace TYPO3\CMS\Beuser\Controller; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; use TYPO3\CMS\Core\Authentication\BackendUserAuthentication; use TYPO3\CMS\Core\Session\Backend\SessionBackendInterface; use TYPO3\CMS\Core\Session\SessionManager; use TYPO3\CMS\Core\Utility\GeneralUtility; +use TYPO3\CMS\Core\Utility\HttpUtility; use TYPO3\CMS\Extbase\Mvc\Controller\ActionController; use TYPO3\CMS\Extbase\Mvc\View\ViewInterface; use TYPO3\CMS\Extbase\Utility\LocalizationUtility; @@ -58,6 +61,11 @@ class BackendUserController extends ActionController */ protected $backendUserSessionRepository; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + /** * @param \TYPO3\CMS\Beuser\Service\ModuleDataStorageService $moduleDataStorageService */ @@ -90,6 +98,11 @@ class BackendUserController extends ActionController $this->backendUserSessionRepository = $backendUserSessionRepository; } + public function injectEventDispatcher(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Load and persist module data * @@ -279,10 +292,15 @@ class BackendUserController extends ActionController ] ); - $this->emitSwitchUserSignal($targetUser); + $event = new SwitchUserEvent( + $this->getBackendUserAuthentication()->getSessionId(), + $targetUser, + $this->getBackendUserAuthentication()->user + ); + $this->eventDispatcher->dispatch($event); $redirectUrl = 'index.php' . ($GLOBALS['TYPO3_CONF_VARS']['BE']['interfaces'] ? '' : '?commandLI=1'); - \TYPO3\CMS\Core\Utility\HttpUtility::redirect($redirectUrl); + HttpUtility::redirect($redirectUrl); } } @@ -313,16 +331,6 @@ class BackendUserController extends ActionController return $latestUserUids; } - /** - * Emit a signal when using the "switch to user" functionality - * - * @param array $targetUser - */ - protected function emitSwitchUserSignal(array $targetUser) - { - $this->signalSlotDispatcher->dispatch(__CLASS__, 'switchUser', [$targetUser]); - } - /** * @return BackendUserAuthentication */ diff --git a/typo3/sysext/beuser/Configuration/Services.yaml b/typo3/sysext/beuser/Configuration/Services.yaml index 2415fd433f1b..7e181563c246 100644 --- a/typo3/sysext/beuser/Configuration/Services.yaml +++ b/typo3/sysext/beuser/Configuration/Services.yaml @@ -6,3 +6,11 @@ services: TYPO3\CMS\Beuser\: resource: '../Classes/*' + + # Listener for old Signal Slots + TYPO3\CMS\Beuser\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onSwitchUser' + event: TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent diff --git a/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php b/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php new file mode 100644 index 000000000000..5989e205e58e --- /dev/null +++ b/typo3/sysext/core/Classes/Configuration/Event/ModifyLoadedPageTsConfigEvent.php @@ -0,0 +1,59 @@ +<?php +declare(strict_types = 1); + +namespace TYPO3\CMS\Core\Configuration\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * Extensions can modify pageTSConfig entries that can be overridden or added, based on the root line + */ +final class ModifyLoadedPageTsConfigEvent +{ + /** + * @var array + */ + private $tsConfig; + + /** + * @var array + */ + private $rootLine; + + public function __construct(array $tsConfig, array $rootLine) + { + $this->tsConfig = $tsConfig; + $this->rootLine = $rootLine; + } + + public function getTsConfig(): array + { + return $this->tsConfig; + } + + public function addTsConfig(string $tsConfig): void + { + $this->tsConfig[] = $tsConfig; + } + + public function setTsConfig(array $tsConfig): void + { + $this->tsConfig = $tsConfig; + } + + public function getRootLine(): array + { + return $this->rootLine; + } +} diff --git a/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php b/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php index ee5c631ad095..6c88536044cc 100644 --- a/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php +++ b/typo3/sysext/core/Classes/Configuration/Loader/PageTsConfigLoader.php @@ -15,6 +15,8 @@ namespace TYPO3\CMS\Core\Configuration\Loader; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\TypoScript\Parser\TypoScriptParser; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; @@ -33,6 +35,16 @@ use TYPO3\CMS\Core\Utility\PathUtility; */ class PageTsConfigLoader { + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Main method to get all PageTSconfig from the rootline including the defaultTSconfig settings. * @param array $rootLine @@ -87,7 +99,10 @@ class PageTsConfigLoader } $tsData['page_' . $page['uid']] = $page['TSconfig'] ?? ''; } + + $event = $this->eventDispatcher->dispatch(new ModifyLoadedPageTsConfigEvent($tsData, $rootLine)); + // Apply includes - return TypoScriptParser::checkIncludeLines_array($tsData); + return TypoScriptParser::checkIncludeLines_array($event->getTsConfig()); } } diff --git a/typo3/sysext/core/Configuration/Services.yaml b/typo3/sysext/core/Configuration/Services.yaml index dc518deb9ad1..fc6ed9326f69 100644 --- a/typo3/sysext/core/Configuration/Services.yaml +++ b/typo3/sysext/core/Configuration/Services.yaml @@ -26,6 +26,9 @@ services: TYPO3\CMS\Core\Http\MiddlewareDispatcher: autoconfigure: false + TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader: + public: true + TYPO3\CMS\Core\Database\Schema\SqlReader: public: true diff --git a/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst b/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst index a44f9dcbe90e..7b0c050bb103 100644 --- a/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst +++ b/typo3/sysext/core/Documentation/Changelog/master/Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst @@ -10,21 +10,37 @@ Description =========== The following Signal Slots have been replaced by new PSR-14 events -which are a 1:1 equivalent: +which can be used as 1:1 equivalents: -- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider::getPageRenderer` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::preInitAfter` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::initAfter` +- :php:`TYPO3\CMS\Backend\Utility\BackendUtility::getPagesTSconfigPreInclude` +- :php:`TYPO3\CMS\Beuser\Controller\BackendUserController::switchUser` - :php:`TYPO3\CMS\Core\Database\SoftReferenceIndex::setTypoLinkPartsElement` - :php:`TYPO3\CMS\Core\Database\ReferenceIndex::shouldExcludeTableFromReferenceIndex` +- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` - :php:`TYPO3\CMS\Core\Utility\ExtensionManagementUtility::tcaIsBeingBuilt` +- :php:`TYPO3\CMS\Impexp\Utility\ImportExportUtility::afterImportExportInitialisation` - :php:`TYPO3\CMS\Install\Service\SqlExpectedSchemaService::tablesDefinitionIsBeingBuilt` -- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Lang\Service\TranslationService::postProcessMirrorUrl` +- :php:`TYPO3\CMS\Linkvalidator\LinkAnalyzer::beforeAnalyzeRecord` +- :php:`TYPO3\CMS\Seo\Canonical\CanonicalGenerator::beforeGeneratingCanonical` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` -In addition, the following public constant, marking a signal name, is deprecated: +In addition, the following public constants, marking a signal name, are deprecated: - :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::SIGNAL_PostProcessTreeData` - +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` Impact ====== @@ -42,12 +58,25 @@ Migration Use the new PSR-14 alternatives: +- :php:`TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent` +- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent` - :php:`TYPO3\CMS\Core\Imaging\Event\ModifyIconForResourcePropertiesEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\IsTableExcludedFromReferenceIndexEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\AppendLinkHandlerElementsEvent` - :php:`TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent` - :php:`TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent` - :php:`TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent` -- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent` +- :php:`TYPO3\CMS\Impexp\Event\BeforeImportEvent` +- :php:`TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent` +- :php:`TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent` +- :php:`TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent` +- :php:`TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent` .. index:: PHP-API, FullyScanned, ext:core diff --git a/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst b/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst index 078fd11018a2..a193a51ec01c 100644 --- a/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst +++ b/typo3/sysext/core/Documentation/Changelog/master/Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst @@ -13,24 +13,50 @@ PSR-14 EventDispatching allows for TYPO3 Extensions or PHP packages to extend TY The following new PSR-14 events have been introduced: +- :php:`TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent` +- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent` +- :php:`TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent` - :php:`TYPO3\CMS\Core\Imaging\Event\ModifyIconForResourcePropertiesEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\IsTableExcludedFromReferenceIndexEvent` - :php:`TYPO3\CMS\Core\DataHandling\Event\AppendLinkHandlerElementsEvent` - :php:`TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent` - :php:`TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent` - :php:`TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent` -- :php:`TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent` +- :php:`TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent` +- :php:`TYPO3\CMS\Impexp\Event\BeforeImportEvent` +- :php:`TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent` +- :php:`TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent` +- :php:`TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent` +- :php:`TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent` +- :php:`TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent` They replace the existing Extbase-based Signal Slots -- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` +- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider::getPageRenderer` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::preInitAfter` +- :php:`TYPO3\CMS\Backend\Controller\EditDocumentController::initAfter` +- :php:`TYPO3\CMS\Backend\Utility\BackendUtility::getPagesTSconfigPreInclude` +- :php:`TYPO3\CMS\Beuser\Controller\BackendUserController::switchUser` - :php:`TYPO3\CMS\Core\Database\SoftReferenceIndex::setTypoLinkPartsElement` - :php:`TYPO3\CMS\Core\Database\ReferenceIndex::shouldExcludeTableFromReferenceIndex` +- :php:`TYPO3\CMS\Core\Imaging\IconFactory::buildIconForResourceSignal` +- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` - :php:`TYPO3\CMS\Core\Utility\ExtensionManagementUtility::tcaIsBeingBuilt` +- :php:`TYPO3\CMS\Impexp\Utility\ImportExportUtility::afterImportExportInitialisation` - :php:`TYPO3\CMS\Install\Service\SqlExpectedSchemaService::tablesDefinitionIsBeingBuilt` -- :php:`TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider::PostProcessTreeData` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::getSystemInformation` -- :php:`TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem::loadMessages` +- :php:`TYPO3\CMS\Lang\Service\TranslationService::postProcessMirrorUrl` +- :php:`TYPO3\CMS\Linkvalidator\LinkAnalyzer::beforeAnalyzeRecord` +- :php:`TYPO3\CMS\Seo\Canonical\CanonicalGenerator::beforeGeneratingCanonical` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss` +- :php:`TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss` Impact diff --git a/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php b/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php index 75e03915564a..9083a654ddbe 100644 --- a/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php +++ b/typo3/sysext/core/Tests/Unit/Configuration/Loader/PageTsConfigLoaderTest.php @@ -15,6 +15,9 @@ namespace TYPO3\CMS\Core\Tests\Unit\Configuration\Loader; * The TYPO3 project - inspiring people to share! */ +use Prophecy\Argument; +use Psr\EventDispatcher\EventDispatcherInterface; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Configuration\Loader\PageTsConfigLoader; use TYPO3\TestingFramework\Core\Unit\UnitTestCase; @@ -29,7 +32,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'default' => $GLOBALS['TYPO3_CONF_VARS']['BE']['defaultPageTSconfig'] ]; $expectedString = implode('"\n[GLOBAL]\n"', $expected); - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); + $event = new ModifyLoadedPageTsConfigEvent($expected, []); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); $result = $subject->collect([]); self::assertSame($expected, $result); @@ -48,7 +54,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'page_27' => '', ]; $rootLine = [['uid' => 0, 'pid' => 0], ['uid' => 13, 'TSconfig' => 'waiting for = love'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); } @@ -66,7 +75,10 @@ class PageTsConfigLoaderTest extends UnitTestCase 'page_27' => '', ]; $rootLine = [['uid' => 13, 'TSconfig' => 'waiting for = love', 'tsconfig_includes' => 'EXT:core/Tests/Unit/Configuration/Loader/Fixtures/included.typoscript'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); } @@ -83,7 +95,10 @@ class PageTsConfigLoaderTest extends UnitTestCase ]; $expectedString = implode("\n[GLOBAL]\n", $expected); $rootLine = [['uid' => 13, 'TSconfig' => 'waiting for = love', 'tsconfig_includes' => 'EXT:core/Tests/Unit/Configuration/Loader/Fixtures/me_does_not_exist.typoscript'], ['uid' => 27, 'TSconfig' => '']]; - $subject = new PageTsConfigLoader(); + $eventDispatcher = $this->prophesize(EventDispatcherInterface::class); + $event = new ModifyLoadedPageTsConfigEvent($expected, $rootLine); + $eventDispatcher->dispatch(Argument::type(ModifyLoadedPageTsConfigEvent::class))->willReturn($event); + $subject = new PageTsConfigLoader($eventDispatcher->reveal()); $result = $subject->collect($rootLine); self::assertSame($expected, $result); diff --git a/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php b/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php index 1d239c58f40e..287347e97d40 100644 --- a/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php +++ b/typo3/sysext/extbase/Classes/SignalSlot/Dispatcher.php @@ -17,9 +17,18 @@ namespace TYPO3\CMS\Extbase\SignalSlot; */ use Psr\Log\LoggerInterface; +use TYPO3\CMS\Backend\Authentication\Event\SwitchUserEvent; use TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent; use TYPO3\CMS\Backend\Backend\ToolbarItems\SystemInformationToolbarItem; +use TYPO3\CMS\Backend\Controller\EditDocumentController; +use TYPO3\CMS\Backend\Controller\Event\AfterFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\Controller\Event\BeforeFormEnginePageInitializedEvent; +use TYPO3\CMS\Backend\LoginProvider\Event\ModifyPageLayoutOnLoginProviderSelectionEvent; +use TYPO3\CMS\Backend\LoginProvider\UsernamePasswordLoginProvider; +use TYPO3\CMS\Backend\Utility\BackendUtility; +use TYPO3\CMS\Beuser\Controller\BackendUserController; use TYPO3\CMS\Core\Configuration\Event\AfterTcaCompilationEvent; +use TYPO3\CMS\Core\Configuration\Event\ModifyLoadedPageTsConfigEvent; use TYPO3\CMS\Core\Database\Event\AlterTableDefinitionStatementsEvent; use TYPO3\CMS\Core\Database\ReferenceIndex; use TYPO3\CMS\Core\Database\SoftReferenceIndex; @@ -78,6 +87,18 @@ use TYPO3\CMS\Core\Tree\Event\ModifyTreeDataEvent; use TYPO3\CMS\Core\Tree\TableConfiguration\DatabaseTreeDataProvider; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Extbase\Object\ObjectManagerInterface; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; +use TYPO3\CMS\Impexp\Utility\ImportExportUtility; +use TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent; +use TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent; +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; +use TYPO3\CMS\Seo\Canonical\CanonicalGenerator; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; +use TYPO3\CMS\Workspaces\Service\GridDataService; /** * A dispatcher which dispatches signals by calling its registered slot methods @@ -174,9 +195,40 @@ class Dispatcher implements \TYPO3\CMS\Core\SingletonInterface DatabaseTreeDataProvider::class => [ 'PostProcessTreeData' => ModifyTreeDataEvent::class, ], + BackendUserController::class => [ + 'switchUser' => SwitchUserEvent::class + ], + BackendUtility::class => [ + 'getPagesTSconfigPreInclude' => ModifyLoadedPageTsConfigEvent::class + ], + EditDocumentController::class => [ + 'preInitAfter' => BeforeFormEnginePageInitializedEvent::class, + 'initAfter' => AfterFormEnginePageInitializedEvent::class, + ], SystemInformationToolbarItem::class => [ 'getSystemInformation' => SystemInformationToolbarCollectorEvent::class, 'loadMessages' => SystemInformationToolbarCollectorEvent::class + ], + UsernamePasswordLoginProvider::class => [ + 'getPageRenderer' => ModifyPageLayoutOnLoginProviderSelectionEvent::class + ], + 'TYPO3\\CMS\\Lang\\Service\\TranslationService' => [ + 'postProcessMirrorUrl' => ModifyLanguagePackRemoteBaseUrlEvent::class + ], + LinkAnalyzer::class => [ + 'beforeAnalyzeRecord' => BeforeRecordIsAnalyzedEvent::class + ], + ImportExportUtility::class => [ + 'afterImportExportInitialisation' => BeforeImportEvent::class + ], + CanonicalGenerator::class => [ + 'beforeGeneratingCanonical' => ModifyUrlForCanonicalTagEvent::class + ], + GridDataService::class => [ + GridDataService::SIGNAL_GenerateDataArray_BeforeCaching => AfterCompiledCacheableDataForWorkspaceEvent::class, + GridDataService::SIGNAL_GenerateDataArray_PostProcesss => AfterDataGeneratedForWorkspaceEvent::class, + GridDataService::SIGNAL_GetDataArray_PostProcesss => GetVersionedDataEvent::class, + GridDataService::SIGNAL_SortDataArray_PostProcesss => SortVersionedDataEvent::class, ] ]; diff --git a/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..04d9c80afe9e --- /dev/null +++ b/typo3/sysext/impexp/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,49 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Impexp\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; +use TYPO3\CMS\Impexp\Utility\ImportExportUtility; + +/** + * This class provides a replacement for all existing signals in EXT:impexp of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function emitAfterImportExportInitialisationSignal(BeforeImportEvent $event): void + { + $this->signalSlotDispatcher->dispatch( + ImportExportUtility::class, + 'afterImportExportInitialisation', + [$event->getImport()] + ); + } +} diff --git a/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php b/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php new file mode 100644 index 000000000000..608fd995fdf1 --- /dev/null +++ b/typo3/sysext/impexp/Classes/Event/BeforeImportEvent.php @@ -0,0 +1,39 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Impexp\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Impexp\Import; + +/** + * This event is triggered when an import file is about to be imported + */ +final class BeforeImportEvent +{ + /** + * @var Import + */ + private $import; + + public function __construct(Import $import) + { + $this->import = $import; + } + + public function getImport(): Import + { + return $this->import; + } +} diff --git a/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php b/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php index 485e41b00628..864a2c6f3f9b 100644 --- a/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php +++ b/typo3/sysext/impexp/Classes/Utility/ImportExportUtility.php @@ -14,9 +14,10 @@ namespace TYPO3\CMS\Impexp\Utility; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Log\LogManager; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; +use TYPO3\CMS\Impexp\Event\BeforeImportEvent; use TYPO3\CMS\Impexp\Import; /** @@ -31,6 +32,16 @@ class ImportExportUtility */ protected $import; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * @return Import|null */ @@ -59,7 +70,7 @@ class ImportExportUtility $this->import = GeneralUtility::makeInstance(Import::class); $this->import->init(); - $this->emitAfterImportExportInitialisationSignal($this->import); + $this->eventDispatcher->dispatch(new BeforeImportEvent($this->import)); $importResponse = 0; if ($file && @is_file($file)) { @@ -84,26 +95,4 @@ class ImportExportUtility } return $importResponse; } - - /** - * Get the SignalSlot dispatcher - * - * @return Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return GeneralUtility::makeInstance(Dispatcher::class); - } - - /** - * Emits a signal after initialization - * - * @param Import $import - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotException - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotReturnException - */ - protected function emitAfterImportExportInitialisationSignal(Import $import) - { - $this->getSignalSlotDispatcher()->dispatch(__CLASS__, 'afterImportExportInitialisation', [$import]); - } } diff --git a/typo3/sysext/impexp/Configuration/Services.yaml b/typo3/sysext/impexp/Configuration/Services.yaml index 825fd786666c..4099126b0450 100644 --- a/typo3/sysext/impexp/Configuration/Services.yaml +++ b/typo3/sysext/impexp/Configuration/Services.yaml @@ -6,3 +6,14 @@ services: TYPO3\CMS\Impexp\: resource: '../Classes/*' + + TYPO3\CMS\Impexp\Utility\ImportExportUtility: + public: true + + # Listener for old Signal Slots + TYPO3\CMS\Impexp\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'emitAfterImportExportInitialisationSignal' + event: TYPO3\CMS\Impexp\Event\BeforeImportEvent diff --git a/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..03c21602830a --- /dev/null +++ b/typo3/sysext/install/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,54 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Install\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Core\Http\Uri; +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent; + +/** + * This class provides a replacement for all existing signals in EXT:install of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onLanguagePackProcessMirrorUrl(ModifyLanguagePackRemoteBaseUrlEvent $event): void + { + $languagePackBaseUrl = (string)$event->getBaseUrl(); + $this->signalSlotDispatcher->dispatch( + 'TYPO3\\CMS\\Lang\\Service\\TranslationService', + 'postProcessMirrorUrl', + [ + 'extensionKey' => $event->getPackageKey(), + 'mirrorUrl' => &$languagePackBaseUrl, + ] + ); + $event->setBaseUrl(new Uri($languagePackBaseUrl)); + } +} diff --git a/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php b/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php new file mode 100644 index 000000000000..64e3dd879403 --- /dev/null +++ b/typo3/sysext/install/Classes/Service/Event/ModifyLanguagePackRemoteBaseUrlEvent.php @@ -0,0 +1,55 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Install\Service\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use Psr\Http\Message\UriInterface; + +/** + * Event to modify the main URL of a language + */ +final class ModifyLanguagePackRemoteBaseUrlEvent +{ + /** + * @var UriInterface + */ + private $baseUrl; + + /** + * @var string + */ + private $packageKey; + + public function __construct(UriInterface $baseUrl, string $packageKey) + { + $this->baseUrl = $baseUrl; + $this->packageKey = $packageKey; + } + + public function getBaseUrl(): UriInterface + { + return $this->baseUrl; + } + + public function setBaseUrl(UriInterface $baseUrl): void + { + $this->baseUrl = $baseUrl; + } + + public function getPackageKey(): string + { + return $this->packageKey; + } +} diff --git a/typo3/sysext/install/Classes/Service/LanguagePackService.php b/typo3/sysext/install/Classes/Service/LanguagePackService.php index 480411ab0a57..89c73fca93bd 100644 --- a/typo3/sysext/install/Classes/Service/LanguagePackService.php +++ b/typo3/sysext/install/Classes/Service/LanguagePackService.php @@ -15,16 +15,17 @@ namespace TYPO3\CMS\Install\Service; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Symfony\Component\Finder\Finder; use TYPO3\CMS\Core\Configuration\Features; use TYPO3\CMS\Core\Core\Environment; +use TYPO3\CMS\Core\Http\Uri; use TYPO3\CMS\Core\Localization\Locales; use TYPO3\CMS\Core\Package\PackageManager; use TYPO3\CMS\Core\Registry; use TYPO3\CMS\Core\Utility\ExtensionManagementUtility; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\PathUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; /** * Service class handling language pack details @@ -44,11 +45,17 @@ class LanguagePackService */ protected $registry; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + private const DEFAULT_LANGUAGE_PACK_URL = 'https://typo3.org/fileadmin/ter/'; private const BETA_LANGUAGE_PACK_URL = 'https://beta-translation.typo3.org/fileadmin/ter/'; - public function __construct() + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher ?? GeneralUtility::getContainer()->get(EventDispatcherInterface::class); $this->locales = GeneralUtility::makeInstance(Locales::class); $this->registry = GeneralUtility::makeInstance(Registry::class); } @@ -232,17 +239,9 @@ class LanguagePackService $languagePackBaseUrl = self::BETA_LANGUAGE_PACK_URL; } - // Allow to modify the base url on the fly by calling a signal - $signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - $signalSlotDispatcher->dispatch( - 'TYPO3\\CMS\\Lang\\Service\\TranslationService', - 'postProcessMirrorUrl', - [ - 'extensionKey' => $key, - 'mirrorUrl' => &$languagePackBaseUrl, - ] - ); - + // Allow to modify the base url on the fly + $event = $this->eventDispatcher->dispatch(new Event\ModifyLanguagePackRemoteBaseUrlEvent(new Uri($languagePackBaseUrl), $key)); + $languagePackBaseUrl = $event->getBaseUrl(); $path = ExtensionManagementUtility::extPath($key); $majorVersion = explode('.', TYPO3_branch)[0]; if (strpos($path, '/sysext/') !== false) { diff --git a/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php b/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php index c3b01d1381c9..887b0689306f 100644 --- a/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php +++ b/typo3/sysext/install/Configuration/ExtensionScanner/Php/ClassConstantMatcher.php @@ -357,5 +357,29 @@ return [ 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_BeforeCaching' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GenerateDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_GetDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], + ], + 'TYPO3\CMS\Workspaces\Service\GridDataService::SIGNAL_SortDataArray_PostProcesss' => [ + 'restFiles' => [ + 'Feature-89733-NewPSR-14EventsForExistingSignalSlotsInCoreExtension.rst', + 'Deprecation-89733-SignalSlotsInCoreExtensionMigratedToPSR-14Events.rst', + ], ] ]; diff --git a/typo3/sysext/install/Configuration/Services.yaml b/typo3/sysext/install/Configuration/Services.yaml index 15f76349eb9f..001682a48bb9 100644 --- a/typo3/sysext/install/Configuration/Services.yaml +++ b/typo3/sysext/install/Configuration/Services.yaml @@ -8,3 +8,12 @@ services: identifier: 'install/show-latest-errors' method: 'appendMessage' event: TYPO3\CMS\Backend\Backend\Event\SystemInformationToolbarCollectorEvent + TYPO3\CMS\Install\Compatibility\SlotReplacement: + autowire: true + autoconfigure: true + public: false + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onLanguagePackProcessMirrorUrl' + event: TYPO3\CMS\Install\Service\Event\ModifyLanguagePackRemoteBaseUrlEvent diff --git a/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..9bf46d91cdcd --- /dev/null +++ b/typo3/sysext/linkvalidator/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,57 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Linkvalidator\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent; +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; + +/** + * This class provides a replacement for all existing signals in EXT:linkvalidator of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function beforeLinkAnalyzerRecordIsAnalyzed(BeforeRecordIsAnalyzedEvent $event): void + { + [$results, $record] = $this->signalSlotDispatcher->dispatch( + LinkAnalyzer::class, + 'beforeAnalyzeRecord', + [ + $event->getResults(), + $event->getRecord(), + $event->getTableName(), + $event->getFields(), + $event->getLinkAnalyzer() + ] + ); + $event->setRecord($record); + $event->setResults($results); + } +} diff --git a/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php b/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php new file mode 100644 index 000000000000..52ce7443cbbe --- /dev/null +++ b/typo3/sysext/linkvalidator/Classes/Event/BeforeRecordIsAnalyzedEvent.php @@ -0,0 +1,94 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Linkvalidator\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Linkvalidator\LinkAnalyzer; + +/** + * Event that is fired to modify results (= add results) or modify the record before the linkanalyzer analyzes + * the record. + */ +final class BeforeRecordIsAnalyzedEvent +{ + /** + * @var string + */ + private $tableName; + + /** + * @var array + */ + private $record; + + /** + * @var array + */ + private $fields; + + /** + * @var array + */ + private $results; + + /** + * @var LinkAnalyzer + */ + private $linkAnalyzer; + + public function __construct(string $tableName, array $record, array $fields, LinkAnalyzer $linkAnalyzer, array $results) + { + $this->tableName = $tableName; + $this->record = $record; + $this->fields = $fields; + $this->linkAnalyzer = $linkAnalyzer; + $this->results = $results; + } + + public function getTableName(): string + { + return $this->tableName; + } + + public function getRecord(): array + { + return $this->record; + } + + public function setRecord(array $record): void + { + $this->record = $record; + } + + public function getFields(): array + { + return $this->fields; + } + + public function getResults(): array + { + return $this->results; + } + + public function setResults(array $results): void + { + $this->results = $results; + } + + public function getLinkAnalyzer(): LinkAnalyzer + { + return $this->linkAnalyzer; + } +} diff --git a/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php b/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php index c7a8006b721f..21da1eb79863 100644 --- a/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php +++ b/typo3/sysext/linkvalidator/Classes/LinkAnalyzer.php @@ -14,6 +14,7 @@ namespace TYPO3\CMS\Linkvalidator; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Backend\Utility\BackendUtility; use TYPO3\CMS\Core\Database\Connection; use TYPO3\CMS\Core\Database\ConnectionPool; @@ -94,10 +95,13 @@ class LinkAnalyzer protected $tsConfig = []; /** - * Fill hookObjectsArr with different link types and possible XClasses. + * @var EventDispatcherInterface */ - public function __construct() + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) { + $this->eventDispatcher = $eventDispatcher; $this->getLanguageService()->includeLLFile('EXT:linkvalidator/Resources/Private/Language/Module/locallang.xlf'); } @@ -268,7 +272,10 @@ class LinkAnalyzer */ public function analyzeRecord(array &$results, $table, array $fields, array $record) { - list($results, $record) = $this->emitBeforeAnalyzeRecordSignal($results, $record, $table, $fields); + $event = new Event\BeforeRecordIsAnalyzedEvent($table, $record, $fields, $this, $results); + $this->eventDispatcher->dispatch($event); + $results = $event->getResults(); + $record = $event->getRecord(); // Put together content of all relevant fields $haystack = ''; @@ -572,40 +579,6 @@ class LinkAnalyzer return false; } - /** - * Emits a signal before the record is analyzed - * - * @param array $results Array of broken links - * @param array $record Record to analyze - * @param string $table Table name of the record - * @param array $fields Array of fields to analyze - * @return array - */ - protected function emitBeforeAnalyzeRecordSignal($results, $record, $table, $fields) - { - return $this->getSignalSlotDispatcher()->dispatch( - self::class, - 'beforeAnalyzeRecord', - [$results, $record, $table, $fields, $this] - ); - } - - /** - * @return \TYPO3\CMS\Extbase\SignalSlot\Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return $this->getObjectManager()->get(\TYPO3\CMS\Extbase\SignalSlot\Dispatcher::class); - } - - /** - * @return \TYPO3\CMS\Extbase\Object\ObjectManager - */ - protected function getObjectManager() - { - return GeneralUtility::makeInstance(\TYPO3\CMS\Extbase\Object\ObjectManager::class); - } - /** * @return LanguageService */ diff --git a/typo3/sysext/linkvalidator/Configuration/Services.yaml b/typo3/sysext/linkvalidator/Configuration/Services.yaml index 90fb8b53ff14..ac4cecda0067 100644 --- a/typo3/sysext/linkvalidator/Configuration/Services.yaml +++ b/typo3/sysext/linkvalidator/Configuration/Services.yaml @@ -24,3 +24,10 @@ services: identifier: 'rte-check-link-to-file' event: TYPO3\CMS\Core\Html\Event\BrokenLinkAnalysisEvent method: 'checkFileLink' + + TYPO3\CMS\Linkvalidator\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'beforeLinkAnalyzerRecordIsAnalyzed' + event: TYPO3\CMS\Linkvalidator\Event\BeforeRecordIsAnalyzedEvent diff --git a/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php b/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php index e3cd3cafff80..032842db047b 100644 --- a/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php +++ b/typo3/sysext/linkvalidator/Tests/Functional/LinkAnalyzerTest.php @@ -16,6 +16,7 @@ namespace TYPO3\CMS\Linkvalidator\Tests\Functional; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Core\Bootstrap; use TYPO3\CMS\Linkvalidator\LinkAnalyzer; use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase; @@ -34,7 +35,6 @@ class LinkAnalyzerTest extends FunctionalTestCase protected function setUp(): void { parent::setUp(); - Bootstrap::initializeLanguageObject(); } @@ -110,7 +110,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -172,7 +172,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -234,7 +234,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); @@ -295,7 +295,7 @@ class LinkAnalyzerTest extends FunctionalTestCase $this->importDataSet($inputFile); - $linkAnalyzer = new LinkAnalyzer(); + $linkAnalyzer = new LinkAnalyzer($this->prophesize(EventDispatcherInterface::class)->reveal()); $linkAnalyzer->init($searchFields, $pidList, $tsConfig); $linkAnalyzer->getLinkStatistics($config); diff --git a/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php b/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php index 6b5b4ca7f1cb..c87a1b7d8ff7 100644 --- a/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php +++ b/typo3/sysext/seo/Classes/Canonical/CanonicalGenerator.php @@ -16,11 +16,12 @@ namespace TYPO3\CMS\Seo\Canonical; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use TYPO3\CMS\Core\Domain\Repository\PageRepository; use TYPO3\CMS\Core\Utility\GeneralUtility; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Frontend\Controller\TypoScriptFrontendController; use TYPO3\CMS\Frontend\Utility\CanonicalizationUtility; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; /** * Class to add the canonical tag to the page @@ -40,38 +41,21 @@ class CanonicalGenerator protected $pageRepository; /** - * @var Dispatcher + * @var EventDispatcherInterface */ - protected $signalSlotDispatcher; + protected $eventDispatcher; - /** - * CanonicalGenerator constructor - * - * @param TypoScriptFrontendController $typoScriptFrontendController - * @param Dispatcher $signalSlotDispatcher - */ - public function __construct(TypoScriptFrontendController $typoScriptFrontendController = null, Dispatcher $signalSlotDispatcher = null) + public function __construct(TypoScriptFrontendController $typoScriptFrontendController = null, EventDispatcherInterface $eventDispatcher = null) { - if ($typoScriptFrontendController === null) { - $typoScriptFrontendController = $this->getTypoScriptFrontendController(); - } - if ($signalSlotDispatcher === null) { - $signalSlotDispatcher = GeneralUtility::makeInstance(Dispatcher::class); - } - $this->typoScriptFrontendController = $typoScriptFrontendController; - $this->signalSlotDispatcher = $signalSlotDispatcher; + $this->eventDispatcher = $eventDispatcher ?? GeneralUtility::getContainer()->get(EventDispatcherInterface::class); + $this->typoScriptFrontendController = $typoScriptFrontendController ?? $this->getTypoScriptFrontendController(); $this->pageRepository = GeneralUtility::makeInstance(PageRepository::class); } - /** - * @return string - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotException - * @throws \TYPO3\CMS\Extbase\SignalSlot\Exception\InvalidSlotReturnException - */ public function generate(): string { - $href = ''; - $this->signalSlotDispatcher->dispatch(self::class, 'beforeGeneratingCanonical', [&$href]); + $event = $this->eventDispatcher->dispatch(new ModifyUrlForCanonicalTagEvent('')); + $href = $event->getUrl(); if (empty($href) && (int)$this->typoScriptFrontendController->page['no_index'] === 1) { return ''; diff --git a/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..756b45fbdbe9 --- /dev/null +++ b/typo3/sysext/seo/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,51 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Seo\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Seo\Canonical\CanonicalGenerator; +use TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent; + +/** + * This class provides a replacement for all existing signals in EXT:seo of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function beforeGeneratingCanonical(ModifyUrlForCanonicalTagEvent $event): void + { + $href = $event->getUrl(); + $this->signalSlotDispatcher->dispatch( + CanonicalGenerator::class, + 'beforeGeneratingCanonical', + [&$href] + ); + $event->setUrl($href); + } +} diff --git a/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php b/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php new file mode 100644 index 000000000000..4c634e4c0c03 --- /dev/null +++ b/typo3/sysext/seo/Classes/Event/ModifyUrlForCanonicalTagEvent.php @@ -0,0 +1,43 @@ +<?php +declare(strict_types = 1); + +namespace TYPO3\CMS\Seo\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +/** + * PSR-14 to alter (or empty) a canonical URL for the href="" attribute of a canonical URL. + */ +final class ModifyUrlForCanonicalTagEvent +{ + /** + * @var string + */ + private $url; + + public function __construct(string $url) + { + $this->url = $url; + } + + public function getUrl(): string + { + return $this->url; + } + + public function setUrl(string $url): void + { + $this->url = $url; + } +} diff --git a/typo3/sysext/seo/Configuration/Services.yaml b/typo3/sysext/seo/Configuration/Services.yaml index 484cb85e9651..79ba4065337e 100644 --- a/typo3/sysext/seo/Configuration/Services.yaml +++ b/typo3/sysext/seo/Configuration/Services.yaml @@ -6,3 +6,10 @@ services: TYPO3\CMS\Seo\: resource: '../Classes/*' + + TYPO3\CMS\Seo\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'beforeGeneratingCanonical' + event: TYPO3\CMS\Seo\Event\ModifyUrlForCanonicalTagEvent diff --git a/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php b/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php new file mode 100644 index 000000000000..88be5cd7c7a2 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Compatibility/SlotReplacement.php @@ -0,0 +1,87 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Compatibility; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Extbase\SignalSlot\Dispatcher as SignalSlotDispatcher; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * This class provides a replacement for all existing signals in EXT:workspaces of TYPO3 Core, which now act as a + * simple wrapper for PSR-14 events with a simple ("first prioritized") listener implementation. + * + * @internal Please note that this class will likely be removed in TYPO3 v11, and Extension Authors should + * switch to PSR-14 event listeners. + */ +class SlotReplacement +{ + /** + * @var SignalSlotDispatcher + */ + protected $signalSlotDispatcher; + + public function __construct(SignalSlotDispatcher $signalSlotDispatcher) + { + $this->signalSlotDispatcher = $signalSlotDispatcher; + } + + public function onGenerateDataArrayBeforeCaching(AfterCompiledCacheableDataForWorkspaceEvent $event): void + { + [$obj, $dataArray, $versions] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GenerateDataArray_BeforeCaching, + [$event->getGridService(), $event->getData(), $event->getVersions()] + ); + $event->setData($dataArray); + $event->setVersions($versions); + } + + public function onGenerateDataArrayPostProcessing(AfterDataGeneratedForWorkspaceEvent $event): void + { + [$obj, $dataArray] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GenerateDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getVersions()] + ); + $event->setData($dataArray); + } + + public function onGetDataPostProcessing(GetVersionedDataEvent $event): void + { + [$obj, $dataArray, $start, $limit, $dataArrayPart] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_GetDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getStart(), $event->getLimit(), $event->getDataArrayPart()] + ); + $event->setData($dataArray); + $event->setDataArrayPart($dataArrayPart); + } + + public function onSortDataPostProcessing(SortVersionedDataEvent $event): void + { + [$obj, $dataArray, $sortingColumn, $sortingDirection] = $this->signalSlotDispatcher->dispatch( + GridDataService::class, + GridDataService::SIGNAL_SortDataArray_PostProcesss, + [$event->getGridService(), $event->getData(), $event->getSortColumn(), $event->getSortDirection()] + ); + $event->setData($dataArray); + $event->setSortColumn($sortingColumn); + $event->setSortDirection($sortingDirection); + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php b/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php new file mode 100644 index 000000000000..db81d8605b91 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/AfterCompiledCacheableDataForWorkspaceEvent.php @@ -0,0 +1,71 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all chacheable data of versions of a workspace. + */ +final class AfterCompiledCacheableDataForWorkspaceEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $versions; + + public function __construct(GridDataService $gridService, array $data, array $versions) + { + $this->gridService = $gridService; + $this->data = $data; + $this->versions = $versions; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getVersions(): array + { + return $this->versions; + } + + public function setVersions(array $versions): void + { + $this->versions = $versions; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php b/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php new file mode 100644 index 000000000000..70ba704e6ca9 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/AfterDataGeneratedForWorkspaceEvent.php @@ -0,0 +1,71 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all data of versions of a workspace. + */ +final class AfterDataGeneratedForWorkspaceEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $versions; + + public function __construct(GridDataService $gridService, array $data, array $versions) + { + $this->gridService = $gridService; + $this->data = $data; + $this->versions = $versions; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getVersions(): array + { + return $this->versions; + } + + public function setVersions(array $versions): void + { + $this->versions = $versions; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php b/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php new file mode 100644 index 000000000000..c9b5fc933793 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/GetVersionedDataEvent.php @@ -0,0 +1,95 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module to find all data of versions of a workspace. + * In comparison to AfterDataGeneratedForWorkspaceEvent, this one contains the + * cleaned / prepared data with an optional limit applied depending on the view. + */ +final class GetVersionedDataEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var array + */ + private $dataArrayPart; + + /** + * @var int + */ + private $start; + + /** + * @var int + */ + private $limit; + + public function __construct(GridDataService $gridService, array $data, int $start, int $limit, array $dataArrayPart) + { + $this->gridService = $gridService; + $this->data = $data; + $this->start = $start; + $this->limit = $limit; + $this->dataArrayPart = $dataArrayPart; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getDataArrayPart(): array + { + return $this->dataArrayPart; + } + + public function setDataArrayPart(array $dataArrayPart): void + { + $this->dataArrayPart = $dataArrayPart; + } + + public function getStart(): int + { + return $this->start; + } + + public function getLimit(): int + { + return $this->limit; + } +} diff --git a/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php b/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php new file mode 100644 index 000000000000..b23dfb6c7713 --- /dev/null +++ b/typo3/sysext/workspaces/Classes/Event/SortVersionedDataEvent.php @@ -0,0 +1,87 @@ +<?php +declare(strict_types = 1); +namespace TYPO3\CMS\Workspaces\Event; + +/* + * This file is part of the TYPO3 CMS project. + * + * It is free software; you can redistribute it and/or modify it under + * the terms of the GNU General Public License, either version 2 + * of the License, or any later version. + * + * For the full copyright and license information, please read the + * LICENSE.txt file that was distributed with this source code. + * + * The TYPO3 project - inspiring people to share! + */ + +use TYPO3\CMS\Workspaces\Service\GridDataService; + +/** + * Used in the workspaces module after sorting all data for versions of a workspace. + */ +final class SortVersionedDataEvent +{ + /** + * @var GridDataService + */ + private $gridService; + + /** + * @var array + */ + private $data; + + /** + * @var string + */ + private $sortColumn; + + /** + * @var string + */ + private $sortDirection; + + public function __construct(GridDataService $gridService, array $data, string $sortColumn, string $sortDirection) + { + $this->gridService = $gridService; + $this->data = $data; + $this->sortColumn = $sortColumn; + $this->sortDirection = $sortDirection; + } + + public function getGridService(): GridDataService + { + return $this->gridService; + } + + public function getData(): array + { + return $this->data; + } + + public function setData(array $data): void + { + $this->data = $data; + } + + public function getSortColumn(): string + { + return $this->sortColumn; + } + + public function setSortColumn(string $sortColumn): void + { + $this->sortColumn = $sortColumn; + } + + public function getSortDirection(): string + { + return $this->sortDirection; + } + + public function setSortDirection(string $sortDirection): void + { + $this->sortDirection = $sortDirection; + } +} diff --git a/typo3/sysext/workspaces/Classes/Service/GridDataService.php b/typo3/sysext/workspaces/Classes/Service/GridDataService.php index cf22174f5786..ba7469136d37 100644 --- a/typo3/sysext/workspaces/Classes/Service/GridDataService.php +++ b/typo3/sysext/workspaces/Classes/Service/GridDataService.php @@ -14,6 +14,7 @@ namespace TYPO3\CMS\Workspaces\Service; * The TYPO3 project - inspiring people to share! */ +use Psr\EventDispatcher\EventDispatcherInterface; use Psr\Log\LoggerAwareInterface; use Psr\Log\LoggerAwareTrait; use TYPO3\CMS\Backend\Configuration\TranslationConfigurationProvider; @@ -24,8 +25,11 @@ use TYPO3\CMS\Core\Imaging\IconFactory; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Versioning\VersionState; use TYPO3\CMS\Extbase\Object\ObjectManager; -use TYPO3\CMS\Extbase\SignalSlot\Dispatcher; use TYPO3\CMS\Workspaces\Domain\Model\CombinedRecord; +use TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent; +use TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent; +use TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent; use TYPO3\CMS\Workspaces\Preview\PreviewUriBuilder; /** @@ -35,9 +39,21 @@ class GridDataService implements LoggerAwareInterface { use LoggerAwareTrait; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GenerateDataArray_BeforeCaching = 'generateDataArray.beforeCaching'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GenerateDataArray_PostProcesss = 'generateDataArray.postProcess'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_GetDataArray_PostProcesss = 'getDataArray.postProcess'; + /** + * @deprecated will be removed in TYPO3 v11 in favor of PSR-14 events + */ const SIGNAL_SortDataArray_PostProcesss = 'sortDataArray.postProcess'; const GridColumn_Collection = 'Workspaces_Collection'; @@ -84,6 +100,16 @@ class GridDataService implements LoggerAwareInterface */ protected $integrityService; + /** + * @var EventDispatcherInterface + */ + protected $eventDispatcher; + + public function __construct(EventDispatcherInterface $eventDispatcher) + { + $this->eventDispatcher = $eventDispatcher; + } + /** * Generates grid list array from given versions. * @@ -214,9 +240,12 @@ class GridDataService implements LoggerAwareInterface } } } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, array $versions) - list($this->dataArray, $versions) = $this->emitSignal(self::SIGNAL_GenerateDataArray_BeforeCaching, $this->dataArray, $versions); + + // Trigger a PSR-14 event + $event = new AfterCompiledCacheableDataForWorkspaceEvent($this, $this->dataArray, $versions); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + $versions = $event->getVersions(); // Enrich elements after everything has been processed: foreach ($this->dataArray as &$element) { $identifier = $element['table'] . ':' . $element['t3ver_oid']; @@ -227,9 +256,11 @@ class GridDataService implements LoggerAwareInterface } $this->setDataArrayIntoCache($versions, $filterTxt); } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, array $versions) - list($this->dataArray) = $this->emitSignal(self::SIGNAL_GenerateDataArray_PostProcesss, $this->dataArray, $versions); + + // Trigger a PSR-14 event + $event = new AfterDataGeneratedForWorkspaceEvent($this, $this->dataArray, $versions); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); $this->sortDataArray(); $this->resolveDataArrayDependencies(); } @@ -277,9 +308,11 @@ class GridDataService implements LoggerAwareInterface } } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, $start, $limit, array $dataArrayPart) - list($this->dataArray, $start, $limit, $dataArrayPart) = $this->emitSignal(self::SIGNAL_GetDataArray_PostProcesss, $this->dataArray, $start, $limit, $dataArrayPart); + // Trigger a PSR-14 event + $event = new GetVersionedDataEvent($this, $this->dataArray, $start, $limit, $dataArrayPart); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + return $event->getDataArrayPart(); return $dataArrayPart; } @@ -375,9 +408,12 @@ class GridDataService implements LoggerAwareInterface } else { $this->logger->critical('Try to sort "' . $this->sort . '" in "\\TYPO3\\CMS\\Workspaces\\Service\\GridDataService::sortDataArray" but $this->dataArray is empty! This might be the bug #26422 which could not be reproduced yet.'); } - // Suggested slot method: - // methodName(\TYPO3\CMS\Workspaces\Service\GridDataService $gridData, array $dataArray, $sortColumn, $sortDirection) - list($this->dataArray, $this->sort, $this->sortDir) = $this->emitSignal(self::SIGNAL_SortDataArray_PostProcesss, $this->dataArray, $this->sort, $this->sortDir); + // Trigger an event for extensibility + $event = new SortVersionedDataEvent($this, $this->dataArray, $this->sort, $this->sortDir); + $this->eventDispatcher->dispatch($event); + $this->dataArray = $event->getData(); + $this->sort = $event->getSortColumn(); + $this->sortDir = $event->getSortDirection(); } /** @@ -623,22 +659,6 @@ class GridDataService implements LoggerAwareInterface return $this->integrityService; } - /** - * Emits a signal to be handled by any registered slots. - * - * @param string $signalName Name of the signal - * @param array|mixed[] $arguments - * @return array - */ - protected function emitSignal($signalName, ...$arguments) - { - // Arguments are always ($this, [method argument], [method argument], ...) - $signalArguments = $arguments; - array_unshift($signalArguments, $this); - $slotReturn = $this->getSignalSlotDispatcher()->dispatch(GridDataService::class, $signalName, $signalArguments); - return array_slice($slotReturn, 1); - } - /** * @return Dependency\CollectionService */ @@ -655,14 +675,6 @@ class GridDataService implements LoggerAwareInterface return $this->getObjectManager()->get(AdditionalColumnService::class); } - /** - * @return Dispatcher - */ - protected function getSignalSlotDispatcher() - { - return $this->getObjectManager()->get(Dispatcher::class); - } - /** * @return ObjectManager */ diff --git a/typo3/sysext/workspaces/Configuration/Services.yaml b/typo3/sysext/workspaces/Configuration/Services.yaml index a2a64f288aa8..9c62f633de7c 100644 --- a/typo3/sysext/workspaces/Configuration/Services.yaml +++ b/typo3/sysext/workspaces/Configuration/Services.yaml @@ -11,3 +11,25 @@ services: class: TYPO3\CMS\Core\Cache\Frontend\FrontendInterface factory: ['@TYPO3\CMS\Core\Cache\CacheManager', 'getCache'] arguments: ['workspaces_cache'] + + TYPO3\CMS\Workspaces\Service\GridDataService: + public: true + + TYPO3\CMS\Workspaces\Compatibility\SlotReplacement: + tags: + - name: event.listener + identifier: 'legacy-slot' + method: 'onGenerateDataArrayBeforeCaching' + event: TYPO3\CMS\Workspaces\Event\AfterCompiledCacheableDataForWorkspaceEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onGenerateDataArrayPostProcessing' + event: TYPO3\CMS\Workspaces\Event\AfterDataGeneratedForWorkspaceEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onGetDataPostProcessing' + event: TYPO3\CMS\Workspaces\Event\GetVersionedDataEvent + - name: event.listener + identifier: 'legacy-slot' + method: 'onSortDataPostProcessing' + event: TYPO3\CMS\Workspaces\Event\SortVersionedDataEvent diff --git a/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php b/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php index bb1d88cfa0aa..3dfb34e9a38b 100644 --- a/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php +++ b/typo3/sysext/workspaces/Tests/Unit/Controller/Remote/RemoteServerTest.php @@ -112,7 +112,7 @@ class RemoteServerTest extends UnitTestCase $liveFileReferences = $this->getFileReferenceProphecies($fileFileReferenceList); $versionFileReferences = $this->getFileReferenceProphecies($versionFileReferenceList); - $subject = $this->getAccessibleMock(RemoteServer::class, ['__none']); + $subject = $this->getAccessibleMock(RemoteServer::class, ['__none'], [], '', false); $result = $subject->_call( 'prepareFileReferenceDifferences', $liveFileReferences, -- GitLab