From 1612659c311f9532a6935f3ca216242aa9f51e72 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Christian=20Mu=CC=88ller?= Date: Mon, 17 Jun 2024 18:41:01 +0200 Subject: [PATCH] FEATURE: References on creation and Copy This reworks references so that multiple reference properties can be set via a single command and also references can be attached to `CreateNodeAggregateWithNode` which is also used for copying nodes. --- .../ReferenceIntegrityIsProvided.feature | 3 +- .../DoctrineDbalContentGraphProjection.php | 64 +++++---- .../Command/PerformanceMeasurementService.php | 2 + ...SetNodeReferences_ConstraintChecks.feature | 51 +++---- ...etNodeReferences_WithoutDimensions.feature | 43 ++---- ...3-SetNodeReferences_WithDimensions.feature | 3 +- ...4-SetNodeReferences_PropertyScopes.feature | 87 +++++++++--- ...odeVariation_After_NodeReferencing.feature | 17 +-- ...bleNodeAggregate_WithoutDimensions.feature | 3 +- ...isableNodeAggregate_WithDimensions.feature | 3 +- ...bleNodeAggregate_WithoutDimensions.feature | 3 +- ...EnableNodeAggregate_WithDimensions.feature | 3 +- ...oveNodeAggregate_WithoutDimensions.feature | 3 +- ...RemoveNodeAggregate_WithDimensions.feature | 3 +- .../NodeReferencesOnForkContentStream.feature | 3 +- .../NodeCopying/CopyNode_NoDimensions.feature | 25 +++- .../RemoveNodeAggregateAfterDisabling.feature | 3 +- .../Features/NodeTraversal/References.feature | 27 ++-- .../Features/NodeTraversal/Timestamps.feature | 3 +- .../IntactContentGraph.feature | 3 +- .../ReferenceIntegrityIsProvided.feature | 6 +- .../04-AllFeaturePublication.feature | 6 +- .../Feature/Common/ConstraintChecks.php | 25 ++-- .../Common/NodeReferencingInternals.php | 43 ++++++ .../Feature/Common/TetheredNodeInternals.php | 3 + .../Command/CreateNodeAggregateWithNode.php | 32 ++++- ...gregateWithNodeAndSerializedProperties.php | 19 ++- .../Event/NodeAggregateWithNodeWasCreated.php | 4 + .../Feature/NodeCreation/NodeCreation.php | 9 +- .../Dto/NodeReferenceSnapshot.php | 57 -------- .../Dto/NodeReferencesSnapshot.php | 106 --------------- .../Dto/NodeSubtreeSnapshot.php | 7 +- .../NodeDuplicationCommandHandler.php | 1 + .../Command/SetNodeReferences.php | 7 +- .../Command/SetSerializedNodeReferences.php | 11 +- .../Dto/NodeReferenceNameToEmpty.php | 30 ++++ .../Dto/NodeReferenceToWrite.php | 18 ++- .../Dto/NodeReferencesToWrite.php | 98 +++++++++----- .../Dto/SerializedNodeReference.php | 19 +-- .../Dto/SerializedNodeReferences.php | 128 ++++++++++++++---- .../Event/NodeReferencesWereSet.php | 4 - .../NodeReferencing/NodeReferencing.php | 103 +++++++------- .../RootNodeCreation/RootNodeHandling.php | 2 + .../Classes/Projection/CatchUp.php | 1 + .../Classes/NodeDataToEventsProcessor.php | 11 +- .../Bootstrap/Features/NodeReferencing.php | 44 ++++-- 46 files changed, 634 insertions(+), 512 deletions(-) create mode 100644 Neos.ContentRepository.Core/Classes/Feature/Common/NodeReferencingInternals.php delete mode 100644 Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferenceSnapshot.php delete mode 100644 Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferencesSnapshot.php create mode 100644 Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferenceNameToEmpty.php diff --git a/Neos.ContentGraph.DoctrineDbalAdapter/Tests/Behavior/Features/Projection/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature b/Neos.ContentGraph.DoctrineDbalAdapter/Tests/Behavior/Features/Projection/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature index 28e18f5fba4..fdae3c9ede3 100644 --- a/Neos.ContentGraph.DoctrineDbalAdapter/Tests/Behavior/Features/Projection/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature +++ b/Neos.ContentGraph.DoctrineDbalAdapter/Tests/Behavior/Features/Projection/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature @@ -53,8 +53,7 @@ Feature: Run integrity violation detection regarding reference relations | Key | Value | | sourceOriginDimensionSpacePoint | {"language":"de"} | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | And I detach the following reference relation from its source: | Key | Value | | contentStreamId | "cs-identifier" | diff --git a/Neos.ContentGraph.DoctrineDbalAdapter/src/DoctrineDbalContentGraphProjection.php b/Neos.ContentGraph.DoctrineDbalAdapter/src/DoctrineDbalContentGraphProjection.php index 07ec7c03755..e1722d8c847 100644 --- a/Neos.ContentGraph.DoctrineDbalAdapter/src/DoctrineDbalContentGraphProjection.php +++ b/Neos.ContentGraph.DoctrineDbalAdapter/src/DoctrineDbalContentGraphProjection.php @@ -30,7 +30,9 @@ use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; use Neos\ContentRepository\Core\Feature\NodeModification\Event\NodePropertiesWereSet; use Neos\ContentRepository\Core\Feature\NodeMove\Event\NodeAggregateWasMoved; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferenceNameToEmpty; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReference; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\NodeReferencing\Event\NodeReferencesWereSet; use Neos\ContentRepository\Core\Feature\NodeRemoval\Event\NodeAggregateWasRemoved; use Neos\ContentRepository\Core\Feature\NodeRenaming\Event\NodeAggregateNameWasChanged; @@ -434,6 +436,7 @@ private function whenNodeAggregateWithNodeWasCreated(NodeAggregateWithNodeWasCre $event->originDimensionSpacePoint, $event->succeedingSiblingsForCoverage, $event->initialPropertyValues, + $event->nodeReferences, $event->nodeAggregateClassification, $event->nodeName, $eventEnvelope, @@ -520,40 +523,52 @@ function (NodeRecord $node) use ($eventEnvelope) { ); // remove old - try { + foreach ($event->references->getReferenceNames() as $referenceName) { + try { $this->dbal->delete($this->tableNames->referenceRelation(), [ 'nodeanchorpoint' => $nodeAnchorPoint?->value, 'name' => $event->referenceName->value ]); - } catch (DBALException $e) { + } catch (DbalException $e) { throw new \RuntimeException(sprintf('Failed to remove reference relation: %s', $e->getMessage()), 1716486309, $e); + } } // set new - $position = 0; - /** @var SerializedNodeReference $reference */ - foreach ($event->references as $reference) { - $referencePropertiesJson = null; - if ($reference->properties !== null) { - try { - $referencePropertiesJson = \json_encode($reference->properties, JSON_THROW_ON_ERROR | JSON_FORCE_OBJECT); - } catch (\JsonException $e) { - throw new \RuntimeException(sprintf('Failed to JSON-encode reference properties: %s', $e->getMessage()), 1716486271, $e); - } - } + $nodeAnchorPoint && $this->writeReferencesForTargetAnchorPoint($event->references, $nodeAnchorPoint); + } + } + + private function writeReferencesForTargetAnchorPoint(SerializedNodeReferences $nodeReferences, NodeRelationAnchorPoint $nodeAnchorPoint): void + { + $position = 0; + /** @var NodeReferenceNameToEmpty|SerializedNodeReference $reference */ + foreach ($nodeReferences as $reference) { + if ($reference instanceof NodeReferenceNameToEmpty) { + // Reference empty happens separately + continue; + } + + $referencePropertiesJson = null; + if ($reference->properties !== null && $reference->properties->count() > 0) { try { - $this->dbal->insert($this->tableNames->referenceRelation(), [ - 'name' => $event->referenceName->value, - 'position' => $position, - 'nodeanchorpoint' => $nodeAnchorPoint?->value, - 'destinationnodeaggregateid' => $reference->targetNodeAggregateId->value, - 'properties' => $referencePropertiesJson, - ]); - } catch (DBALException $e) { - throw new \RuntimeException(sprintf('Failed to insert reference relation: %s', $e->getMessage()), 1716486309, $e); + $referencePropertiesJson = \json_encode($reference->properties, JSON_THROW_ON_ERROR | JSON_FORCE_OBJECT); + } catch (\JsonException $e) { + throw new \RuntimeException(sprintf('Failed to JSON-encode reference properties: %s', $e->getMessage()), 1716486271, $e); } - $position++; } + try { + $this->dbal->insert($this->tableNames->referenceRelation(), [ + 'name' => $reference->referenceName->value, + 'position' => $position, + 'nodeanchorpoint' => $nodeAnchorPoint?->value, + 'destinationnodeaggregateid' => $reference->targetNodeAggregateId->value, + 'properties' => $referencePropertiesJson, + ]); + } catch (DbalException $e) { + throw new \RuntimeException(sprintf('Failed to insert reference relation: %s', $e->getMessage()), 1716486309, $e); + } + $position++; } } @@ -776,6 +791,7 @@ private function createNodeWithHierarchy( OriginDimensionSpacePoint $originDimensionSpacePoint, InterdimensionalSiblings $coverageSucceedingSiblings, SerializedPropertyValues $propertyDefaultValuesAndTypes, + SerializedNodeReferences $references, NodeAggregateClassification $nodeAggregateClassification, ?NodeName $nodeName, EventEnvelope $eventEnvelope, @@ -831,6 +847,8 @@ private function createNodeWithHierarchy( } } } + + $this->writeReferencesForTargetAnchorPoint($references, $node->relationAnchorPoint); } private function connectHierarchy( diff --git a/Neos.ContentRepository.BehavioralTests/Classes/Command/PerformanceMeasurementService.php b/Neos.ContentRepository.BehavioralTests/Classes/Command/PerformanceMeasurementService.php index 21c344d2f9d..e0fb76afd99 100644 --- a/Neos.ContentRepository.BehavioralTests/Classes/Command/PerformanceMeasurementService.php +++ b/Neos.ContentRepository.BehavioralTests/Classes/Command/PerformanceMeasurementService.php @@ -29,6 +29,7 @@ use Neos\ContentRepository\Core\Feature\ContentStreamForking\Command\ForkContentStream; use Neos\ContentRepository\Core\Feature\NodeCreation\Event\NodeAggregateWithNodeWasCreated; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\RootNodeCreation\Event\RootNodeAggregateWithNodeWasCreated; use Neos\ContentRepository\Core\Feature\WorkspaceCreation\Command\CreateRootWorkspace; use Neos\ContentRepository\Core\NodeType\NodeTypeName; @@ -142,6 +143,7 @@ private function createHierarchy( null, SerializedPropertyValues::createEmpty(), NodeAggregateClassification::CLASSIFICATION_REGULAR, + SerializedNodeReferences::createEmpty(), ); $sumSoFar++; $this->createHierarchy( diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/01-SetNodeReferences_ConstraintChecks.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/01-SetNodeReferences_ConstraintChecks.feature index 9c3517cef20..7469d5f00e8 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/01-SetNodeReferences_ConstraintChecks.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/01-SetNodeReferences_ConstraintChecks.feature @@ -63,8 +63,7 @@ Feature: Constraint checks on SetNodeReferences When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "ContentStreamIsClosed" # checks for contentStreamId @@ -73,8 +72,7 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | workspaceName | "i-do-not-exist" | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "WorkspaceDoesNotExist" with code 1513924741 # checks for sourceNodeAggregateId @@ -82,16 +80,14 @@ Feature: Constraint checks on SetNodeReferences When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "i-do-not-exist" | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "NodeAggregateCurrentlyDoesNotExist" with code 1541678486 Scenario: Try to reference nodes in a root node aggregate When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "lady-eleonode-rootford" | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "NodeAggregateIsRoot" # checks for sourceOriginDimensionSpacePoint @@ -100,8 +96,7 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"undeclared":"undefined"} | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "DimensionSpacePointNotFound" with code 1505929456 Scenario: Try to reference nodes in an origin dimension space point the source node aggregate does not occupy @@ -109,8 +104,7 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"language":"en"} | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "DimensionSpacePointIsNotYetOccupied" with code 1552595396 # checks for destinationnodeAggregateIds @@ -118,40 +112,35 @@ Feature: Constraint checks on SetNodeReferences When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target":"i-do-not-exist"}] | + | references | {"referenceProperty": [{"target":"i-do-not-exist"}]} | Then the last command should have thrown an exception of type "NodeAggregateCurrentlyDoesNotExist" with code 1541678486 Scenario: Try to reference a root node aggregate When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target":"lady-eleonode-rootford"}] | + | references | {"referenceProperty": [{"target":"lady-eleonode-rootford"}]} | Then the last command should have thrown an exception of type "NodeAggregateIsRoot" Scenario: Try to set references exceeding the maxItems count When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "constrainedReferenceCount" | - | references | [{"target":"anthony-destinode"}, {"target":"berta-destinode"}] | + | references | {"constrainedReferenceCount": [{"target":"anthony-destinode"}, {"target":"berta-destinode"}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1700150156 Scenario: Try to set references exceeding the maxItems count for legacy property reference declaration When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target":"anthony-destinode"}, {"target":"berta-destinode"}] | + | references | {"referenceProperty": [{"target":"anthony-destinode"}, {"target":"berta-destinode"}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1700150156 Scenario: Try to reference a node aggregate of a type not matching the constraints When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "constrainedReferenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"constrainedReferenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1648502149 Scenario: Try to reference a node aggregate which does not cover the source origin @@ -166,8 +155,7 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | referenceName | "referenceProperty" | - | references | [{"target":"sir-david-nodenborough"}] | + | references | {"referenceProperty": [{"target":"sir-david-nodenborough"}]} | Then the last command should have thrown an exception of type "NodeAggregateDoesCurrentlyNotCoverDimensionSpacePoint" # checks for referenceName @@ -175,16 +163,14 @@ Feature: Constraint checks on SetNodeReferences When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "i-do-not-exist" | - | references | [{"target":"anthony-destinode"}] | + | references | {"i-do-not-exist": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1618670106 Scenario: Try to reference nodes in a property that is not of type reference(s): When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "nonReferenceProperty" | - | references | [{"target":"anthony-destinode"}] | + | references | {"nonReferenceProperty": [{"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1618670106 Scenario: Try to reference a node aggregate using a property the reference does not declare @@ -192,8 +178,7 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | nodeAggregateId | "nody-mc-nodeface" | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencePropertyWithProperties" | - | references | [{"target":"anthony-destinode", "properties":{"i-do-not-exist": "whatever"}}] | + | references | {"referencePropertyWithProperties": [{"target":"anthony-destinode", "properties":{"i-do-not-exist": "whatever"}}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1658406662 Scenario: Try to set a property with a value of a wrong type @@ -201,14 +186,12 @@ Feature: Constraint checks on SetNodeReferences | Key | Value | | nodeAggregateId | "nody-mc-nodeface" | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencePropertyWithProperties" | - | references | [{"target":"anthony-destinode", "properties":{"postalAddress": "28 31st of February Street"}}] | + | references | {"referencePropertyWithProperties": [{"target":"anthony-destinode", "properties":{"postalAddress": "28 31st of February Street"}}]} | Then the last command should have thrown an exception of type "ReferenceCannotBeSet" with code 1658406762 Scenario: Node reference cannot hold multiple targets to the same node When the command SetNodeReferences is executed with payload and exceptions are caught: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencesProperty" | - | references | [{"target":"anthony-destinode"}, {"target":"anthony-destinode"}] | + | references | {"referencesProperty": [{"target":"anthony-destinode"}, {"target":"anthony-destinode"}]} | Then the last command should have thrown an exception of type "InvalidArgumentException" with code 1700150910 diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/02-SetNodeReferences_WithoutDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/02-SetNodeReferences_WithoutDimensions.feature index 1db0714c012..b0be1c8b752 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/02-SetNodeReferences_WithoutDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/02-SetNodeReferences_WithoutDimensions.feature @@ -67,8 +67,7 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -84,8 +83,7 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencePropertyWithProperty" | - | references | [{"target": "anthony-destinode", "properties":{"text":"my text", "dayOfWeek":"DayOfWeek:https://schema.org/Friday", "postalAddress":"PostalAddress:dummy"}}] | + | references | {"referencePropertyWithProperty": [{"target": "anthony-destinode", "properties":{"text":"my text", "dayOfWeek":"DayOfWeek:https://schema.org/Friday", "postalAddress":"PostalAddress:dummy"}}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -101,8 +99,7 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencesProperty" | - | references | [{"target": "berta-destinode"}, {"target": "carl-destinode"}] | + | references | {"referencesProperty": [{"target": "berta-destinode"}, {"target": "carl-destinode"}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -124,8 +121,7 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referencesPropertyWithProperty" | - | references | [{"target":"berta-destinode", "properties":{"text":"my text", "dayOfWeek":"DayOfWeek:https://schema.org/Wednesday", "postalAddress":"PostalAddress:dummy"}}, {"target":"carl-destinode", "properties":{"text":"my other text", "dayOfWeek":"DayOfWeek:https://schema.org/Friday", "postalAddress":"PostalAddress:anotherDummy"}}] | + | references | {"referencesPropertyWithProperty": [{"target":"berta-destinode", "properties":{"text":"my text", "dayOfWeek":"DayOfWeek:https://schema.org/Wednesday", "postalAddress":"PostalAddress:dummy"}}, {"target":"carl-destinode", "properties":{"text":"my other text", "dayOfWeek":"DayOfWeek:https://schema.org/Friday", "postalAddress":"PostalAddress:anotherDummy"}}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -147,14 +143,12 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "berta-destinode"}, {"target": "carl-destinode"}] | - | referenceName | "referencesProperty" | + | references | {"referencesProperty": [{"target": "berta-destinode"}, {"target": "carl-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "anthony-destinode"}] | - | referenceName | "referencesProperty" | + | references | {"referencesProperty": [{"target": "anthony-destinode"}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -177,14 +171,12 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "berta-destinode"}, {"target": "carl-destinode"}] | - | referenceName | "referencesProperty" | + | references | {"referencesProperty": [{"target": "berta-destinode"}, {"target": "carl-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "carl-destinode"}, {"target": "berta-destinode"}] | - | referenceName | "referencesProperty" | + | references | {"referencesProperty": [{"target": "carl-destinode"}, {"target": "berta-destinode"}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: @@ -197,14 +189,12 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "berta-destinode"}, {"target": "carl-destinode"}] | - | referenceName | "referencesProperty" | + | references | {"referencesProperty": [{"target": "berta-destinode"}, {"target": "carl-destinode"}]} | And the command SetNodeReferences is executed with payload: - | Key | Value | - | sourceNodeAggregateId | "source-nodandaise" | - | references | [] | - | referenceName | "referencesProperty" | + | Key | Value | + | sourceNodeAggregateId | "source-nodandaise" | + | references | {"referencesProperty": []} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} @@ -221,14 +211,12 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "anthony-destinode"}] | - | referenceName | "referenceProperty" | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "node-wan-kenodi" | - | references | [{"target": "anthony-destinode"}] | - | referenceName | "referenceProperty" | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | Then I expect node aggregate identifier "anthony-destinode" to lead to node cs-identifier;anthony-destinode;{} And I expect this node to be referenced by: @@ -240,8 +228,7 @@ Feature: Node References without Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | references | [{"target": "anthony-destinode"}] | - | referenceName | "restrictedReferenceProperty" | + | references | {"restrictedReferenceProperty": [{"target": "anthony-destinode"}]} | Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{} And I expect this node to have the following references: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/03-SetNodeReferences_WithDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/03-SetNodeReferences_WithDimensions.feature index 4183ec3ff32..a0ed9627e0f 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/03-SetNodeReferences_WithDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/03-SetNodeReferences_WithDimensions.feature @@ -43,8 +43,7 @@ Feature: Node References with Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | When I am in workspace "live" and dimension space point {"language": "de"} Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{"language": "de"} diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/04-SetNodeReferences_PropertyScopes.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/04-SetNodeReferences_PropertyScopes.feature index 047b825522e..59e20bab676 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/04-SetNodeReferences_PropertyScopes.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/04-SetNodeReferences_PropertyScopes.feature @@ -65,55 +65,108 @@ Feature: Set node properties with different scopes | sourceOrigin | {"language":"mul"} | | targetOrigin | {"language":"gsw"} | - Scenario: Set node properties + Scenario: Set node properties in separate commands And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "unscopedReference" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"unscopedReference": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "unscopedReferences" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"unscopedReferences": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "nodeScopedReference" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"nodeScopedReference": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "nodeScopedReferences" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"nodeScopedReferences": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "nodeAggregateScopedReference" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"nodeAggregateScopedReference": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "nodeAggregateScopedReferences" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"nodeAggregateScopedReferences": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "specializationsScopedReference" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"specializationsScopedReference": [{"target": "anthony-destinode"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "specializationsScopedReferences" | | sourceOriginDimensionSpacePoint | {"language": "de"} | - | references | [{"target": "anthony-destinode"}] | + | references | {"specializationsScopedReferences": [{"target": "anthony-destinode"}]} | + + When I am in workspace "live" and dimension space point {"language": "mul"} + Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{"language": "mul"} + And I expect this node to have the following references: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | nodeAggregateScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + + And I expect node aggregate identifier "anthony-destinode" to lead to node cs-identifier;anthony-destinode;{"language": "mul"} + And I expect this node to be referenced by: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;source-nodandaise;{"language": "mul"} | null | + | nodeAggregateScopedReferences | cs-identifier;source-nodandaise;{"language": "mul"} | null | + + When I am in workspace "live" and dimension space point {"language": "de"} + Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{"language": "de"} + And I expect this node to have the following references: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | nodeAggregateScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | nodeScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | nodeScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | specializationsScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | specializationsScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | unscopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | unscopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + And I expect node aggregate identifier "anthony-destinode" to lead to node cs-identifier;anthony-destinode;{"language": "mul"} + And I expect this node to be referenced by: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;source-nodandaise;{"language": "de"} | null | + | nodeAggregateScopedReferences | cs-identifier;source-nodandaise;{"language": "de"} | null | + | nodeScopedReference | cs-identifier;source-nodandaise;{"language": "de"} | null | + | nodeScopedReferences | cs-identifier;source-nodandaise;{"language": "de"} | null | + | specializationsScopedReference | cs-identifier;source-nodandaise;{"language": "de"} | null | + | specializationsScopedReferences | cs-identifier;source-nodandaise;{"language": "de"} | null | + | unscopedReference | cs-identifier;source-nodandaise;{"language": "de"} | null | + | unscopedReferences | cs-identifier;source-nodandaise;{"language": "de"} | null | + + When I am in workspace "live" and dimension space point {"language": "gsw"} + Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{"language": "gsw"} + And I expect this node to have the following references: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | nodeAggregateScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | specializationsScopedReference | cs-identifier;anthony-destinode;{"language": "mul"} | null | + | specializationsScopedReferences | cs-identifier;anthony-destinode;{"language": "mul"} | null | + And I expect node aggregate identifier "anthony-destinode" to lead to node cs-identifier;anthony-destinode;{"language": "mul"} + And I expect this node to be referenced by: + | Name | Node | Properties | + | nodeAggregateScopedReference | cs-identifier;source-nodandaise;{"language": "gsw"} | null | + | nodeAggregateScopedReferences | cs-identifier;source-nodandaise;{"language": "gsw"} | null | + | specializationsScopedReference | cs-identifier;source-nodandaise;{"language": "gsw"} | null | + | specializationsScopedReferences | cs-identifier;source-nodandaise;{"language": "gsw"} | null | + + + Scenario: Set node properties in single command + And the command SetNodeReferences is executed with payload: + | Key | Value | + | sourceNodeAggregateId | "source-nodandaise" | + | sourceOriginDimensionSpacePoint | {"language": "de"} | + | references | {"unscopedReference": [{"target": "anthony-destinode"}], "unscopedReferences": [{"target": "anthony-destinode"}], "nodeScopedReference": [{"target": "anthony-destinode"}], "nodeScopedReferences": [{"target": "anthony-destinode"}], "nodeAggregateScopedReference": [{"target": "anthony-destinode"}], "nodeAggregateScopedReferences": [{"target": "anthony-destinode"}], "specializationsScopedReference": [{"target": "anthony-destinode"}], "specializationsScopedReferences": [{"target": "anthony-destinode"}]}| When I am in workspace "live" and dimension space point {"language": "mul"} Then I expect node aggregate identifier "source-nodandaise" to lead to node cs-identifier;source-nodandaise;{"language": "mul"} diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/05-NodeVariation_After_NodeReferencing.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/05-NodeVariation_After_NodeReferencing.feature index 8b3cfacf360..2bd63d7f6eb 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/05-NodeVariation_After_NodeReferencing.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/05-NodeReferencing/05-NodeVariation_After_NodeReferencing.feature @@ -43,8 +43,7 @@ Feature: Node References with Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | When the command CreateNodeVariant is executed with payload: | Key | Value | @@ -81,7 +80,7 @@ Feature: Node References with Dimensions | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"language": "ch"} | | referenceName | "referenceProperty" | - | references | [{"target": "source-nodandaise"}] | + | references | {"referenceProperty": [{"target": "source-nodandaise"}]} | # reference to self (modified 2 lines above) When I am in workspace "live" and dimension space point {"language": "ch"} @@ -115,8 +114,7 @@ Feature: Node References with Dimensions | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"language": "ch"} | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | # on the specialization, the reference exists. @@ -150,8 +148,7 @@ Feature: Node References with Dimensions When the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | When the command CreateNodeVariant is executed with payload: | Key | Value | @@ -197,8 +194,7 @@ Feature: Node References with Dimensions | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | | sourceOriginDimensionSpacePoint | {"language": "en"} | - | referenceName | "referenceProperty" | - | references | [{"target": "source-nodandaise"}] | + | references | {"referenceProperty": [{"target": "source-nodandaise"}]} | # reference to self (modified 2 lines above) When I am in workspace "live" and dimension space point {"language": "en"} @@ -244,8 +240,7 @@ Feature: Node References with Dimensions | Key | Value | | sourceNodeAggregateId | "ch-only" | | sourceOriginDimensionSpacePoint | {"language": "ch"} | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | # here we generalize When the command CreateNodeVariant is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/02-DisableNodeAggregate_WithoutDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/02-DisableNodeAggregate_WithoutDimensions.feature index bd36fbddbfa..3e62f811648 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/02-DisableNodeAggregate_WithoutDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/02-DisableNodeAggregate_WithoutDimensions.feature @@ -37,8 +37,7 @@ Feature: Disable a node aggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "preceding-nodenborough" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | Scenario: Disable node with arbitrary strategy since dimensions are not involved When the command DisableNodeAggregate is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/03-DisableNodeAggregate_WithDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/03-DisableNodeAggregate_WithDimensions.feature index be46253d908..c13c4c624f9 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/03-DisableNodeAggregate_WithDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/03-DisableNodeAggregate_WithDimensions.feature @@ -39,8 +39,7 @@ Feature: Disable a node aggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "preceding-nodenborough" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | # We need both a real and a virtual specialization to test the different selection strategies And the command CreateNodeVariant is executed with payload: | Key | Value | diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/05-EnableNodeAggregate_WithoutDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/05-EnableNodeAggregate_WithoutDimensions.feature index 6412f130a09..45ffddb9123 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/05-EnableNodeAggregate_WithoutDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/05-EnableNodeAggregate_WithoutDimensions.feature @@ -37,8 +37,7 @@ Feature: Enable a node aggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "preceding-nodenborough" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | Scenario: Enable a previously disabled node with arbitrary strategy since dimensions are not involved Given the command DisableNodeAggregate is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/06-EnableNodeAggregate_WithDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/06-EnableNodeAggregate_WithDimensions.feature index cd21fb18d63..fe63d7b4ed6 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/06-EnableNodeAggregate_WithDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/06-NodeDisabling/06-EnableNodeAggregate_WithDimensions.feature @@ -40,8 +40,7 @@ Feature: Enable a node aggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "preceding-nodenborough" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | # We need both a real and a virtual specialization to test the different selection strategies And the command CreateNodeVariant is executed with payload: | Key | Value | diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/02-RemoveNodeAggregate_WithoutDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/02-RemoveNodeAggregate_WithoutDimensions.feature index b219e7026a8..7fedbdde418 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/02-RemoveNodeAggregate_WithoutDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/02-RemoveNodeAggregate_WithoutDimensions.feature @@ -36,8 +36,7 @@ Feature: Remove NodeAggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "nodingers-cat" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | Scenario: Remove a node aggregate When the command RemoveNodeAggregate is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/03-RemoveNodeAggregate_WithDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/03-RemoveNodeAggregate_WithDimensions.feature index 543720123f0..8412a984dfc 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/03-RemoveNodeAggregate_WithDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/07-NodeRemoval/03-RemoveNodeAggregate_WithDimensions.feature @@ -38,8 +38,7 @@ Feature: Remove NodeAggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "nodingers-cat" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | Scenario: Remove a node aggregate with strategy allSpecializations When the command RemoveNodeAggregate is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ContentStreamForking/NodeReferencesOnForkContentStream.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ContentStreamForking/NodeReferencesOnForkContentStream.feature index e21dd731eb3..cbb94ebd063 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ContentStreamForking/NodeReferencesOnForkContentStream.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ContentStreamForking/NodeReferencesOnForkContentStream.feature @@ -42,8 +42,7 @@ Feature: On forking a content stream, node references should be copied as well. Given the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "source-nodandaise" | - | referenceName | "referenceProperty" | - | references | [{"target": "anthony-destinode"}] | + | references | {"referenceProperty": [{"target": "anthony-destinode"}]} | # Uses ForkContentStream implicitly When the command CreateWorkspace is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeCopying/CopyNode_NoDimensions.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeCopying/CopyNode_NoDimensions.feature index 7cd82f605ef..2c42faeda4c 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeCopying/CopyNode_NoDimensions.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeCopying/CopyNode_NoDimensions.feature @@ -5,7 +5,9 @@ Feature: Copy nodes (without dimensions) Given using no content dimensions And using the following node types: """yaml - 'Neos.ContentRepository.Testing:Document': [] + 'Neos.ContentRepository.Testing:Document': + references: + ref: [] """ And using identifier "default", I define a content repository And I am in content repository "default" @@ -67,3 +69,24 @@ Feature: Copy nodes (without dimensions) | nodeAggregateIdMapping | {"sir-nodeward-nodington-iii": "sir-nodeward-nodington-iii-copy"} | Then I expect node aggregate identifier "sir-nodeward-nodington-iii-copy" to lead to node cs-identifier;sir-nodeward-nodington-iii-copy;{} + + Scenario: Copy References + When I am in workspace "live" and dimension space point {} + And the command SetNodeReferences is executed with payload: + | Key | Value | + | sourceNodeAggregateId | "sir-nodeward-nodington-iii" | + | references | {"ref": [{"target": "sir-david-nodenborough"}]} | + + Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node cs-identifier;sir-nodeward-nodington-iii;{} + And the command CopyNodesRecursively is executed, copying the current node aggregate with payload: + | Key | Value | + | targetDimensionSpacePoint | {} | + | targetParentNodeAggregateId | "nody-mc-nodeface" | + | targetNodeName | "target-nn" | + | targetSucceedingSiblingnodeAggregateId | null | + | nodeAggregateIdMapping | {"sir-nodeward-nodington-iii": "sir-nodeward-nodington-iii-copy"} | + + And I expect node aggregate identifier "sir-nodeward-nodington-iii-copy" to lead to node cs-identifier;sir-nodeward-nodington-iii-copy;{} + And I expect this node to have the following references: + | Name | Node | Properties | + | ref | cs-identifier;sir-david-nodenborough;{} | null | diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeRemoval/RemoveNodeAggregateAfterDisabling.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeRemoval/RemoveNodeAggregateAfterDisabling.feature index 8c893ec00db..1b59e1932d4 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeRemoval/RemoveNodeAggregateAfterDisabling.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeRemoval/RemoveNodeAggregateAfterDisabling.feature @@ -37,8 +37,7 @@ Feature: Disable a node aggregate And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "preceding-nodenborough" | - | referenceName | "references" | - | references | [{"target": "sir-david-nodenborough"}] | + | references | {"references": [{"target": "sir-david-nodenborough"}]} | Scenario: Restore a hidden node by removing and recreating it Given the command DisableNodeAggregate is executed with payload: diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/References.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/References.feature index 3e67ec9f6d1..0c1e81310f2 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/References.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/References.feature @@ -95,48 +95,39 @@ Feature: Find and count references and their target nodes using the findReferenc And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "a" | - | referenceName | "refs" | - | references | [{"target":"b1"}, {"target":"a2a2"}] | + | references | {"refs": [{"target":"b1"}, {"target":"a2a2"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "b1" | - | referenceName | "ref" | - | references | [{"target":"a"}] | + | references | {"ref": [{"target":"a"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "b" | - | referenceName | "refs" | - | references | [{"target":"a2", "properties": {"foo": "bar"}}, {"target":"a2a1", "properties": {"foo": "baz"}}] | + | references | {"refs": [{"target":"a2", "properties": {"foo": "bar"}}, {"target":"a2a1", "properties": {"foo": "baz"}}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "a" | - | referenceName | "ref" | - | references | [{"target":"b1", "properties": {"foo": "bar"}}] | + | references | {"ref": [{"target":"b1", "properties": {"foo": "bar"}}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "a2" | - | referenceName | "ref" | - | references | [{"target":"a2a3"}] | + | references | {"ref": [{"target":"a2a3"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "a2a3" | - | referenceName | "ref" | - | references | [{"target":"a2"}] | + | references | {"ref": [{"target":"a2"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "b" | - | referenceName | "refs" | - | references | [{"target":"a3", "properties": {"foo": "bar"}}] | + | references | {"refs": [{"target":"a3", "properties": {"foo": "bar"}}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "c" | - | referenceName | "refs" | - | references | [{"target":"b1", "properties": {"foo": "foos"}}] | + | references | {"refs": [{"target":"b1", "properties": {"foo": "foos"}}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | sourceNodeAggregateId | "c" | - | referenceName | "ref" | - | references | [{"target":"b"}] | + | references | {"ref": [{"target":"b"}]} | And the command DisableNodeAggregate is executed with payload: | Key | Value | | nodeAggregateId | "a2a3" | diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/Timestamps.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/Timestamps.feature index c1dc2accc80..642a28f2b71 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/Timestamps.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/NodeTraversal/Timestamps.feature @@ -178,8 +178,7 @@ Feature: Behavior of Node timestamp properties "created", "originalCreated", "la | workspaceName | "user-test" | | sourceOriginDimensionSpacePoint | {"language": "ch"} | | sourceNodeAggregateId | "a" | - | referenceName | "ref" | - | references | [{"target": "b"}] | + | references | {"ref": [{"target": "b"}]} | And I am in workspace "user-test" and dimension space point {"language":"de"} Then I expect the node "a" to have the following timestamps: | created | originalCreated | lastModified | originalLastModified | diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/IntactContentGraph.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/IntactContentGraph.feature index f8a79df5f17..941fb738e8e 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/IntactContentGraph.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/IntactContentGraph.feature @@ -68,7 +68,6 @@ Feature: Create an intact content graph and run integrity violation detection | contentStreamId | "cs-identifier" | | sourceNodeAggregateId | "nody-mc-nodeface" | | affectedSourceOriginDimensionSpacePoints | [{"language":"de"}] | - | referenceName | "referenceProperty" | - | references | [{"targetNodeAggregateId":"sir-david-nodenborough", "properties":null}] | + | references | {"referenceProperty": [{"target":"sir-david-nodenborough", "properties":null}]} | And I run integrity violation detection Then I expect the integrity violation detection result to contain exactly 0 errors diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature index 4ec6cc1f57e..c59da30480b 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/ProjectionIntegrityViolationDetection/ReferenceIntegrityIsProvided.feature @@ -42,8 +42,7 @@ Feature: Run integrity violation detection regarding reference relations | contentStreamId | "cs-identifier" | | sourceNodeAggregateId | "source-nodandaise" | | affectedSourceOriginDimensionSpacePoints | [{"language":"de"}] | - | referenceName | "referenceProperty" | - | references | [{"targetNodeAggregateId":"anthony-destinode", "properties":null}] | + | references | {"referenceProperty": [{"target":"anthony-destinode", "properties":null}]} | And I run integrity violation detection Then I expect the integrity violation detection result to contain exactly 1 error And I expect integrity violation detection result error number 1 to have code 1597919585 @@ -65,8 +64,7 @@ Feature: Run integrity violation detection regarding reference relations | contentStreamId | "cs-identifier" | | sourceNodeAggregateId | "source-nodandaise" | | affectedSourceOriginDimensionSpacePoints | [{"language":"de"}] | - | referenceName | "referenceProperty" | - | references | [{"targetNodeAggregateId":"anthony-destinode", "properties":null}] | + | references | {"referenceProperty": [{"target":"anthony-destinode", "properties":null}]} | And I run integrity violation detection Then I expect the integrity violation detection result to contain exactly 1 error And I expect integrity violation detection result error number 1 to have code 1597919585 diff --git a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/W8-IndividualNodePublication/04-AllFeaturePublication.feature b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/W8-IndividualNodePublication/04-AllFeaturePublication.feature index cefedf691f0..58518bbc1b9 100644 --- a/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/W8-IndividualNodePublication/04-AllFeaturePublication.feature +++ b/Neos.ContentRepository.BehavioralTests/Tests/Behavior/Features/W8-IndividualNodePublication/04-AllFeaturePublication.feature @@ -299,15 +299,13 @@ Feature: Publishing hide/show scenario of nodes | workspaceName | "user-test" | | sourceNodeAggregateId | "sir-david-nodenborough" | | sourceOriginDimensionSpacePoint | {} | - | referenceName | "referenceProperty" | - | references | [{"target":"sir-nodeward-nodington-iii"}] | + | references | {"referenceProperty": [{"target":"sir-nodeward-nodington-iii"}]} | And the command SetNodeReferences is executed with payload: | Key | Value | | workspaceName | "user-test" | | sourceNodeAggregateId | "nody-mc-nodeface" | | sourceOriginDimensionSpacePoint | {} | - | referenceName | "referenceProperty" | - | references | [{"target":"sir-nodeward-nodington-iii"}] | + | references | {"referenceProperty": [{"target":"sir-nodeward-nodington-iii"}]} | When the command PublishIndividualNodesFromWorkspace is executed with payload: | Key | Value | diff --git a/Neos.ContentRepository.Core/Classes/Feature/Common/ConstraintChecks.php b/Neos.ContentRepository.Core/Classes/Feature/Common/ConstraintChecks.php index 9cb894f5d76..f3892a50933 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/Common/ConstraintChecks.php +++ b/Neos.ContentRepository.Core/Classes/Feature/Common/ConstraintChecks.php @@ -32,6 +32,7 @@ use Neos\ContentRepository\Core\Projection\ContentGraph\Filter\FindPrecedingSiblingNodesFilter; use Neos\ContentRepository\Core\Projection\ContentGraph\Filter\FindSucceedingSiblingNodesFilter; use Neos\ContentRepository\Core\Projection\ContentGraph\NodeAggregate; +use Neos\ContentRepository\Core\Projection\ContentGraph\Reference; use Neos\ContentRepository\Core\Projection\ContentGraph\VisibilityConstraints; use Neos\ContentRepository\Core\SharedModel\Exception\ContentStreamDoesNotExistYet; use Neos\ContentRepository\Core\SharedModel\Exception\ContentStreamIsClosed; @@ -243,21 +244,23 @@ protected function requireNodeTypeToAllowNodesOfTypeInReference( } } - protected function requireNodeTypeToAllowNumberOfReferencesInReference(SerializedNodeReferences $nodeReferences, ReferenceName $referenceName, NodeTypeName $nodeTypeName): void + protected function requireNodeTypeToAllowNumberOfReferencesInReference(SerializedNodeReferences $nodeReferences, NodeTypeName $nodeTypeName): void { $nodeType = $this->requireNodeType($nodeTypeName); - $maxItems = $nodeType->getReferences()[$referenceName->value]['constraints']['maxItems'] ?? null; - if ($maxItems === null) { - return; - } + foreach ($nodeReferences->references as $referenceName => $references) { + $maxItems = $nodeType->getReferences()[$referenceName]['constraints']['maxItems'] ?? null; + if ($maxItems === null) { + continue; + } - if ($maxItems < count($nodeReferences)) { - throw ReferenceCannotBeSet::becauseTheItemsCountConstraintsAreNotMatched( - $referenceName, - $nodeTypeName, - count($nodeReferences) - ); + if ($maxItems < count($references)) { + throw ReferenceCannotBeSet::becauseTheItemsCountConstraintsAreNotMatched( + ReferenceName::fromString($referenceName), + $nodeTypeName, + count($references) + ); + } } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/Common/NodeReferencingInternals.php b/Neos.ContentRepository.Core/Classes/Feature/Common/NodeReferencingInternals.php new file mode 100644 index 00000000000..28b8050b9f8 --- /dev/null +++ b/Neos.ContentRepository.Core/Classes/Feature/Common/NodeReferencingInternals.php @@ -0,0 +1,43 @@ +getIterator() as $reference) { + if ($reference instanceof NodeReferenceNameToEmpty) { + $serializedReferences[] = $reference; + continue; + } + + $serializedReferences[] = new SerializedNodeReference( + $reference->referenceName, + $reference->targetNodeAggregateId, + $reference->properties + ? $this->getPropertyConverter()->serializeReferencePropertyValues( + $reference->properties, + $this->requireNodeType($nodeTypeName), + $reference->referenceName + ) : null + ); + } + + return SerializedNodeReferences::fromReferences($serializedReferences); + } +} diff --git a/Neos.ContentRepository.Core/Classes/Feature/Common/TetheredNodeInternals.php b/Neos.ContentRepository.Core/Classes/Feature/Common/TetheredNodeInternals.php index a8717cb2088..6bb6a55ebcb 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/Common/TetheredNodeInternals.php +++ b/Neos.ContentRepository.Core/Classes/Feature/Common/TetheredNodeInternals.php @@ -18,6 +18,7 @@ use Neos\ContentRepository\Core\EventStore\Events; use Neos\ContentRepository\Core\Feature\NodeCreation\Event\NodeAggregateWithNodeWasCreated; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\NodeVariation\Event\NodePeerVariantWasCreated; use Neos\ContentRepository\Core\Infrastructure\Property\PropertyConverter; use Neos\ContentRepository\Core\NodeType\NodeTypeName; @@ -104,6 +105,7 @@ protected function createEventsForMissingTetheredNode( $tetheredNodeTypeDefinition->name, $defaultProperties, NodeAggregateClassification::CLASSIFICATION_TETHERED, + SerializedNodeReferences::createEmpty() ); $creationOriginDimensionSpacePoint = $rootGeneralizationOrigin; } @@ -124,6 +126,7 @@ protected function createEventsForMissingTetheredNode( $tetheredNodeTypeDefinition->name, $defaultProperties, NodeAggregateClassification::CLASSIFICATION_TETHERED, + SerializedNodeReferences::createEmpty(), ) ); } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNode.php b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNode.php index 3616ceda798..c66a07df7ae 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNode.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNode.php @@ -18,6 +18,7 @@ use Neos\ContentRepository\Core\DimensionSpace\OriginDimensionSpacePoint; use Neos\ContentRepository\Core\Feature\NodeCreation\Dto\NodeAggregateIdsByNodePaths; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\PropertyValuesToWrite; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferencesToWrite; use Neos\ContentRepository\Core\NodeType\NodeTypeName; use Neos\ContentRepository\Core\Projection\ContentGraph\ContentSubgraphInterface; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; @@ -34,7 +35,7 @@ */ final readonly class CreateNodeAggregateWithNode implements CommandInterface { - /** + /**x * @param WorkspaceName $workspaceName The workspace in which the create operation is to be performed * @param NodeAggregateId $nodeAggregateId The unique identifier of the node aggregate to create * @param NodeTypeName $nodeTypeName Name of the node type of the new node @@ -44,6 +45,7 @@ * @param NodeAggregateId|null $succeedingSiblingNodeAggregateId Node aggregate id of the node's succeeding sibling (optional). If not given, the node will be added as the parent's first child * @param NodeName|null $nodeName The node's optional name. Set if there is a meaningful relation to its parent that should be named. * @param NodeAggregateIdsByNodePaths $tetheredDescendantNodeAggregateIds Predefined aggregate ids of tethered child nodes per path. For any tethered node that has no matching entry in this set, the node aggregate id is generated randomly. Since tethered nodes may have tethered child nodes themselves, this works for multiple levels ({@see self::withTetheredDescendantNodeAggregateIds()}) + * @param NodeReferencesToWrite|null $references Initial references this node will have (optional). If not given, no references are created */ private function __construct( public WorkspaceName $workspaceName, @@ -55,6 +57,7 @@ private function __construct( public ?NodeAggregateId $succeedingSiblingNodeAggregateId, public ?NodeName $nodeName, public NodeAggregateIdsByNodePaths $tetheredDescendantNodeAggregateIds, + public ?NodeReferencesToWrite $references, ) { } @@ -66,10 +69,11 @@ private function __construct( * @param NodeAggregateId $parentNodeAggregateId The id of the node aggregate underneath which the new node is added * @param NodeAggregateId|null $succeedingSiblingNodeAggregateId Node aggregate id of the node's succeeding sibling (optional). If not given, the node will be added as the parent's first child * @param PropertyValuesToWrite|null $initialPropertyValues The node's initial property values. Will be merged over the node type's default property values + * @param NodeReferencesToWrite|null $references Initial references this node will have (optional). If not given, no references are created */ - public static function create(WorkspaceName $workspaceName, NodeAggregateId $nodeAggregateId, NodeTypeName $nodeTypeName, OriginDimensionSpacePoint $originDimensionSpacePoint, NodeAggregateId $parentNodeAggregateId, ?NodeAggregateId $succeedingSiblingNodeAggregateId = null, ?PropertyValuesToWrite $initialPropertyValues = null): self + public static function create(WorkspaceName $workspaceName, NodeAggregateId $nodeAggregateId, NodeTypeName $nodeTypeName, OriginDimensionSpacePoint $originDimensionSpacePoint, NodeAggregateId $parentNodeAggregateId, ?NodeAggregateId $succeedingSiblingNodeAggregateId = null, ?PropertyValuesToWrite $initialPropertyValues = null, ?NodeReferencesToWrite $references = null): self { - return new self($workspaceName, $nodeAggregateId, $nodeTypeName, $originDimensionSpacePoint, $parentNodeAggregateId, $initialPropertyValues ?: PropertyValuesToWrite::createEmpty(), $succeedingSiblingNodeAggregateId, null, NodeAggregateIdsByNodePaths::createEmpty()); + return new self($workspaceName, $nodeAggregateId, $nodeTypeName, $originDimensionSpacePoint, $parentNodeAggregateId, $initialPropertyValues ?: PropertyValuesToWrite::createEmpty(), $succeedingSiblingNodeAggregateId, null, NodeAggregateIdsByNodePaths::createEmpty(), $references); } public function withInitialPropertyValues(PropertyValuesToWrite $newInitialPropertyValues): self @@ -84,6 +88,7 @@ public function withInitialPropertyValues(PropertyValuesToWrite $newInitialPrope $this->succeedingSiblingNodeAggregateId, $this->nodeName, $this->tetheredDescendantNodeAggregateIds, + $this->references, ); } @@ -127,6 +132,7 @@ public function withTetheredDescendantNodeAggregateIds(NodeAggregateIdsByNodePat $this->succeedingSiblingNodeAggregateId, $this->nodeName, $tetheredDescendantNodeAggregateIds, + $this->references, ); } @@ -148,6 +154,26 @@ public function withNodeName(NodeName $nodeName): self $this->succeedingSiblingNodeAggregateId, $nodeName, $this->tetheredDescendantNodeAggregateIds, + $references, + ); + } + + /** + * Adds references to this creation command + */ + public function withReferences(NodeReferencesToWrite $references): self + { + return new self( + $this->workspaceName, + $this->nodeAggregateId, + $this->nodeTypeName, + $this->originDimensionSpacePoint, + $this->parentNodeAggregateId, + $this->initialPropertyValues, + $this->succeedingSiblingNodeAggregateId, + $this->nodeName, + $this->tetheredDescendantNodeAggregateIds, + $references, ); } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNodeAndSerializedProperties.php b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNodeAndSerializedProperties.php index f295cd6dabe..0c64000710a 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNodeAndSerializedProperties.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Command/CreateNodeAggregateWithNodeAndSerializedProperties.php @@ -20,6 +20,7 @@ use Neos\ContentRepository\Core\Feature\Common\RebasableToOtherWorkspaceInterface; use Neos\ContentRepository\Core\Feature\NodeCreation\Dto\NodeAggregateIdsByNodePaths; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\WorkspacePublication\Dto\NodeIdToPublishOrDiscard; use Neos\ContentRepository\Core\NodeType\NodeTypeName; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; @@ -48,6 +49,7 @@ * @param NodeAggregateId|null $succeedingSiblingNodeAggregateId Node aggregate id of the node's succeeding sibling (optional). If not given, the node will be added as the parent's first child * @param NodeName|null $nodeName The node's optional name. Set if there is a meaningful relation to its parent that should be named. * @param NodeAggregateIdsByNodePaths $tetheredDescendantNodeAggregateIds Predefined aggregate ids of tethered child nodes per path. For any tethered node that has no matching entry in this set, the node aggregate id is generated randomly. Since tethered nodes may have tethered child nodes themselves, this works for multiple levels ({@see self::withTetheredDescendantNodeAggregateIds()}) + * @param SerializedNodeReferences|null $references The node's initial references (serialized). */ private function __construct( public WorkspaceName $workspaceName, @@ -58,7 +60,8 @@ private function __construct( public SerializedPropertyValues $initialPropertyValues, public ?NodeAggregateId $succeedingSiblingNodeAggregateId, public ?NodeName $nodeName, - public NodeAggregateIdsByNodePaths $tetheredDescendantNodeAggregateIds + public NodeAggregateIdsByNodePaths $tetheredDescendantNodeAggregateIds, + public ?SerializedNodeReferences $references, ) { } @@ -70,10 +73,11 @@ private function __construct( * @param NodeAggregateId $parentNodeAggregateId The id of the node aggregate underneath which the new node is added * @param NodeAggregateId|null $succeedingSiblingNodeAggregateId Node aggregate id of the node's succeeding sibling (optional). If not given, the node will be added as the parent's first child * @param SerializedPropertyValues|null $initialPropertyValues The node's initial property values (serialized). Will be merged over the node type's default property values + * @param SerializedNodeReferences|null $references The node's initial references (serialized). */ - public static function create(WorkspaceName $workspaceName, NodeAggregateId $nodeAggregateId, NodeTypeName $nodeTypeName, OriginDimensionSpacePoint $originDimensionSpacePoint, NodeAggregateId $parentNodeAggregateId, NodeAggregateId $succeedingSiblingNodeAggregateId = null, SerializedPropertyValues $initialPropertyValues = null): self + public static function create(WorkspaceName $workspaceName, NodeAggregateId $nodeAggregateId, NodeTypeName $nodeTypeName, OriginDimensionSpacePoint $originDimensionSpacePoint, NodeAggregateId $parentNodeAggregateId, NodeAggregateId $succeedingSiblingNodeAggregateId = null, SerializedPropertyValues $initialPropertyValues = null, SerializedNodeReferences $references = null): self { - return new self($workspaceName, $nodeAggregateId, $nodeTypeName, $originDimensionSpacePoint, $parentNodeAggregateId, $initialPropertyValues ?? SerializedPropertyValues::createEmpty(), $succeedingSiblingNodeAggregateId, null, NodeAggregateIdsByNodePaths::createEmpty()); + return new self($workspaceName, $nodeAggregateId, $nodeTypeName, $originDimensionSpacePoint, $parentNodeAggregateId, $initialPropertyValues ?? SerializedPropertyValues::createEmpty(), $succeedingSiblingNodeAggregateId, null, NodeAggregateIdsByNodePaths::createEmpty(), $references); } /** @@ -98,7 +102,8 @@ public static function fromArray(array $array): self : null, isset($array['tetheredDescendantNodeAggregateIds']) ? NodeAggregateIdsByNodePaths::fromArray($array['tetheredDescendantNodeAggregateIds']) - : NodeAggregateIdsByNodePaths::createEmpty() + : NodeAggregateIdsByNodePaths::createEmpty(), + isset($array['references']) ? SerializedNodeReferences::fromArray($array['references']) : null, ); } @@ -120,7 +125,8 @@ public function withTetheredDescendantNodeAggregateIds( $this->initialPropertyValues, $this->succeedingSiblingNodeAggregateId, $this->nodeName, - $tetheredDescendantNodeAggregateIds + $tetheredDescendantNodeAggregateIds, + $this->references, ); } @@ -173,7 +179,8 @@ public function createCopyForWorkspace( $this->initialPropertyValues, $this->succeedingSiblingNodeAggregateId, $this->nodeName, - $this->tetheredDescendantNodeAggregateIds + $this->tetheredDescendantNodeAggregateIds, + $this->references ); } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Event/NodeAggregateWithNodeWasCreated.php b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Event/NodeAggregateWithNodeWasCreated.php index 0141ffd51e4..2a4c17a0164 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Event/NodeAggregateWithNodeWasCreated.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/Event/NodeAggregateWithNodeWasCreated.php @@ -21,6 +21,7 @@ use Neos\ContentRepository\Core\Feature\Common\InterdimensionalSiblings; use Neos\ContentRepository\Core\Feature\Common\PublishableToWorkspaceInterface; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\NodeType\NodeTypeName; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateClassification; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; @@ -49,6 +50,7 @@ public function __construct( public ?NodeName $nodeName, public SerializedPropertyValues $initialPropertyValues, public NodeAggregateClassification $nodeAggregateClassification, + public SerializedNodeReferences $nodeReferences, ) { } @@ -80,6 +82,7 @@ public function withWorkspaceNameAndContentStreamId(WorkspaceName $targetWorkspa $this->nodeName, $this->initialPropertyValues, $this->nodeAggregateClassification, + $this->nodeReferences, ); } @@ -103,6 +106,7 @@ public static function fromArray(array $values): self isset($values['nodeName']) ? NodeName::fromString($values['nodeName']) : null, SerializedPropertyValues::fromArray($values['initialPropertyValues']), NodeAggregateClassification::from($values['nodeAggregateClassification']), + isset($values['nodeReferences']) ? SerializedNodeReferences::fromArray($values['nodeReferences']) : SerializedNodeReferences::createEmpty(), ); } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/NodeCreation.php b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/NodeCreation.php index a0533f52e97..78a03c31095 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/NodeCreation.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeCreation/NodeCreation.php @@ -22,6 +22,7 @@ use Neos\ContentRepository\Core\Feature\Common\InterdimensionalSiblings; use Neos\ContentRepository\Core\Feature\Common\NodeAggregateEventPublisher; use Neos\ContentRepository\Core\Feature\Common\NodeCreationInternals; +use Neos\ContentRepository\Core\Feature\Common\NodeReferencingInternals; use Neos\ContentRepository\Core\Feature\ContentStreamEventStreamName; use Neos\ContentRepository\Core\Feature\NodeCreation\Command\CreateNodeAggregateWithNode; use Neos\ContentRepository\Core\Feature\NodeCreation\Command\CreateNodeAggregateWithNodeAndSerializedProperties; @@ -29,6 +30,7 @@ use Neos\ContentRepository\Core\Feature\NodeCreation\Event\NodeAggregateWithNodeWasCreated; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\PropertyValuesToWrite; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Infrastructure\Property\PropertyConverter; use Neos\ContentRepository\Core\Infrastructure\Property\PropertyType; use Neos\ContentRepository\Core\NodeType\NodeType; @@ -49,7 +51,7 @@ */ trait NodeCreation { - use NodeCreationInternals; + use NodeCreationInternals, NodeReferencingInternals; abstract protected function getInterDimensionalVariationGraph(): DimensionSpace\InterDimensionalVariationGraph; @@ -86,7 +88,8 @@ private function handleCreateNodeAggregateWithNode( $this->getPropertyConverter()->serializePropertyValues( $command->initialPropertyValues->withoutUnsets(), $this->requireNodeType($command->nodeTypeName) - ) + ), + $command->references ? $this->mapNodeReferencesToSerializedNodeReferences($command->references, $command->nodeTypeName) : SerializedNodeReferences::createEmpty() ); if (!$command->tetheredDescendantNodeAggregateIds->isEmpty()) { $lowLevelCommand = $lowLevelCommand->withTetheredDescendantNodeAggregateIds($command->tetheredDescendantNodeAggregateIds); @@ -250,6 +253,7 @@ private function createRegularWithNode( $command->nodeName, $initialPropertyValues, NodeAggregateClassification::CLASSIFICATION_REGULAR, + $command->references ?? SerializedNodeReferences::createEmpty() ); } @@ -290,6 +294,7 @@ private function handleTetheredChildNodes( $tetheredNodeTypeDefinition->name, $initialPropertyValues, NodeAggregateClassification::CLASSIFICATION_TETHERED, + SerializedNodeReferences::createEmpty(), ); array_push($events, ...iterator_to_array($this->handleTetheredChildNodes( diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferenceSnapshot.php b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferenceSnapshot.php deleted file mode 100644 index 0e6aef7caab..00000000000 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferenceSnapshot.php +++ /dev/null @@ -1,57 +0,0 @@ -values = $values; - } - - /** - * @param array $values - */ - public static function fromArray(array $values): self - { - return new self(SerializedPropertyValues::fromArray($values)); - } - - public function getValues(): SerializedPropertyValues - { - return $this->values; - } - - public function count(): int - { - return count($this->values); - } - - public function jsonSerialize(): SerializedPropertyValues - { - return $this->values; - } -} diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferencesSnapshot.php b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferencesSnapshot.php deleted file mode 100644 index f29ea76f26c..00000000000 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeReferencesSnapshot.php +++ /dev/null @@ -1,106 +0,0 @@ - - * @internal todo not yet finished - */ -final class NodeReferencesSnapshot implements \IteratorAggregate, \Countable, \JsonSerializable -{ - /** - * @var array - */ - private array $references; - - /** - * @param array $references - */ - private function __construct(array $references) - { - $this->references = $references; - } - - public function merge(self $other): self - { - return new self(array_merge($this->references, $other->getReferences())); - } - - /** - * @return array - */ - public function getReferences(): array - { - return $this->references; - } - - /** - * @param array|NodeReferenceSnapshot> $nodeReferences - */ - public static function fromArray(array $nodeReferences): self - { - $values = []; - foreach ($nodeReferences as $nodeReferenceName => $nodeReferenceValue) { - if (is_array($nodeReferenceValue)) { - $values[$nodeReferenceName] = NodeReferenceSnapshot::fromArray($nodeReferenceValue); - } elseif ($nodeReferenceValue instanceof NodeReferenceSnapshot) { - $values[$nodeReferenceName] = $nodeReferenceValue; - } else { - /** @var mixed $nodeReferenceValue */ - throw new \InvalidArgumentException(sprintf( - 'Invalid nodeReferences value. Expected instance of %s, got: %s', - NodeReferenceSnapshot::class, - is_object($nodeReferenceValue) ? get_class($nodeReferenceValue) : gettype($nodeReferenceValue) - ), 1546524480); - } - } - - return new self($values); - } - - /** - * @todo what is this supposed to do? - * Good question. - */ - public static function fromReferences(References $nodeReferences): self - { - $values = []; - - return new self($values); - } - - /** - * @return \Traversable - */ - public function getIterator(): \Traversable - { - yield from $this->references; - } - - public function count(): int - { - return count($this->references); - } - - /** - * @return array - */ - public function jsonSerialize(): array - { - return $this->references; - } -} diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeSubtreeSnapshot.php b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeSubtreeSnapshot.php index 2ca897bc6fc..630d613b8cf 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeSubtreeSnapshot.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/Dto/NodeSubtreeSnapshot.php @@ -5,6 +5,7 @@ namespace Neos\ContentRepository\Core\Feature\NodeDuplication\Dto; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\NodeType\NodeTypeName; use Neos\ContentRepository\Core\Projection\ContentGraph\ContentSubgraphInterface; use Neos\ContentRepository\Core\Projection\ContentGraph\Filter\FindChildNodesFilter; @@ -30,7 +31,7 @@ private function __construct( public ?NodeName $nodeName, public NodeAggregateClassification $nodeAggregateClassification, public SerializedPropertyValues $propertyValues, - public NodeReferencesSnapshot $nodeReferences, + public SerializedNodeReferences $nodeReferences, public array $childNodes ) { foreach ($childNodes as $childNode) { @@ -58,7 +59,7 @@ public static function fromSubgraphAndStartNode(ContentSubgraphInterface $subgra $sourceNode->name, $sourceNode->classification, $properties->serialized(), - NodeReferencesSnapshot::fromReferences( + SerializedNodeReferences::fromReadReferences( $subgraph->findReferences($sourceNode->aggregateId, FindReferencesFilter::create()) ), $childNodes @@ -105,7 +106,7 @@ public static function fromArray(array $array): self isset($array['nodeName']) ? NodeName::fromString($array['nodeName']) : null, NodeAggregateClassification::from($array['nodeAggregateClassification']), SerializedPropertyValues::fromArray($array['propertyValues']), - NodeReferencesSnapshot::fromArray($array['nodeReferences']), + SerializedNodeReferences::fromArray($array['nodeReferences']), $childNodes ); } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/NodeDuplicationCommandHandler.php b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/NodeDuplicationCommandHandler.php index 7ef51daad82..ad680a5ab9f 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/NodeDuplicationCommandHandler.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeDuplication/NodeDuplicationCommandHandler.php @@ -214,6 +214,7 @@ private function createEventsForNodeToInsert( $targetNodeName, $nodeToInsert->propertyValues, $nodeToInsert->nodeAggregateClassification, + $nodeToInsert->nodeReferences, ); foreach ($nodeToInsert->childNodes as $childNodeToInsert) { diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetNodeReferences.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetNodeReferences.php index 4e893eff1a1..70da56b2c2d 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetNodeReferences.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetNodeReferences.php @@ -27,14 +27,12 @@ * @param WorkspaceName $workspaceName The workspace in which the create operation is to be performed * @param NodeAggregateId $sourceNodeAggregateId The identifier of the node aggregate to set references * @param OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint The dimension space for which the references should be set - * @param ReferenceName $referenceName Name of the reference to set * @param NodeReferencesToWrite $references Unserialized reference(s) to set */ private function __construct( public WorkspaceName $workspaceName, public NodeAggregateId $sourceNodeAggregateId, public OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, - public ReferenceName $referenceName, public NodeReferencesToWrite $references, ) { } @@ -43,11 +41,10 @@ private function __construct( * @param WorkspaceName $workspaceName The workspace in which the create operation is to be performed * @param NodeAggregateId $sourceNodeAggregateId The identifier of the node aggregate to set references * @param OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint The dimension space for which the references should be set - * @param ReferenceName $referenceName Name of the reference to set * @param NodeReferencesToWrite $references Unserialized reference(s) to set */ - public static function create(WorkspaceName $workspaceName, NodeAggregateId $sourceNodeAggregateId, OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, ReferenceName $referenceName, NodeReferencesToWrite $references): self + public static function create(WorkspaceName $workspaceName, NodeAggregateId $sourceNodeAggregateId, OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, NodeReferencesToWrite $references): self { - return new self($workspaceName, $sourceNodeAggregateId, $sourceOriginDimensionSpacePoint, $referenceName, $references); + return new self($workspaceName, $sourceNodeAggregateId, $sourceOriginDimensionSpacePoint, $references); } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetSerializedNodeReferences.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetSerializedNodeReferences.php index 3f635af08e0..9e06942a0b1 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetSerializedNodeReferences.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Command/SetSerializedNodeReferences.php @@ -21,8 +21,6 @@ use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\WorkspacePublication\Dto\NodeIdToPublishOrDiscard; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; -use Neos\ContentRepository\Core\SharedModel\Node\ReferenceName; -use Neos\ContentRepository\Core\SharedModel\Workspace\ContentStreamId; use Neos\ContentRepository\Core\SharedModel\Workspace\WorkspaceName; /** @@ -42,14 +40,12 @@ * @param WorkspaceName $workspaceName The workspace in which the create operation is to be performed * @param NodeAggregateId $sourceNodeAggregateId The identifier of the node aggregate to set references * @param OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint The dimension space for which the references should be set - * @param ReferenceName $referenceName Name of the reference to set * @param SerializedNodeReferences $references Serialized reference(s) to set */ private function __construct( public WorkspaceName $workspaceName, public NodeAggregateId $sourceNodeAggregateId, public OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, - public ReferenceName $referenceName, public SerializedNodeReferences $references, ) { } @@ -58,12 +54,11 @@ private function __construct( * @param WorkspaceName $workspaceName The workspace in which the create operation is to be performed * @param NodeAggregateId $sourceNodeAggregateId The identifier of the node aggregate to set references * @param OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint The dimension space for which the references should be set - * @param ReferenceName $referenceName Name of the reference to set * @param SerializedNodeReferences $references Serialized reference(s) to set */ - public static function create(WorkspaceName $workspaceName, NodeAggregateId $sourceNodeAggregateId, OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, ReferenceName $referenceName, SerializedNodeReferences $references): self + public static function create(WorkspaceName $workspaceName, NodeAggregateId $sourceNodeAggregateId, OriginDimensionSpacePoint $sourceOriginDimensionSpacePoint, SerializedNodeReferences $references): self { - return new self($workspaceName, $sourceNodeAggregateId, $sourceOriginDimensionSpacePoint, $referenceName, $references); + return new self($workspaceName, $sourceNodeAggregateId, $sourceOriginDimensionSpacePoint, $references); } /** @@ -75,7 +70,6 @@ public static function fromArray(array $array): self WorkspaceName::fromString($array['workspaceName']), NodeAggregateId::fromString($array['sourceNodeAggregateId']), OriginDimensionSpacePoint::fromArray($array['sourceOriginDimensionSpacePoint']), - ReferenceName::fromString($array['referenceName']), SerializedNodeReferences::fromArray($array['references']), ); } @@ -103,7 +97,6 @@ public function createCopyForWorkspace( $targetWorkspaceName, $this->sourceNodeAggregateId, $this->sourceOriginDimensionSpacePoint, - $this->referenceName, $this->references, ); } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferenceNameToEmpty.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferenceNameToEmpty.php new file mode 100644 index 00000000000..fab5ef4cc10 --- /dev/null +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferenceNameToEmpty.php @@ -0,0 +1,30 @@ + $array + * @param array{"referenceName": string, "target": string, "properties"?: array} $array + * @see NodeReferencesToWrite::fromArray() */ public static function fromArray(array $array): self { return new self( - NodeAggregateId::fromString($array['targetNodeAggregateId']), + ReferenceName::fromString($array['referenceName']), + NodeAggregateId::fromString($array['target']), isset($array['properties']) ? PropertyValuesToWrite::fromArray($array['properties']) : null ); } /** - * @return array + * Provides a limited array representation which can be safely serialized in context of {@see NodeReferencesToWrite} + * + * @return array{"target": NodeAggregateId, "properties": PropertyValuesToWrite|null} */ - public function jsonSerialize(): array + public function targetAndPropertiesToArray(): array { return [ - 'targetNodeAggregateId' => $this->targetNodeAggregateId, + 'target' => $this->targetNodeAggregateId, 'properties' => $this->properties ]; } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferencesToWrite.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferencesToWrite.php index 3fefefbde63..9f840b9994b 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferencesToWrite.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/NodeReferencesToWrite.php @@ -14,8 +14,8 @@ namespace Neos\ContentRepository\Core\Feature\NodeReferencing\Dto; -use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateIds; +use Neos\ContentRepository\Core\SharedModel\Node\ReferenceName; /** * Node references to write, supports arbitrary objects as reference property values. @@ -23,19 +23,38 @@ * * We expect the value types to match the NodeType's property types (this is validated in the command handler). * - * @implements \IteratorAggregate + * @implements \IteratorAggregate * @api used as part of commands */ -final readonly class NodeReferencesToWrite implements \IteratorAggregate, \Countable, \JsonSerializable +final readonly class NodeReferencesToWrite implements \JsonSerializable, \IteratorAggregate { /** - * @var array + * @var array> */ public array $references; - private function __construct(NodeReferenceToWrite ...$references) + private function __construct(NodeReferenceToWrite|NodeReferenceNameToEmpty ...$references) { - $this->references = $references; + $resultingReferences = []; + foreach ($references as $reference) { + $referenceNameExists = isset($resultingReferences[$reference->referenceName->value]); + if ($reference instanceof NodeReferenceNameToEmpty) { + if ($referenceNameExists && count($resultingReferences[$reference->referenceName->value]) > 0) { + throw new \InvalidArgumentException(sprintf('You cannot set references for the ReferenceName %s while also deleting references for the same name.', $reference->referenceName->value), 1718193611); + } + $resultingReferences[$reference->referenceName->value] = []; + continue; + } + if ($referenceNameExists && count($resultingReferences[$reference->referenceName->value]) === 0) { + throw new \InvalidArgumentException(sprintf('You cannot set references for the ReferenceName %s while also deleting references for the same name.', $reference->referenceName->value), 1718193720); + } + if (!$referenceNameExists) { + $resultingReferences[$reference->referenceName->value] = []; + } + $resultingReferences[$reference->referenceName->value][] = $reference; + + } + $this->references = $resultingReferences; } /** @@ -47,32 +66,34 @@ public static function fromReferences(array $references): self } /** - * @param array> $values + * @param array}>> $namesAndReferences */ - public static function fromArray(array $values): self + public static function fromArray(array $namesAndReferences): self { - return new self(...array_map( - fn (array $serializedReference): NodeReferenceToWrite - => NodeReferenceToWrite::fromArray($serializedReference), - $values - )); - } + $result = []; + foreach ($namesAndReferences as $name => $references) { + if ($references === []) { + $result[] = new NodeReferenceNameToEmpty(ReferenceName::fromString($name)); + continue; + } - /** - * Unset all references for this reference name. - */ - public static function createEmpty(): self - { - return new self(); + $result = [ + ...$result, + ...array_map(static function ($serializedReference) use ($name) { + $serializedReference['referenceName'] = $name; + return NodeReferenceToWrite::fromArray($serializedReference); + }, $references) + ]; + } + + return new self(...$result); } - public static function fromNodeAggregateIds(NodeAggregateIds $nodeAggregateIds): self + public static function fromReferenceNameAndNodeAggregateIds(ReferenceName $referenceName, NodeAggregateIds $nodeAggregateIds): self { - return new self(...array_map( - fn (NodeAggregateId $nodeAggregateId): NodeReferenceToWrite - => new NodeReferenceToWrite($nodeAggregateId, null), - iterator_to_array($nodeAggregateIds) - )); + return new self(...array_map(static function ($nodeAggregateId) use ($referenceName) { + return new NodeReferenceToWrite($referenceName, $nodeAggregateId, null); + }, iterator_to_array($nodeAggregateIds))); } /** @@ -83,24 +104,31 @@ public static function fromJsonString(string $jsonString): self return self::fromArray(\json_decode($jsonString, true, 512, JSON_THROW_ON_ERROR)); } + public function merge(NodeReferencesToWrite $nodeReferencesToWrite): self + { + return new self(...$this->getIterator(), ...$nodeReferencesToWrite->getIterator()); + } + /** - * @return \Traversable + * @return \Traversable */ public function getIterator(): \Traversable { - yield from $this->references; + foreach ($this->references as $name => $references) { + if ($references === []) { + yield new NodeReferenceNameToEmpty(ReferenceName::fromString($name)); + } + foreach ($references as $reference) { + yield $reference; + } + } } /** - * @return array + * @return array>> */ public function jsonSerialize(): array { - return $this->references; - } - - public function count(): int - { - return count($this->references); + return array_map(static fn(array $referenceToWriteObjects) => array_map(static fn(NodeReferenceToWrite $nodeReference) => $nodeReference->targetAndPropertiesToArray(), $referenceToWriteObjects), $this->references); } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReference.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReference.php index 2efe13b1778..b4f144fb270 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReference.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReference.php @@ -16,38 +16,41 @@ use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; +use Neos\ContentRepository\Core\SharedModel\Node\ReferenceName; /** * "Raw" / Serialized node reference as saved in the event log // in projections. * - * @internal + * @internal implementation detail of {@see SerializedNodeReferences} */ -final readonly class SerializedNodeReference implements \JsonSerializable +final readonly class SerializedNodeReference { public function __construct( + public ReferenceName $referenceName, public NodeAggregateId $targetNodeAggregateId, public ?SerializedPropertyValues $properties ) { } /** - * @param array $array + * @param array{"referenceName": string, "target": string, "properties"?: array} $array */ public static function fromArray(array $array): self { return new self( - NodeAggregateId::fromString($array['targetNodeAggregateId']), - $array['properties'] ? SerializedPropertyValues::fromArray($array['properties']) : null + ReferenceName::fromString($array['referenceName']), + NodeAggregateId::fromString($array['target']), + isset($array['properties']) ? SerializedPropertyValues::fromArray($array['properties']) : null ); } /** - * @return array + * @return array{"target": NodeAggregateId, "properties": SerializedPropertyValues|null} */ - public function jsonSerialize(): array + public function targetAndPropertiesToArray(): array { return [ - 'targetNodeAggregateId' => $this->targetNodeAggregateId, + 'target' => $this->targetNodeAggregateId, 'properties' => $this->properties ]; } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReferences.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReferences.php index 4fbe1fb6737..bb03c219299 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReferences.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Dto/SerializedNodeReferences.php @@ -14,32 +14,53 @@ namespace Neos\ContentRepository\Core\Feature\NodeReferencing\Dto; -use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; +use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Projection\ContentGraph\References; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateIds; +use Neos\ContentRepository\Core\SharedModel\Node\ReferenceName; /** * A collection of SerializedNodeReference objects, to be used when creating reference relations. * - * @implements \IteratorAggregate - * @internal + * @implements \IteratorAggregate + * @api used in commands and events */ -final readonly class SerializedNodeReferences implements \IteratorAggregate, \Countable, \JsonSerializable +final readonly class SerializedNodeReferences implements \JsonSerializable, \IteratorAggregate { /** - * @var array + * @var array> */ public array $references; - private function __construct(SerializedNodeReference ...$references) + private function __construct(SerializedNodeReference|NodeReferenceNameToEmpty ...$references) { $existingTargets = []; + $resultingReferences = []; foreach ($references as $reference) { - if (isset($existingTargets[$reference->targetNodeAggregateId->value])) { + $referenceNameExists = isset($resultingReferences[$reference->referenceName->value]); + if ($reference instanceof NodeReferenceNameToEmpty) { + if ($referenceNameExists && count($resultingReferences[$reference->referenceName->value]) > 0) { + throw new \InvalidArgumentException(sprintf('You cannot delete all references for the ReferenceName "%s" while also adding references for the same name.', $reference->referenceName->value), 1718193611); + } + $resultingReferences[$reference->referenceName->value] = []; + continue; + } + + if (isset($existingTargets[$reference->referenceName->value][$reference->targetNodeAggregateId->value])) { throw new \InvalidArgumentException(sprintf('Duplicate entry in references to write. Target "%s" already exists in collection.', $reference->targetNodeAggregateId->value), 1700150910); } - $existingTargets[$reference->targetNodeAggregateId->value] = true; + if ($referenceNameExists && $resultingReferences[$reference->referenceName->value] === []) { + throw new \InvalidArgumentException(sprintf('You cannot set references for the ReferenceName "%s" after deleting references for the same name.', $reference->referenceName->value), 1718193720); + } + + if (!$referenceNameExists) { + $resultingReferences[$reference->referenceName->value] = []; + } + + $existingTargets[$reference->referenceName->value][$reference->targetNodeAggregateId->value] = true; + $resultingReferences[$reference->referenceName->value][] = $reference; } - $this->references = $references; + $this->references = $resultingReferences; } /** @@ -51,23 +72,44 @@ public static function fromReferences(array $references): self } /** - * @param array> $referenceData + * @param array}>> $namesAndReferences */ - public static function fromArray(array $referenceData): self + public static function fromArray(array $namesAndReferences): self + { + $result = []; + foreach ($namesAndReferences as $name => $references) { + if ($references === []) { + $result[] = new NodeReferenceNameToEmpty(ReferenceName::fromString($name)); + continue; + } + + $result = [ + ...$result, + ...array_map(static function ($serializedReference) use ($name): SerializedNodeReference { + $serializedReference['referenceName'] = $name; + return SerializedNodeReference::fromArray($serializedReference); + }, $references) + ]; + } + + return new self(...$result); + } + + public static function fromReferenceNameAndNodeAggregateIds(ReferenceName $referenceName, NodeAggregateIds $nodeAggregateIds): self { - return new self(...array_map( - fn (array $referenceDatum): SerializedNodeReference => SerializedNodeReference::fromArray($referenceDatum), - $referenceData - )); + return new self(...array_map(static function ($nodeAggregateId) use ($referenceName) { + return new SerializedNodeReference($referenceName, $nodeAggregateId, null); + }, iterator_to_array($nodeAggregateIds))); } - public static function fromNodeAggregateIds(NodeAggregateIds $nodeAggregateIds): self + public static function fromReadReferences(References $references): self { - return new self(...array_map( - static fn (NodeAggregateId $nodeAggregateId): SerializedNodeReference - => new SerializedNodeReference($nodeAggregateId, null), - iterator_to_array($nodeAggregateIds) - )); + $serializedReferences = []; + foreach ($references as $reference) { + $serializedReferences[] = new SerializedNodeReference($reference->name, $reference->node->aggregateId, $reference->properties ? $reference->properties->serialized() : SerializedPropertyValues::createEmpty()); + } + + return new self(...$serializedReferences); } public static function fromJsonString(string $jsonString): self @@ -75,29 +117,57 @@ public static function fromJsonString(string $jsonString): self return self::fromArray(\json_decode($jsonString, true)); } - public function merge(self $other): self + public static function createEmpty(): self { - return new self(...array_merge($this->references, $other->references)); + return new self(); + } + + public function merge(SerializedNodeReferences $nodeReferencesToWrite): self + { + return new self(...$this->getIterator(), ...$nodeReferencesToWrite->getIterator()); } /** - * @return \Traversable + * @return \Traversable */ public function getIterator(): \Traversable { - yield from $this->references; + foreach ($this->references as $name => $references) { + if ($references === []) { + yield new NodeReferenceNameToEmpty(ReferenceName::fromString($name)); + } + foreach ($references as $reference) { + yield $reference; + } + } } - public function count(): int + /** + * @return ReferenceName[] + */ + public function getReferenceNames(): array { - return count($this->references); + return array_map(static fn(string $name) => ReferenceName::fromString($name), array_keys($this->references)); + } + + public function getForReferenceName(ReferenceName $referenceName): SerializedNodeReferences + { + if (!isset($this->references[$referenceName->value])) { + throw new \InvalidArgumentException(sprintf('The given ReferenceName "%s" is not set in this instance of SerializedNodeReference.', $referenceName->value), 1718264159); + } + + if ($this->references[$referenceName->value] === []) { + return new self(new NodeReferenceNameToEmpty($referenceName)); + } + + return new self(...$this->references[$referenceName->value]); } /** - * @return array + * @return array>> */ public function jsonSerialize(): array { - return $this->references; + return array_map(static fn(array $referenceToWriteObjects) => array_map(static fn(SerializedNodeReference $nodeReference) => $nodeReference->targetAndPropertiesToArray(), $referenceToWriteObjects), $this->references); } } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Event/NodeReferencesWereSet.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Event/NodeReferencesWereSet.php index 6db631b6430..14e1fcc896e 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Event/NodeReferencesWereSet.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/Event/NodeReferencesWereSet.php @@ -10,7 +10,6 @@ use Neos\ContentRepository\Core\Feature\Common\PublishableToWorkspaceInterface; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; -use Neos\ContentRepository\Core\SharedModel\Node\ReferenceName; use Neos\ContentRepository\Core\SharedModel\Workspace\ContentStreamId; use Neos\ContentRepository\Core\SharedModel\Workspace\WorkspaceName; @@ -37,7 +36,6 @@ public function __construct( * declared for the given reference in the node aggregate's type */ public OriginDimensionSpacePointSet $affectedSourceOriginDimensionSpacePoints, - public ReferenceName $referenceName, public SerializedNodeReferences $references, ) { } @@ -59,7 +57,6 @@ public function withWorkspaceNameAndContentStreamId(WorkspaceName $targetWorkspa $contentStreamId, $this->nodeAggregateId, $this->affectedSourceOriginDimensionSpacePoints, - $this->referenceName, $this->references, ); } @@ -73,7 +70,6 @@ public static function fromArray(array $values): self ? NodeAggregateId::fromString($values['sourceNodeAggregateId']) : NodeAggregateId::fromString($values['nodeAggregateId']), OriginDimensionSpacePointSet::fromArray($values['affectedSourceOriginDimensionSpacePoints']), - ReferenceName::fromString($values['referenceName']), SerializedNodeReferences::fromArray($values['references']), ); } diff --git a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/NodeReferencing.php b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/NodeReferencing.php index 6749780c5ed..a49dfde6c92 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/NodeReferencing.php +++ b/Neos.ContentRepository.Core/Classes/Feature/NodeReferencing/NodeReferencing.php @@ -19,13 +19,17 @@ use Neos\ContentRepository\Core\EventStore\EventsToPublish; use Neos\ContentRepository\Core\Feature\Common\ConstraintChecks; use Neos\ContentRepository\Core\Feature\Common\NodeAggregateEventPublisher; +use Neos\ContentRepository\Core\Feature\Common\NodeReferencingInternals; use Neos\ContentRepository\Core\Feature\ContentStreamEventStreamName; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\PropertyScope; use Neos\ContentRepository\Core\Feature\NodeReferencing\Command\SetNodeReferences; use Neos\ContentRepository\Core\Feature\NodeReferencing\Command\SetSerializedNodeReferences; -use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferenceToWrite; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferenceNameToEmpty; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferencesToWrite; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReference; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\NodeReferencing\Event\NodeReferencesWereSet; +use Neos\ContentRepository\Core\NodeType\NodeTypeName; use Neos\ContentRepository\Core\Projection\ContentGraph\ContentGraphInterface; use Neos\ContentRepository\Core\Projection\ContentGraph\NodeAggregate; use Neos\ContentRepository\Core\SharedModel\Exception\ContentStreamDoesNotExistYet; @@ -36,7 +40,7 @@ */ trait NodeReferencing { - use ConstraintChecks; + use ConstraintChecks, NodeReferencingInternals; abstract protected function requireProjectedNodeAggregate( ContentGraphInterface $contentGraph, @@ -59,9 +63,12 @@ private function handleSetNodeReferences( $nodeTypeName = $sourceNodeAggregate->nodeTypeName; foreach ($command->references as $reference) { + if ($reference instanceof NodeReferenceNameToEmpty) { + continue; + } if ($reference->properties) { $this->validateReferenceProperties( - $command->referenceName, + $reference->referenceName, $reference->properties, $nodeTypeName ); @@ -72,20 +79,7 @@ private function handleSetNodeReferences( $command->workspaceName, $command->sourceNodeAggregateId, $command->sourceOriginDimensionSpacePoint, - $command->referenceName, - Dto\SerializedNodeReferences::fromReferences(array_map( - fn (NodeReferenceToWrite $reference): SerializedNodeReference => new SerializedNodeReference( - $reference->targetNodeAggregateId, - $reference->properties - ? $this->getPropertyConverter()->serializeReferencePropertyValues( - $reference->properties, - $this->requireNodeType($nodeTypeName), - $command->referenceName - ) - : null - ), - $command->references->references - )), + $this->mapNodeReferencesToSerializedNodeReferences($command->references, $nodeTypeName), ); return $this->handleSetSerializedNodeReferences($lowLevelCommand, $commandHandlingDependencies); @@ -112,52 +106,59 @@ private function handleSetSerializedNodeReferences( $sourceNodeAggregate, $command->sourceOriginDimensionSpacePoint ); - $this->requireNodeTypeToDeclareReference($sourceNodeAggregate->nodeTypeName, $command->referenceName); + + $sourceNodeType = $this->requireNodeType($sourceNodeAggregate->nodeTypeName); + $events = []; $this->requireNodeTypeToAllowNumberOfReferencesInReference( $command->references, - $command->referenceName, $sourceNodeAggregate->nodeTypeName ); - foreach ($command->references as $reference) { - assert($reference instanceof SerializedNodeReference); - $destinationNodeAggregate = $this->requireProjectedNodeAggregate( - $contentGraph, - $reference->targetNodeAggregateId - ); - $this->requireNodeAggregateToNotBeRoot($destinationNodeAggregate); - $this->requireNodeAggregateToCoverDimensionSpacePoint( - $destinationNodeAggregate, - $command->sourceOriginDimensionSpacePoint->toDimensionSpacePoint() - ); - $this->requireNodeTypeToAllowNodesOfTypeInReference( - $sourceNodeAggregate->nodeTypeName, - $command->referenceName, - $destinationNodeAggregate->nodeTypeName - ); - } - - $sourceNodeType = $this->requireNodeType($sourceNodeAggregate->nodeTypeName); - $scopeDeclaration = $sourceNodeType->getReferences()[$command->referenceName->value]['scope'] ?? ''; - $scope = PropertyScope::tryFrom($scopeDeclaration) ?: PropertyScope::SCOPE_NODE; + foreach ($command->references->getReferenceNames() as $referenceName) { + $this->requireNodeTypeToDeclareReference($sourceNodeAggregate->nodeTypeName, $referenceName); + $scopeDeclaration = $sourceNodeType->getReferences()[$referenceName->value]['scope'] ?? ''; + $scope = PropertyScope::tryFrom($scopeDeclaration) ?: PropertyScope::SCOPE_NODE; + // TODO: Optimize affected sets into one event + $affectedReferences = $command->references->getForReferenceName($referenceName); + + foreach ($affectedReferences as $reference) { + assert($reference instanceof SerializedNodeReference || $reference instanceof NodeReferenceNameToEmpty); + if ($reference instanceof NodeReferenceNameToEmpty) { + continue; + } + $destinationNodeAggregate = $this->requireProjectedNodeAggregate( + $contentGraph, + $reference->targetNodeAggregateId + ); + $this->requireNodeAggregateToNotBeRoot($destinationNodeAggregate); + $this->requireNodeAggregateToCoverDimensionSpacePoint( + $destinationNodeAggregate, + $command->sourceOriginDimensionSpacePoint->toDimensionSpacePoint() + ); + $this->requireNodeTypeToAllowNodesOfTypeInReference( + $sourceNodeAggregate->nodeTypeName, + $reference->referenceName, + $destinationNodeAggregate->nodeTypeName + ); + } - $affectedOrigins = $scope->resolveAffectedOrigins( - $command->sourceOriginDimensionSpacePoint, - $sourceNodeAggregate, - $this->interDimensionalVariationGraph - ); + $affectedOrigins = $scope->resolveAffectedOrigins( + $command->sourceOriginDimensionSpacePoint, + $sourceNodeAggregate, + $this->interDimensionalVariationGraph + ); - $events = Events::with( - new NodeReferencesWereSet( + $events[] = new NodeReferencesWereSet( $contentGraph->getWorkspaceName(), $contentGraph->getContentStreamId(), $command->sourceNodeAggregateId, $affectedOrigins, - $command->referenceName, - $command->references, - ) - ); + $affectedReferences, + ); + } + + $events = Events::fromArray($events); return new EventsToPublish( ContentStreamEventStreamName::fromContentStreamId($contentGraph->getContentStreamId()) diff --git a/Neos.ContentRepository.Core/Classes/Feature/RootNodeCreation/RootNodeHandling.php b/Neos.ContentRepository.Core/Classes/Feature/RootNodeCreation/RootNodeHandling.php index f09fc214aaf..d16fedd7c04 100644 --- a/Neos.ContentRepository.Core/Classes/Feature/RootNodeCreation/RootNodeHandling.php +++ b/Neos.ContentRepository.Core/Classes/Feature/RootNodeCreation/RootNodeHandling.php @@ -25,6 +25,7 @@ use Neos\ContentRepository\Core\Feature\NodeCreation\Dto\NodeAggregateIdsByNodePaths; use Neos\ContentRepository\Core\Feature\NodeCreation\Event\NodeAggregateWithNodeWasCreated; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\SerializedPropertyValues; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\RootNodeCreation\Command\CreateRootNodeAggregateWithNode; use Neos\ContentRepository\Core\Feature\RootNodeCreation\Command\UpdateRootNodeAggregateDimensions; use Neos\ContentRepository\Core\Feature\RootNodeCreation\Event\RootNodeAggregateDimensionsWereUpdated; @@ -254,6 +255,7 @@ private function createTetheredWithNodeForRoot( $nodeName, $initialPropertyValues, NodeAggregateClassification::CLASSIFICATION_TETHERED, + SerializedNodeReferences::createEmpty(), ); } } diff --git a/Neos.ContentRepository.Core/Classes/Projection/CatchUp.php b/Neos.ContentRepository.Core/Classes/Projection/CatchUp.php index b997978bfe7..4447393fb22 100644 --- a/Neos.ContentRepository.Core/Classes/Projection/CatchUp.php +++ b/Neos.ContentRepository.Core/Classes/Projection/CatchUp.php @@ -107,6 +107,7 @@ public function run(EventStreamInterface $eventStream): SequenceNumber try { ($this->eventHandler)($eventEnvelope); } catch (\Exception $e) { + throw $e; throw new \RuntimeException(sprintf('Exception while catching up to sequence number %d', $eventEnvelope->sequenceNumber->value), 1710707311, $e); } $iteration++; diff --git a/Neos.ContentRepository.LegacyNodeMigration/Classes/NodeDataToEventsProcessor.php b/Neos.ContentRepository.LegacyNodeMigration/Classes/NodeDataToEventsProcessor.php index 56aaa7fbad7..b06bcb4ba1f 100644 --- a/Neos.ContentRepository.LegacyNodeMigration/Classes/NodeDataToEventsProcessor.php +++ b/Neos.ContentRepository.LegacyNodeMigration/Classes/NodeDataToEventsProcessor.php @@ -21,6 +21,7 @@ use Neos\ContentRepository\Core\Feature\NodeModification\Dto\PropertyValuesToWrite; use Neos\ContentRepository\Core\Feature\NodeModification\Event\NodePropertiesWereSet; use Neos\ContentRepository\Core\Feature\NodeMove\Event\NodeAggregateWasMoved; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReference; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\SerializedNodeReferences; use Neos\ContentRepository\Core\Feature\NodeReferencing\Event\NodeReferencesWereSet; use Neos\ContentRepository\Core\Feature\NodeVariation\Event\NodeGeneralizationVariantWasCreated; @@ -275,6 +276,12 @@ public function processNodeDataWithoutFallbackToEmptyDimension(NodeAggregateId $ $serializedPropertyValuesAndReferences = $this->extractPropertyValuesAndReferences($nodeDataRow, $nodeType); + $references = SerializedNodeReferences::createEmpty(); + foreach ($serializedPropertyValuesAndReferences->references as $referencePropertyName => $destinationNodeAggregateIds) { + $localReferences = SerializedNodeReferences::fromReferenceNameAndNodeAggregateIds(ReferenceName::fromString($referencePropertyName), $destinationNodeAggregateIds); + $references = $references->merge($localReferences); + } + if ($this->isAutoCreatedChildNode($parentNodeAggregate->nodeTypeName, $nodeName) && !$this->visitedNodes->containsNodeAggregate($nodeAggregateId)) { // Create tethered node if the node was not found before. // If the node was already visited, we want to create a node variant (and keep the tethering status) @@ -291,6 +298,7 @@ public function processNodeDataWithoutFallbackToEmptyDimension(NodeAggregateId $ $nodeName, $serializedPropertyValuesAndReferences->serializedPropertyValues, NodeAggregateClassification::CLASSIFICATION_TETHERED, + $references, ) ); } elseif ($this->visitedNodes->containsNodeAggregate($nodeAggregateId)) { @@ -314,6 +322,7 @@ public function processNodeDataWithoutFallbackToEmptyDimension(NodeAggregateId $ $nodeName, $serializedPropertyValuesAndReferences->serializedPropertyValues, NodeAggregateClassification::CLASSIFICATION_REGULAR, + $references, ) ); } @@ -322,7 +331,7 @@ public function processNodeDataWithoutFallbackToEmptyDimension(NodeAggregateId $ $this->exportEvent(new SubtreeWasTagged($this->workspaceName, $this->contentStreamId, $nodeAggregateId, $this->interDimensionalVariationGraph->getSpecializationSet($originDimensionSpacePoint->toDimensionSpacePoint(), true, $this->visitedNodes->alreadyVisitedOriginDimensionSpacePoints($nodeAggregateId)->toDimensionSpacePointSet()), SubtreeTag::disabled())); } foreach ($serializedPropertyValuesAndReferences->references as $referencePropertyName => $destinationNodeAggregateIds) { - $this->nodeReferencesWereSetEvents[] = new NodeReferencesWereSet($this->workspaceName, $this->contentStreamId, $nodeAggregateId, new OriginDimensionSpacePointSet([$originDimensionSpacePoint]), ReferenceName::fromString($referencePropertyName), SerializedNodeReferences::fromNodeAggregateIds($destinationNodeAggregateIds)); + $this->nodeReferencesWereSetEvents[] = new NodeReferencesWereSet($this->workspaceName, $this->contentStreamId, $nodeAggregateId, new OriginDimensionSpacePointSet([$originDimensionSpacePoint]), SerializedNodeReferences::fromReferences(array_map(static function ($nodeAggregateId) use ($referencePropertyName) {return new SerializedNodeReference(ReferenceName::fromString($referencePropertyName), $nodeAggregateId, null);}, iterator_to_array($destinationNodeAggregateIds->getIterator())))); } $this->visitedNodes->add($nodeAggregateId, new DimensionSpacePointSet([$originDimensionSpacePoint->toDimensionSpacePoint()]), $nodeTypeName, $nodePath, $parentNodeAggregate->nodeAggregateId); diff --git a/Neos.ContentRepository.TestSuite/Classes/Behavior/Features/Bootstrap/Features/NodeReferencing.php b/Neos.ContentRepository.TestSuite/Classes/Behavior/Features/Bootstrap/Features/NodeReferencing.php index 19fc646c11c..eede0dcdf6d 100644 --- a/Neos.ContentRepository.TestSuite/Classes/Behavior/Features/Bootstrap/Features/NodeReferencing.php +++ b/Neos.ContentRepository.TestSuite/Classes/Behavior/Features/Bootstrap/Features/NodeReferencing.php @@ -19,6 +19,7 @@ use Neos\ContentRepository\Core\Feature\ContentStreamEventStreamName; use Neos\ContentRepository\Core\Feature\NodeModification\Dto\PropertyValuesToWrite; use Neos\ContentRepository\Core\Feature\NodeReferencing\Command\SetNodeReferences; +use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferenceNameToEmpty; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferencesToWrite; use Neos\ContentRepository\Core\Feature\NodeReferencing\Dto\NodeReferenceToWrite; use Neos\ContentRepository\Core\SharedModel\Node\NodeAggregateId; @@ -55,23 +56,13 @@ public function theCommandSetNodeReferencesIsExecutedWithPayload(TableNode $payl $sourceOriginDimensionSpacePoint = isset($commandArguments['sourceOriginDimensionSpacePoint']) ? OriginDimensionSpacePoint::fromArray($commandArguments['sourceOriginDimensionSpacePoint']) : OriginDimensionSpacePoint::fromDimensionSpacePoint($this->currentDimensionSpacePoint); - $references = NodeReferencesToWrite::fromReferences( - array_map( - fn (array $referenceData): NodeReferenceToWrite => new NodeReferenceToWrite( - NodeAggregateId::fromString($referenceData['target']), - isset($referenceData['properties']) - ? $this->deserializeProperties($referenceData['properties']) - : null - ), - $commandArguments['references'] - ) - ); + + $references = $this->mapRawNodeReferencesToNodeReferencesToWrite($commandArguments['references']); $command = SetNodeReferences::create( $workspaceName, NodeAggregateId::fromString($commandArguments['sourceNodeAggregateId']), $sourceOriginDimensionSpacePoint, - ReferenceName::fromString($commandArguments['referenceName']), $references, ); @@ -107,4 +98,33 @@ public function theEventNodeReferencesWereSetWasPublishedWithPayload(TableNode $ $this->publishEvent('NodeReferencesWereSet', $streamName->getEventStreamName(), $eventPayload); } + + protected function mapRawNodeReferencesToNodeReferencesToWrite(array $deserializedTableContent): NodeReferencesToWrite + { + $references = []; + foreach ($deserializedTableContent as $rawReferenceName => $rawReferences) { + $referenceName = ReferenceName::fromString($rawReferenceName); + if ($rawReferences === []) { + $references[] = new NodeReferenceNameToEmpty($referenceName); + continue; + } + + $references = [ + ...$references, + ...array_map( + fn(array $referenceData): NodeReferenceToWrite => new NodeReferenceToWrite( + $referenceName, + NodeAggregateId::fromString($referenceData['target']), + isset($referenceData['properties']) + ? $this->deserializeProperties($referenceData['properties']) + : null + ), + $rawReferences + ) + ]; + } + + return NodeReferencesToWrite::fromReferences($references); + } + }