Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

!!! FEATURE: Publishing Version 3 #5301

Merged
Show file tree
Hide file tree
Changes from 35 commits
Commits
Show all changes
57 commits
Select commit Hold shift + click to select a range
c0d6478
WIP task/schnappsidee-rebase-and-partial-publish-in-memory-event-stream
mhsdesign Oct 19, 2024
7190f83
WIP: Commit for command handling in inSimulation
mhsdesign Oct 21, 2024
f47a5e6
WIP: handlePublishIndividualNodesFromWorkspace with remaining events …
mhsdesign Oct 21, 2024
3f9d534
WIP: move handleRebaseWorkspace to run inSimulation
mhsdesign Oct 21, 2024
5f72ead
WIP: refactor handleDiscardIndividualNodesFromWorkspace to run inSimu…
mhsdesign Oct 21, 2024
686d522
WIP todos
mhsdesign Oct 21, 2024
40cebe8
Merge remote-tracking branch 'origin/9.0' into task/schnappsidee-reba…
mhsdesign Oct 22, 2024
b99a498
TASK: Adjustments after merge of cr first level projection
mhsdesign Oct 22, 2024
6847e5c
Merge branch 'task/schnappsidee-zwo-yield-events-to-publish-in-worksp…
mhsdesign Oct 23, 2024
13de53c
TASK: Introduce CommandSimulator and refactor publish partially and r…
mhsdesign Oct 23, 2024
3f0db1d
TASK: Remove now obsolete `ForkContentStream` command
mhsdesign Oct 23, 2024
c766f0b
TASK: Adjust todo comments
mhsdesign Oct 23, 2024
f1c8af6
BUGFIX: Partial publish breaks uri and change Projection
kitsunet Oct 16, 2024
605715b
TASK: Adjust test 02-RebasingWithAutoCreatedNodes.feature
mhsdesign Oct 23, 2024
3b85cf8
TASK: Adjust exceptions to be a noop if PublishIndividualNodesFromWor…
mhsdesign Oct 23, 2024
20c3827
TASK: Introduce test for quick path if rebased commands emitted 0 events
mhsdesign Oct 23, 2024
6343694
TASK: Add quick paths for partial discarding with tests
mhsdesign Oct 23, 2024
e0815bc
Add todos to optimise partial publish to full publish and discard all
mhsdesign Oct 23, 2024
824ca76
TASK: Make publish all a no-op if there are no changes and also ensur…
mhsdesign Oct 23, 2024
96e58ea
TASK: Remove PublishIndividualNodesFromWorkspace::contentStreamIdForM…
mhsdesign Oct 23, 2024
31cfcc5
Workspace aware simulation
kitsunet Oct 23, 2024
aeee478
TASK: Remove outdated comments and adjust variable namings
mhsdesign Oct 23, 2024
77719c9
Cosmetics & Nagellack
mhsdesign Oct 23, 2024
a54a081
TASK: Remove obsolete check to skip events
mhsdesign Oct 24, 2024
653bb8b
Merge remote-tracking branch 'origin/9.0' into task/schnappsidee-reba…
mhsdesign Oct 24, 2024
a0dd931
TASK: Remove extra interface for workspace command handler and combin…
mhsdesign Oct 24, 2024
40233dc
TASK: Use `currentSequenceNumber` instead of calculating it ourselves
mhsdesign Oct 24, 2024
e345f7c
BUGFIX: Move workspace pointer BEFORE remaining events are emitted
mhsdesign Oct 24, 2024
0d0da5c
TASK: Remove old content-stream after rebase
mhsdesign Oct 24, 2024
c63b019
TASK: Ensure that dangling content streams are removed after publish …
mhsdesign Oct 24, 2024
9c06afc
TASK: Simplify CommandSimulator by exposing handle function only duri…
mhsdesign Oct 24, 2024
25d79d3
TASK: Adjust content stream prune test as rebase is a noop without ch…
mhsdesign Oct 25, 2024
29a27de
TASK: Always rebase during publication and do not copy events directly
mhsdesign Oct 25, 2024
e2b5652
DimensionSpacePoint events enriched
kitsunet Oct 25, 2024
0167efd
Just some linter happiness
kitsunet Oct 25, 2024
359a74e
TASK: Consistently use workspace instead of content graph in workspac…
mhsdesign Oct 25, 2024
4ee595e
TASK: Introduce `InitiatingEventMetadata` utility to centralise acces…
mhsdesign Oct 26, 2024
56fd230
TASK: Introduce ExtractedCommand vo to collect original commands AND …
mhsdesign Oct 26, 2024
f7c39ec
TASK: Combine `NodeAggregateEventPublisher` and `ExtractedCommand` to…
mhsdesign Oct 26, 2024
7430bbc
TASK: Add test that rebase keeps the originalCreated intact
mhsdesign Oct 26, 2024
4e26f33
TASK: Move `CommandHandlingDependencies` to CommandHandler namespace
mhsdesign Oct 26, 2024
f0ba01a
TASK: Dont pass `CommandHandlingDependencies` through half of the world
mhsdesign Oct 26, 2024
c62e6db
TASK: Update todos and require base content stream to be not closed d…
mhsdesign Oct 26, 2024
f6742de
TASK: Remove obsolete `ContentGraphReadModelInterface::buildContentGr…
mhsdesign Oct 26, 2024
d2d9faa
TASK: Turn partial publish into full publish if all nodes are about t…
mhsdesign Oct 26, 2024
936f39d
TASK: Update documentation
mhsdesign Oct 26, 2024
3679157
TASK: Revert currently unused error handling strategy from workspace …
mhsdesign Oct 26, 2024
8ad0a3e
FEATURE: Throw `WorkspaceRebaseFailed` during publication or partial …
mhsdesign Oct 26, 2024
49d5870
BUGFIX: Dont discard remaining user changes if $matchingCommands is e…
mhsdesign Oct 26, 2024
34395d8
TASK: Minor cosmetic adjustments and documentation
mhsdesign Oct 27, 2024
5339f97
TASK: Use `setRollbackOnly` to ensure nothing is commited during the …
mhsdesign Oct 28, 2024
f0e088e
TASK: Make publish a no-op if there are no changes instead of a rebase
mhsdesign Oct 28, 2024
5e12a48
TASK: Turn full discard also into a no-op if there are no changes
mhsdesign Oct 28, 2024
38835ca
TASK: Adjust comments to not reference renamed `NodeAggregateEventPub…
mhsdesign Oct 28, 2024
d442544
TASK: Make `CommandsThatFailed::sequenceNumber` internal only for tes…
mhsdesign Oct 28, 2024
08a0d67
Apply suggestions from code review
mhsdesign Oct 28, 2024
3761fc2
TASK: Declare `$handlers` of `CommandBus` private
mhsdesign Oct 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -241,6 +241,19 @@ public function apply(EventInterface $event, EventEnvelope $eventEnvelope): void
}
}

public function inSimulation(\Closure $fn): mixed
{
if ($this->dbal->isTransactionActive()) {
kitsunet marked this conversation as resolved.
Show resolved Hide resolved
throw new \RuntimeException(sprintf('Invoking %s is not allowed to be invoked recursively. Current transaction nesting %d.', __FUNCTION__, $this->dbal->getTransactionNestingLevel()));
}
$this->dbal->beginTransaction();
try {
return $fn();
} finally {
$this->dbal->rollBack();
}
}

private function whenContentStreamWasClosed(ContentStreamWasClosed $event): void
{
$this->updateContentStreamStatus($event->contentStreamId, ContentStreamStatus::CLOSED);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -213,6 +213,19 @@ public function apply(EventInterface $event, EventEnvelope $eventEnvelope): void
};
}

public function inSimulation(\Closure $fn): mixed
{
if ($this->dbal->isTransactionActive()) {
throw new \RuntimeException(sprintf('Invoking %s is not allowed to be invoked recursively. Current transaction nesting %d.', __FUNCTION__, $this->dbal->getTransactionNestingLevel()));
}
$this->dbal->beginTransaction();
try {
return $fn();
} finally {
$this->dbal->rollBack();
}
}

public function getCheckpointStorage(): DbalCheckpointStorage
{
return $this->checkpointStorage;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,44 +29,38 @@ Feature: Discard individual nodes (basics)
| Key | Value |
| workspaceName | "live" |
| newContentStreamId | "cs-identifier" |
And I am in workspace "live"
And I am in workspace "live" and dimension space point {}
And the command CreateRootNodeAggregateWithNode is executed with payload:
| Key | Value |
| nodeAggregateId | "lady-eleonode-rootford" |
| nodeTypeName | "Neos.ContentRepository:Root" |
And the event NodeAggregateWithNodeWasCreated was published with payload:
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| contentStreamId | "cs-identifier" |
| nodeAggregateId | "sir-david-nodenborough" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| originDimensionSpacePoint | {} |
| coveredDimensionSpacePoints | [{}] |
| parentNodeAggregateId | "lady-eleonode-rootford" |
| initialPropertyValues | {"text": {"type": "string", "value": "Initial t1"}} |
| nodeAggregateClassification | "regular" |
And the event NodeAggregateWithNodeWasCreated was published with payload:
| initialPropertyValues | {"text": "Initial t1"} |
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| contentStreamId | "cs-identifier" |
| nodeAggregateId | "nody-mc-nodeface" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| originDimensionSpacePoint | {} |
| coveredDimensionSpacePoints | [{}] |
| parentNodeAggregateId | "sir-david-nodenborough" |
| initialPropertyValues | {"text": {"type": "string", "value": "Initial t2"}} |
| nodeAggregateClassification | "regular" |
And the event NodeAggregateWithNodeWasCreated was published with payload:
| initialPropertyValues | {"text": "Initial t2"} |
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| contentStreamId | "cs-identifier" |
| nodeAggregateId | "sir-nodeward-nodington-iii" |
| nodeTypeName | "Neos.ContentRepository.Testing:Image" |
| originDimensionSpacePoint | {} |
| coveredDimensionSpacePoints | [{}] |
| parentNodeAggregateId | "lady-eleonode-rootford" |
| initialPropertyValues | {"image": {"type": "string", "value": "Initial image"}} |
| nodeAggregateClassification | "regular" |
| initialPropertyValues | {"image": "Initial image"} |
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| nodeAggregateId | "sir-unchanged" |
| nodeTypeName | "Neos.ContentRepository.Testing:Image" |
| parentNodeAggregateId | "lady-eleonode-rootford" |

# Create user workspace
And the command CreateWorkspace is executed with payload:
Expand Down Expand Up @@ -104,7 +98,7 @@ Feature: Discard individual nodes (basics)
| workspaceName | "user-test" |
| nodesToDiscard | [{"workspaceName": "user-test", "dimensionSpacePoint": {}, "nodeAggregateId": "sir-nodeward-nodington-iii"}] |
| newContentStreamId | "user-cs-identifier-new" |

Then I expect the content stream "user-cs-identifier" to not exist

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-identifier-new;sir-david-nodenborough;{}
Expand All @@ -119,35 +113,60 @@ Feature: Discard individual nodes (basics)
And I expect this node to have the following properties:
| Key | Value |
| image | "Initial image" |

Scenario: It is possible to discard no node
Scenario: Discard no node, non existing ones or unchanged nodes is a no-op
# no node
When the command DiscardIndividualNodesFromWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| nodesToDiscard | [] |
| newContentStreamId | "user-cs-identifier-new" |
Then I expect the content stream "user-cs-identifier-new" to not exist

# unchanged or non existing nodes
When the command DiscardIndividualNodesFromWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| nodesToDiscard | [{"dimensionSpacePoint": {}, "nodeAggregateId": "non-existing-node"}, {"dimensionSpacePoint": {}, "nodeAggregateId": "sir-unchanged"}] |
| newContentStreamId | "user-cs-identifier-new-two" |

# all nodes are still on the original user cs
When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-identifier-new;sir-david-nodenborough;{}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-identifier;sir-david-nodenborough;{}
And I expect this node to have the following properties:
| Key | Value |
| text | "Modified t1" |
Then I expect node aggregate identifier "nody-mc-nodeface" to lead to node user-cs-identifier-new;nody-mc-nodeface;{}
Then I expect node aggregate identifier "nody-mc-nodeface" to lead to node user-cs-identifier;nody-mc-nodeface;{}
And I expect this node to have the following properties:
| Key | Value |
| text | "Modified t2" |
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node user-cs-identifier-new;sir-nodeward-nodington-iii;{}
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node user-cs-identifier;sir-nodeward-nodington-iii;{}
And I expect this node to have the following properties:
| Key | Value |
| image | "Modified image" |

# assert that content stream is still open by writing to it:
And the command SetNodeProperties is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| nodeAggregateId | "sir-nodeward-nodington-iii" |
| originDimensionSpacePoint | {} |
| propertyValues | {"image": "Bla bli blub"} |

Scenario: It is possible to discard all nodes
When the command DiscardIndividualNodesFromWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| nodesToDiscard | [{"workspaceName": "user-test", "dimensionSpacePoint": {}, "nodeAggregateId": "sir-david-nodenborough"}, {"workspaceName": "user-test", "dimensionSpacePoint": {}, "nodeAggregateId": "nody-mc-nodeface"}, {"workspaceName": "user-test", "dimensionSpacePoint": {}, "nodeAggregateId": "sir-nodeward-nodington-iii"}] |
| newContentStreamId | "user-cs-identifier-new" |

# when discarding all nodes we expect a full discard via WorkspaceWasDiscarded
Then I expect exactly 2 events to be published on stream with prefix "Workspace:user"
And event at index 1 is of type "WorkspaceWasDiscarded" with payload:
| Key | Expected |
| workspaceName | "user-test" |
| newContentStreamId | "user-cs-identifier-new" |
| previousContentStreamId | "user-cs-identifier" |

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-identifier-new;sir-david-nodenborough;{}
And I expect this node to have the following properties:
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,120 @@
@contentrepository @adapters=DoctrineDBAL
Feature: Rebasing with no conflict

Background:
Given using no content dimensions
And using the following node types:
"""yaml
'Neos.ContentRepository.Testing:Content': {}
'Neos.ContentRepository.Testing:Document':
childNodes:
child1:
type: 'Neos.ContentRepository.Testing:Content'
child2:
type: 'Neos.ContentRepository.Testing:Content'
"""
And using identifier "default", I define a content repository
And I am in content repository "default"
And the command CreateRootWorkspace is executed with payload:
| Key | Value |
| workspaceName | "live" |
| newContentStreamId | "cs-identifier" |
When I am in workspace "live" and dimension space point {}
And the command CreateRootNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| nodeAggregateId | "lady-eleonode-rootford" |
| nodeTypeName | "Neos.ContentRepository:Root" |
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| nodeAggregateId | "sir-david-nodenborough" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| parentNodeAggregateId | "lady-eleonode-rootford" |

# Create user workspace
And the command CreateWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| baseWorkspaceName | "live" |
| newContentStreamId | "user-cs-identifier" |

Then workspaces live,user-test have status UP_TO_DATE

Scenario: Rebase is a no-op if there are no changes
When the command RebaseWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| rebasedContentStreamId | "user-cs-rebased" |
Then I expect the content stream "user-cs-rebased" to not exist

When I am in workspace "live" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node cs-identifier;sir-david-nodenborough;{}

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-identifier;sir-david-nodenborough;{}

# only if the force flag is used we enforce a fork:
When the command RebaseWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| rebasedContentStreamId | "user-cs-rebased" |
| rebaseErrorHandlingStrategy | "force" |
Then I expect the content stream "user-cs-identifier" to not exist

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-david-nodenborough" to lead to node user-cs-rebased;sir-david-nodenborough;{}

Scenario: Rebase only the base contains changes
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| nodeAggregateId | "sir-nodeward-nodington-iii" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| parentNodeAggregateId | "lady-eleonode-rootford" |
Then workspaces user-test has status OUTDATED

When the command RebaseWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| rebasedContentStreamId | "user-cs-rebased" |
Then I expect the content stream "user-cs-identifier" to not exist
Then workspaces live,user-test have status UP_TO_DATE

When I am in workspace "live" and dimension space point {}
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node cs-identifier;sir-nodeward-nodington-iii;{}

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node user-cs-rebased;sir-nodeward-nodington-iii;{}


Scenario: Rebase workspace and base contains changes
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "live" |
| nodeAggregateId | "sir-nodeward-nodington-iii" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| parentNodeAggregateId | "lady-eleonode-rootford" |
And the command CreateNodeAggregateWithNode is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| nodeAggregateId | "nordisch-nodel" |
| nodeTypeName | "Neos.ContentRepository.Testing:Content" |
| parentNodeAggregateId | "lady-eleonode-rootford" |
Then workspaces user-test has status OUTDATED

When the command RebaseWorkspace is executed with payload:
| Key | Value |
| workspaceName | "user-test" |
| rebasedContentStreamId | "user-cs-rebased" |
Then I expect the content stream "user-cs-identifier" to not exist

Then workspaces live,user-test have status UP_TO_DATE

When I am in workspace "live" and dimension space point {}
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node cs-identifier;sir-nodeward-nodington-iii;{}
Then I expect node aggregate identifier "nordisch-nodel" to lead to no node

When I am in workspace "user-test" and dimension space point {}
Then I expect node aggregate identifier "sir-nodeward-nodington-iii" to lead to node user-cs-rebased;sir-nodeward-nodington-iii;{}
Then I expect node aggregate identifier "nordisch-nodel" to lead to node user-cs-rebased;nordisch-nodel;{}
Loading
Loading