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: References on creation and Copy #5148

Merged
merged 34 commits into from
Nov 1, 2024
Merged
Changes from 1 commit
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
1612659
FEATURE: References on creation and Copy
kitsunet Jun 17, 2024
c20c17b
Fix merge errors and syntax problems
kitsunet Aug 26, 2024
c50fcb7
More syntax fixes
kitsunet Aug 26, 2024
8c40cbb
Remove forgotten debug statement
kitsunet Aug 26, 2024
b600c70
Fix new unit Tests
kitsunet Aug 26, 2024
72d5f61
Fix another new unit test
kitsunet Aug 26, 2024
7519353
Adjust new behat test with References
kitsunet Aug 26, 2024
1bc6e38
Fix broken export test
kitsunet Aug 26, 2024
a9ae648
Merge branch '9.0' into feature/node-reference-copy
kitsunet Sep 3, 2024
a4cc38c
Merge remote-tracking branch 'origin/9.0' into feature/node-reference…
kitsunet Oct 21, 2024
80b719e
Cleaner References write DTOs
kitsunet Oct 22, 2024
be16480
Empty properties in refs without special treatment
kitsunet Oct 22, 2024
5505b5e
Merge branch '9.0' into feature/node-reference-copy
kitsunet Oct 22, 2024
a74c9b3
Cleanup
kitsunet Oct 22, 2024
98dead5
Add event migration for references
kitsunet Oct 23, 2024
8ccb9f0
Merge remote-tracking branch 'origin/9.0' into feature/node-reference…
kitsunet Oct 24, 2024
da531d1
Merge remote-tracking branch 'origin/9.0' into feature/node-reference…
kitsunet Oct 28, 2024
1ffad19
Code cleanup
kitsunet Oct 30, 2024
f0bb466
Code cleanup
kitsunet Oct 30, 2024
27e9550
TASK: Fix migrateReferencesToMultiFormat
mhsdesign Oct 30, 2024
a6ff4a1
TASK: Simplify code by using automatic `jsonSerialize` like in other …
mhsdesign Oct 30, 2024
0c18918
TASK: Use variadics to verify array list shape
mhsdesign Oct 30, 2024
9694c97
TASK: Introduce `fromNameAndTargets` utility and accept flat list in …
mhsdesign Oct 30, 2024
826b47d
TASK: Remove obsolet serialisation logic from vo's that are not persi…
mhsdesign Oct 30, 2024
a28c27d
TASK: Introduce `NodeReferencesToWrite::withReference` and make merge…
mhsdesign Oct 30, 2024
48efbea
TASK: Validate `NodeReferencesForName` to not contain duplicate node …
mhsdesign Oct 31, 2024
a8cfb25
TASK: Simplify `SerializedNodeReferences` capabilities
mhsdesign Oct 31, 2024
d254397
TASK: inline `fromReadReferences` so its @internal
mhsdesign Oct 31, 2024
b800c66
Merge remote-tracking branch 'origin/9.0' into feature/node-reference…
mhsdesign Oct 31, 2024
7cadf90
TASK: Adjust documentation and namings
mhsdesign Oct 31, 2024
3310d74
TASK: Adjust namings to remove obsolete `fromNameAnd` part
mhsdesign Oct 31, 2024
4eee438
TASK: Ensure that `SerializedNodeReferencesForName` are serialized as…
mhsdesign Oct 31, 2024
bc2427c
TASK: Make atomic delete instead of deleting old references one by one
mhsdesign Oct 31, 2024
4041870
TASK: Optimise reference writing by splitting by scope
mhsdesign Oct 31, 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
@@ -97,4 +97,11 @@ public function migratePayloadToValidWorkspaceNamesCommand(string $contentReposi
$eventMigrationService = $this->contentRepositoryRegistry->buildService($contentRepositoryId, $this->eventMigrationServiceFactory);
$eventMigrationService->migratePayloadToValidWorkspaceNames($this->outputLine(...));
}

public function migrateSetReferencesToMultiNameFormatCommand(string $contentRepository = 'default'): void
{
$contentRepositoryId = ContentRepositoryId::fromString($contentRepository);
$eventMigrationService = $this->contentRepositoryRegistry->buildService($contentRepositoryId, $this->eventMigrationServiceFactory);
$eventMigrationService->migrateReferencesToMultiFormat($this->outputLine(...));
}
}
kitsunet marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -240,6 +240,107 @@ public function migratePropertiesToUnset(\Closure $outputFn): void
}
}

public function migrateReferencesToMultiFormat(\Closure $outputFn): void
{
$this->eventsModified = [];
$warnings = 0;

$backupEventTableName = DoctrineEventStoreFactory::databaseTableName($this->contentRepositoryId)
. '_bkp_' . date('Y_m_d_H_i_s');
$outputFn(sprintf('Backup: copying events table to %s', $backupEventTableName));

$this->copyEventTable($backupEventTableName);

$streamName = VirtualStreamName::all();
$eventStream = $this->eventStore->load($streamName);
mhsdesign marked this conversation as resolved.
Show resolved Hide resolved
foreach ($eventStream as $eventEnvelope) {
$outputRewriteNotice = fn(string $message) => $outputFn(sprintf('%s@%s %s', $eventEnvelope->sequenceNumber->value, $eventEnvelope->event->type->value, $message));
if ($eventEnvelope->event->type->value !== 'NodeReferencesWereSet') {
continue;
}

// migrate payload
$eventData = self::decodeEventPayload($eventEnvelope);
if (!isset($eventData['referenceName'])) {
// this event must have the new format already
continue;
}

$referenceNameString = $eventData['referenceName'];
// double array is not a mistake
$newReferencesPayload = [[
'referenceName' => $referenceNameString,
'references' => []
]];

unset($eventData['referenceName']);

// technically the false case should never happen, but we update it anyway
if (is_array($eventData['references'])) {
foreach ($eventData['references'] as $reference) {
$reference['target'] = $reference['targetNodeAggregateId'];
unset($reference['targetNodeAggregateId']);
if ($reference['properties'] === null) {
unset($reference['properties']);
}
$newReferencesPayload[0]['references'][] = $reference;
}
}

$eventData['references'] = $newReferencesPayload;
$this->updateEventPayload($eventEnvelope->sequenceNumber, $eventData);
$outputRewriteNotice('Payload: references migrated');

// optionally also migrate metadata
$eventMetaData = $eventEnvelope->event->metadata?->value;
if (!isset($eventMetaData['commandClass'])) {
continue;
}

if ($eventMetaData['commandClass'] !== SetSerializedNodeReferences::class) {
$warnings++;
$outputFn(sprintf('WARNING: Cannot migrate event metadata of %s as commandClass %s was not expected.', $eventEnvelope->event->type->value, $eventMetaData['commandClass']));
continue;
}

$referenceNameString = $eventMetaData['commandPayload']['referenceName'];
$newReferencesPayload = [
[
'referenceName' => $referenceNameString,
'references' => []
]
];

unset($eventMetaData['commandPayload']['referenceName']);

// technically the false case should never happen, but we update it anyway
if (is_array($eventMetaData['commandPayload']['references'])) {
foreach ($eventMetaData['commandPayload']['references'] as $reference) {
$reference['target'] = $reference['targetNodeAggregateId'];
unset($reference['targetNodeAggregateId']);
if ($reference['properties'] === null) {
unset($reference['properties']);
}
$newReferencesPayload[0]['references'][] = $reference;
}
}

$eventMetaData['commandPayload']['references'] = $newReferencesPayload;
$outputRewriteNotice('Metadata: references migrated');
$this->updateEventMetaData($eventEnvelope->sequenceNumber, $eventMetaData);
}

if (!count($this->eventsModified)) {
$outputFn('Migration was not necessary.');
return;
}

$outputFn();
$outputFn(sprintf('Migration applied to %s events.', count($this->eventsModified)));
if ($warnings) {
$outputFn(sprintf('WARNING: Finished but %d warnings emitted.', $warnings));
}
}

/**
* Adds a dummy workspace name to the events meta-data, so it can be rebased