diff --git a/typo3/sysext/backend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php b/typo3/sysext/backend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php index fe7b6adf7ae6e43e861be34c9b6b9961a486f25a..4481346d87d77e278160a9aea1668083a7fffedd 100644 --- a/typo3/sysext/backend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php +++ b/typo3/sysext/backend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php @@ -61,9 +61,16 @@ class ConditionMatcher extends AbstractConditionMatcher $backend->user->userId = $backendUserAspect->get('id'); $backend->user->userGroupList = implode(',', $backendUserAspect->get('groupIds')); + $workspaceAspect = $this->context->getAspect('workspace'); + $workspace = new \stdClass(); + $workspace->workspaceId = $workspaceAspect->get('id'); + $workspace->isLive = $workspaceAspect->get('isLive'); + $workspace->isOffline = $workspaceAspect->get('isOffline'); + $this->expressionLanguageResolverVariables = [ 'tree' => $tree, 'backend' => $backend, + 'workspace' => $workspace, 'page' => BackendUtility::getRecord('pages', $this->pageId ?? $this->determinePageId()) ?: [], ]; } diff --git a/typo3/sysext/backend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php b/typo3/sysext/backend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php index f1acc61a6a1120f55cf290dea1d1e29106336b89..c8526b34d0646171eba83c95273010a5662f2398 100644 --- a/typo3/sysext/backend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php +++ b/typo3/sysext/backend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php @@ -19,6 +19,8 @@ use TYPO3\CMS\Backend\Configuration\TypoScript\ConditionMatching\ConditionMatche use TYPO3\CMS\Core\Authentication\BackendUserAuthentication; use TYPO3\CMS\Core\Context\Context; use TYPO3\CMS\Core\Context\UserAspect; +use TYPO3\CMS\Core\Context\WorkspaceAspect; +use TYPO3\CMS\Core\Log\Logger; use TYPO3\CMS\Core\Utility\GeneralUtility; use TYPO3\CMS\Core\Utility\StringUtility; use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase; @@ -28,11 +30,6 @@ use TYPO3\TestingFramework\Core\Functional\FunctionalTestCase; */ class ConditionMatcherTest extends FunctionalTestCase { - /** - * @var ConditionMatcher|\PHPUnit\Framework\MockObject\MockObject|\TYPO3\TestingFramework\Core\AccessibleObjectInterface - */ - protected $subject; - /** * Sets up this test case. */ @@ -47,8 +44,6 @@ class ConditionMatcherTest extends FunctionalTestCase $backendUser->user['admin'] = true; $backendUser->groupList = '13,14,15'; GeneralUtility::makeInstance(Context::class)->setAspect('backend.user', new UserAspect($backendUser)); - - $this->subject = new ConditionMatcher(); } /** @@ -58,9 +53,10 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function usergroupConditionMatchesSingleGroupId(): void { - self::assertTrue($this->subject->match('[usergroup(13)]')); - self::assertTrue($this->subject->match('[usergroup("13")]')); - self::assertTrue($this->subject->match('[usergroup(\'13\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[usergroup(13)]')); + self::assertTrue($subject->match('[usergroup("13")]')); + self::assertTrue($subject->match('[usergroup(\'13\')]')); } /** @@ -70,8 +66,9 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function usergroupConditionMatchesMultipleUserGroupId(): void { - self::assertTrue($this->subject->match('[usergroup("999,15,14,13")]')); - self::assertTrue($this->subject->match('[usergroup(\'999,15,14,13\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[usergroup("999,15,14,13")]')); + self::assertTrue($subject->match('[usergroup(\'999,15,14,13\')]')); } /** @@ -81,8 +78,9 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function loginUserConditionMatchesAnyLoggedInUser(): void { - self::assertTrue($this->subject->match('[loginUser("*")]')); - self::assertTrue($this->subject->match('[loginUser(\'*\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[loginUser("*")]')); + self::assertTrue($subject->match('[loginUser(\'*\')]')); } /** @@ -92,9 +90,10 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function loginUserConditionMatchesSingleLoggedInUser(): void { - self::assertTrue($this->subject->match('[loginUser(13)]')); - self::assertTrue($this->subject->match('[loginUser("13")]')); - self::assertTrue($this->subject->match('[loginUser(\'13\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[loginUser(13)]')); + self::assertTrue($subject->match('[loginUser("13")]')); + self::assertTrue($subject->match('[loginUser(\'13\')]')); } /** @@ -105,9 +104,10 @@ class ConditionMatcherTest extends FunctionalTestCase public function loginUserConditionDoesNotMatchSingleLoggedInUser(): void { $GLOBALS['BE_USER']->user['uid'] = 13; - self::assertFalse($this->subject->match('[loginUser(999)]')); - self::assertFalse($this->subject->match('[loginUser("999")]')); - self::assertFalse($this->subject->match('[loginUser(\'999\')]')); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[loginUser(999)]')); + self::assertFalse($subject->match('[loginUser("999")]')); + self::assertFalse($subject->match('[loginUser(\'999\')]')); } /** @@ -117,8 +117,9 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function loginUserConditionMatchesMultipleLoggedInUsers(): void { - self::assertTrue($this->subject->match('[loginUser("999,13")]')); - self::assertTrue($this->subject->match('[loginUser(\'999,13\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[loginUser("999,13")]')); + self::assertTrue($subject->match('[loginUser(\'999,13\')]')); } /** @@ -128,9 +129,57 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function adminUserConditionMatchesAdminUser(): void { - self::assertTrue($this->subject->match('[backend.user.isAdmin == true]')); - self::assertTrue($this->subject->match('[backend.user.isAdmin != false]')); - self::assertTrue($this->subject->match('[backend.user.isAdmin]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[backend.user.isAdmin == true]')); + self::assertTrue($subject->match('[backend.user.isAdmin != false]')); + self::assertTrue($subject->match('[backend.user.isAdmin]')); + } + + /** + * Tests whether checking for workspace id matches current workspace id + * + * @test + */ + public function workspaceIdConditionMatchesCurrentWorkspaceId(): void + { + $this->setUpWorkspaceAspect(0); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[workspace.workspaceId === 0]')); + self::assertTrue($subject->match('[workspace.workspaceId == 0]')); + self::assertTrue($subject->match('[workspace.workspaceId == "0"]')); + self::assertTrue($subject->match('[workspace.workspaceId == \'0\']')); + } + + /** + * Tests whether checking if workspace is live matches + * + * @test + */ + public function workspaceIsLiveMatchesCorrectWorkspaceState(): void + { + $this->setUpWorkspaceAspect(1); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[workspace.isLive]')); + self::assertFalse($subject->match('[workspace.isLive === true]')); + self::assertFalse($subject->match('[workspace.isLive == true]')); + self::assertFalse($subject->match('[workspace.isLive !== false]')); + self::assertFalse($subject->match('[workspace.isLive != false]')); + } + + /** + * Tests whether checking if workspace is offline matches + * + * @test + */ + public function workspaceIsOfflineMatchesCorrectWorkspaceState(): void + { + $this->setUpWorkspaceAspect(1); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[workspace.isOffline]')); + self::assertTrue($subject->match('[workspace.isOffline === true]')); + self::assertTrue($subject->match('[workspace.isOffline == true]')); + self::assertTrue($subject->match('[workspace.isOffline !== false]')); + self::assertTrue($subject->match('[workspace.isOffline != false]')); } /** @@ -140,8 +189,8 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function treeLevelConditionMatchesSingleValue(): void { - $this->subject->__construct(null, 2); - self::assertTrue($this->subject->match('[tree.level == 2]')); + $subject = $this->getConditionMatcher(2); + self::assertTrue($subject->match('[tree.level == 2]')); } /** @@ -151,8 +200,8 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function treeLevelConditionMatchesMultipleValues(): void { - $this->subject->__construct(null, 2); - self::assertTrue($this->subject->match('[tree.level in [999,998,2]]')); + $subject = $this->getConditionMatcher(2); + self::assertTrue($subject->match('[tree.level in [999,998,2]]')); } /** @@ -162,7 +211,8 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function treeLevelConditionDoesNotMatchFaultyValue(): void { - self::assertFalse($this->subject->match('[tree.level == 999]')); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[tree.level == 999]')); } /** @@ -172,7 +222,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDupinRootlineConditionMatchesSinglePageIdInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertTrue($subject->match('[2 in tree.rootLineParentIds]')); self::assertTrue($subject->match('["2" in tree.rootLineParentIds]')); self::assertTrue($subject->match('[\'2\' in tree.rootLineParentIds]')); @@ -185,7 +235,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDupinRootlineConditionDoesNotMatchLastPageIdInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertFalse($subject->match('[3 in tree.rootLineParentIds]')); } @@ -196,7 +246,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDupinRootlineConditionDoesNotMatchPageIdNotInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertFalse($subject->match('[999 in tree.rootLineParentIds]')); } @@ -207,7 +257,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDinRootlineConditionMatchesSinglePageIdInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertTrue($subject->match('[2 in tree.rootLineIds]')); } @@ -218,7 +268,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDinRootlineConditionMatchesLastPageIdInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertTrue($subject->match('[3 in tree.rootLineIds]')); } @@ -229,7 +279,7 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function PIDinRootlineConditionDoesNotMatchPageIdNotInRootline(): void { - $subject = new ConditionMatcher(null, 3); + $subject = $this->getConditionMatcher(3); self::assertFalse($subject->match('[999 in tree.rootLineIds]')); } @@ -241,9 +291,10 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function compatVersionConditionMatchesOlderRelease(): void { - self::assertTrue($this->subject->match('[compatVersion(7.0)]')); - self::assertTrue($this->subject->match('[compatVersion("7.0")]')); - self::assertTrue($this->subject->match('[compatVersion(\'7.0\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[compatVersion(7.0)]')); + self::assertTrue($subject->match('[compatVersion("7.0")]')); + self::assertTrue($subject->match('[compatVersion(\'7.0\')]')); } /** @@ -254,9 +305,10 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function compatVersionConditionMatchesSameRelease(): void { - self::assertTrue($this->subject->match('[compatVersion(' . TYPO3_branch . ')]')); - self::assertTrue($this->subject->match('[compatVersion("' . TYPO3_branch . '")]')); - self::assertTrue($this->subject->match('[compatVersion(\'' . TYPO3_branch . '\')]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[compatVersion(' . TYPO3_branch . ')]')); + self::assertTrue($subject->match('[compatVersion("' . TYPO3_branch . '")]')); + self::assertTrue($subject->match('[compatVersion(\'' . TYPO3_branch . '\')]')); } /** @@ -267,9 +319,10 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function compatVersionConditionDoesNotMatchNewerRelease(): void { - self::assertFalse($this->subject->match('[compatVersion(15.0)]')); - self::assertFalse($this->subject->match('[compatVersion("15.0")]')); - self::assertFalse($this->subject->match('[compatVersion(\'15.0\')]')); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[compatVersion(15.0)]')); + self::assertFalse($subject->match('[compatVersion("15.0")]')); + self::assertFalse($subject->match('[compatVersion(\'15.0\')]')); } /** @@ -281,7 +334,8 @@ class ConditionMatcherTest extends FunctionalTestCase { $testKey = StringUtility::getUniqueId('test'); putenv($testKey . '=testValue'); - self::assertTrue($this->subject->match('[getenv("' . $testKey . '") == "testValue"]')); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[getenv("' . $testKey . '") == "testValue"]')); } /** @@ -289,6 +343,29 @@ class ConditionMatcherTest extends FunctionalTestCase */ public function usingTSFEInATestInBeContextIsAlwaysFalse(): void { - self::assertFalse($this->subject->match('[getTSFE().id == 1]')); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[getTSFE().id == 1]')); + } + + /** + * @param int|null $pageId + * @return ConditionMatcher + */ + protected function getConditionMatcher(int $pageId = null): ConditionMatcher + { + $conditionMatcher = new ConditionMatcher(null, $pageId); + $conditionMatcher->setLogger($this->prophesize(Logger::class)->reveal()); + + return $conditionMatcher; + } + + /** + * Set up workspace aspect. + * + * @param int $workspaceId + */ + protected function setUpWorkspaceAspect(int $workspaceId): void + { + GeneralUtility::makeInstance(Context::class)->setAspect('workspace', new WorkspaceAspect($workspaceId)); } } diff --git a/typo3/sysext/core/Documentation/Changelog/9.4/Feature-85829-ImplementSymfonyExpressionLanguageForTypoScriptConditions.rst b/typo3/sysext/core/Documentation/Changelog/9.4/Feature-85829-ImplementSymfonyExpressionLanguageForTypoScriptConditions.rst index 9c5f236c2af9501e2197968e3187d7d77fd03399..57a11723e0ea3927b35e77472dc5cb6e7f0684ad 100644 --- a/typo3/sysext/core/Documentation/Changelog/9.4/Feature-85829-ImplementSymfonyExpressionLanguageForTypoScriptConditions.rst +++ b/typo3/sysext/core/Documentation/Changelog/9.4/Feature-85829-ImplementSymfonyExpressionLanguageForTypoScriptConditions.rst @@ -111,6 +111,14 @@ The following variables are available. The values are context related. | | | | | .user.userGroupList | String | comma list of group UIDs | +---------------------+------------+------------------------------------------------------------------------------+ +| workspace | Object | object with workspace information | +| | | | +| .workspaceId | Integer | id of current workspace | +| | | | +| .isLive | Boolean | true if current workspace is live | +| | | | +| .isOffline | Boolean | true if current workspace is offline | ++---------------------+------------+------------------------------------------------------------------------------+ | typo3 | Object | object with TYPO3 related information | | | | | | .version | String | TYPO3_version (e.g. 9.4.0-dev) | diff --git a/typo3/sysext/core/Documentation/Changelog/master/Feature-90203-MakeWorkspaceAvailableInTypoScriptConditions.rst b/typo3/sysext/core/Documentation/Changelog/master/Feature-90203-MakeWorkspaceAvailableInTypoScriptConditions.rst new file mode 100644 index 0000000000000000000000000000000000000000..49a207eaa2f83d7b6cbdc1d567b8ea4c7616f00d --- /dev/null +++ b/typo3/sysext/core/Documentation/Changelog/master/Feature-90203-MakeWorkspaceAvailableInTypoScriptConditions.rst @@ -0,0 +1,46 @@ +.. include:: ../../Includes.txt + +=================================================================== +Feature: #90203 - Make workspace available in TypoScript conditions +=================================================================== + +See :issue:`90203` + +Description +=========== + +A new TypoScript expression language variable `workspace` has been added. +It can be used to match a given expression against common workspace parameters. + +Currently, the parameters `workspaceId`, `isLive` and `isOffline` are supported. + +Examples +-------- + +Match the current workspace id: + +.. code-block:: ts + + [workspace.workspaceId === 3] + # Current workspace id equals: 3 + [end] + +Match against current workspace state: + +.. code-block:: ts + + [workspace.isLive] + # Current workspace is live + [end] + + [workspace.isOffline] + # Current workspace is offline + [end] + + +Impact +====== + +The new feature allows matching against several workspace parameters within TypoScript. + +.. index:: TypoScript \ No newline at end of file diff --git a/typo3/sysext/frontend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php b/typo3/sysext/frontend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php index 2a0b2f0e750f67c1af9b5dae56909698ee23e9c0..188a75ef57099d99b928fba152cf030f829d1c54 100644 --- a/typo3/sysext/frontend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php +++ b/typo3/sysext/frontend/Classes/Configuration/TypoScript/ConditionMatching/ConditionMatcher.php @@ -60,9 +60,16 @@ class ConditionMatcher extends AbstractConditionMatcher $frontend->user->userId = $frontendUserAspect->get('id'); $frontend->user->userGroupList = implode(',', $frontendUserAspect->get('groupIds')); + $workspaceAspect = $this->context->getAspect('workspace'); + $workspace = new \stdClass(); + $workspace->workspaceId = $workspaceAspect->get('id'); + $workspace->isLive = $workspaceAspect->get('isLive'); + $workspace->isOffline = $workspaceAspect->get('isOffline'); + $this->expressionLanguageResolverVariables = [ 'tree' => $tree, 'frontend' => $frontend, + 'workspace' => $workspace, 'page' => $GLOBALS['TSFE']->page ?? [], ]; } diff --git a/typo3/sysext/frontend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php b/typo3/sysext/frontend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php index a96e7ea8ec16dd45a7db08fa9ac87a975fd6965a..f5e1775450c3e1f4b29f859ce4551a555651727c 100644 --- a/typo3/sysext/frontend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php +++ b/typo3/sysext/frontend/Tests/Functional/Configuration/TypoScript/ConditionMatching/ConditionMatcherTest.php @@ -18,6 +18,7 @@ namespace TYPO3\CMS\Frontend\Tests\Functional\Configuration\TypoScript\Condition use Prophecy\Argument; use TYPO3\CMS\Core\Context\Context; use TYPO3\CMS\Core\Context\UserAspect; +use TYPO3\CMS\Core\Context\WorkspaceAspect; use TYPO3\CMS\Core\Domain\Repository\PageRepository; use TYPO3\CMS\Core\Http\ServerRequest; use TYPO3\CMS\Core\Log\Logger; @@ -159,6 +160,53 @@ class ConditionMatcherTest extends FunctionalTestCase self::assertTrue($subject->match('[loginUser("*") == false]')); } + /** + * Tests whether checking for workspace id matches current workspace id + * + * @test + */ + public function workspaceIdConditionMatchesCurrentWorkspaceId(): void + { + $this->setUpWorkspaceAspect(0); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[workspace.workspaceId === 0]')); + self::assertTrue($subject->match('[workspace.workspaceId == 0]')); + self::assertTrue($subject->match('[workspace.workspaceId == "0"]')); + self::assertTrue($subject->match('[workspace.workspaceId == \'0\']')); + } + + /** + * Tests whether checking if workspace is live matches + * + * @test + */ + public function workspaceIsLiveMatchesCorrectWorkspaceState(): void + { + $this->setUpWorkspaceAspect(1); + $subject = $this->getConditionMatcher(); + self::assertFalse($subject->match('[workspace.isLive]')); + self::assertFalse($subject->match('[workspace.isLive === true]')); + self::assertFalse($subject->match('[workspace.isLive == true]')); + self::assertFalse($subject->match('[workspace.isLive !== false]')); + self::assertFalse($subject->match('[workspace.isLive != false]')); + } + + /** + * Tests whether checking if workspace is offline matches + * + * @test + */ + public function workspaceIsOfflineMatchesCorrectWorkspaceState(): void + { + $this->setUpWorkspaceAspect(1); + $subject = $this->getConditionMatcher(); + self::assertTrue($subject->match('[workspace.isOffline]')); + self::assertTrue($subject->match('[workspace.isOffline === true]')); + self::assertTrue($subject->match('[workspace.isOffline == true]')); + self::assertTrue($subject->match('[workspace.isOffline !== false]')); + self::assertTrue($subject->match('[workspace.isOffline != false]')); + } + /** * Tests whether treeLevel comparison matches. * @@ -452,6 +500,16 @@ class ConditionMatcherTest extends FunctionalTestCase GeneralUtility::makeInstance(Context::class)->setAspect('frontend.user', new UserAspect($frontendUser, $groups)); } + /** + * Set up workspace aspect. + * + * @param int $workspaceId + */ + protected function setUpWorkspaceAspect(int $workspaceId): void + { + GeneralUtility::makeInstance(Context::class)->setAspect('workspace', new WorkspaceAspect($workspaceId)); + } + /** * @param int $pageId */