From 198263a132b9f3f7a57dec633e15171eb5ae4a60 Mon Sep 17 00:00:00 2001 From: Kevin Hahn Date: Tue, 15 Oct 2024 15:07:45 +0700 Subject: [PATCH 1/5] create SetDefinitionPartOfSpeechChange, make a FK constraint between words and definitions --- .../Changes/SetDefinitionPartOfSpeechChange.cs | 16 ++++++++++++++++ src/SIL.Harmony.Sample/CrdtSampleKernel.cs | 9 ++++++++- .../DbContextTests.VerifyModel.verified.txt | 4 +++- 3 files changed, 27 insertions(+), 2 deletions(-) create mode 100644 src/SIL.Harmony.Sample/Changes/SetDefinitionPartOfSpeechChange.cs diff --git a/src/SIL.Harmony.Sample/Changes/SetDefinitionPartOfSpeechChange.cs b/src/SIL.Harmony.Sample/Changes/SetDefinitionPartOfSpeechChange.cs new file mode 100644 index 0000000..6a60490 --- /dev/null +++ b/src/SIL.Harmony.Sample/Changes/SetDefinitionPartOfSpeechChange.cs @@ -0,0 +1,16 @@ +using SIL.Harmony.Changes; +using SIL.Harmony.Entities; +using SIL.Harmony.Sample.Models; + +namespace SIL.Harmony.Sample.Changes; + +public class SetDefinitionPartOfSpeechChange(Guid entityId, string partOfSpeech) : EditChange(entityId), ISelfNamedType +{ + public string PartOfSpeech { get; } = partOfSpeech; + + public override ValueTask ApplyChange(Definition entity, ChangeContext context) + { + entity.PartOfSpeech = PartOfSpeech; + return ValueTask.CompletedTask; + } +} \ No newline at end of file diff --git a/src/SIL.Harmony.Sample/CrdtSampleKernel.cs b/src/SIL.Harmony.Sample/CrdtSampleKernel.cs index 7f4c3b6..a48720b 100644 --- a/src/SIL.Harmony.Sample/CrdtSampleKernel.cs +++ b/src/SIL.Harmony.Sample/CrdtSampleKernel.cs @@ -45,13 +45,20 @@ public static IServiceCollection AddCrdtDataSample(this IServiceCollection servi .Add() .Add() .Add>() + .Add() .Add>() .Add>() .Add>() ; config.ObjectTypeListBuilder.DefaultAdapter() .Add() - .Add() + .Add(builder => + { + builder.HasOne() + .WithMany() + .HasForeignKey(d => d.WordId) + .OnDelete(DeleteBehavior.Cascade); + }) .Add(); }); return services; diff --git a/src/SIL.Harmony.Tests/DbContextTests.VerifyModel.verified.txt b/src/SIL.Harmony.Tests/DbContextTests.VerifyModel.verified.txt index de7e1f4..32ce952 100644 --- a/src/SIL.Harmony.Tests/DbContextTests.VerifyModel.verified.txt +++ b/src/SIL.Harmony.Tests/DbContextTests.VerifyModel.verified.txt @@ -90,13 +90,15 @@ PartOfSpeech (string) Required SnapshotId (no field, Guid?) Shadow FK Index Text (string) Required - WordId (Guid) Required + WordId (Guid) Required FK Index Keys: Id PK Foreign keys: Definition {'SnapshotId'} -> ObjectSnapshot {'Id'} Unique SetNull + Definition {'WordId'} -> Word {'Id'} Required Cascade Indexes: SnapshotId Unique + WordId Annotations: DiscriminatorProperty: Relational:FunctionName: From 3844dfcdc562c8d6c6739423910542faf18e2b90 Mon Sep 17 00:00:00 2001 From: Kevin Hahn Date: Tue, 15 Oct 2024 15:08:06 +0700 Subject: [PATCH 2/5] write a failing test due to how saving changes are ordered --- src/SIL.Harmony.Tests/SyncTests.cs | 22 +++++++++++++++++++--- 1 file changed, 19 insertions(+), 3 deletions(-) diff --git a/src/SIL.Harmony.Tests/SyncTests.cs b/src/SIL.Harmony.Tests/SyncTests.cs index 4e3f0c0..9d1fef6 100644 --- a/src/SIL.Harmony.Tests/SyncTests.cs +++ b/src/SIL.Harmony.Tests/SyncTests.cs @@ -1,4 +1,5 @@ -using SIL.Harmony.Sample.Models; +using SIL.Harmony.Sample.Changes; +using SIL.Harmony.Sample.Models; namespace SIL.Harmony.Tests; @@ -99,8 +100,23 @@ public async Task SyncMultipleClientChanges(int clientCount) var clientSnapshot = await client.DataModel.GetProjectSnapshot(); var simpleSnapshot = clientSnapshot.Snapshots.Should().ContainKey(entitySnapshot.EntityId).WhoseValue; var entity = await client.DataModel.GetBySnapshotId(simpleSnapshot.Id); - entity.Should().BeEquivalentTo(serverEntity); + entity.Should().BeEquivalentTo(serverEntity); } } } -} + + [Fact] + public async Task CanSync_AddDependentWithMultipleChanges() + { + var entity1Id = Guid.NewGuid(); + var definitionId = Guid.NewGuid(); + await _client1.WriteNextChange(_client1.SetWord(entity1Id, "entity1")); + await _client1.WriteNextChange(_client1.NewDefinition(entity1Id, "def1", "pos1", definitionId: definitionId)); + await _client1.WriteNextChange(new SetDefinitionPartOfSpeechChange(definitionId, "pos2")); + + await _client2.DataModel.SyncWith(_client1.DataModel); + + _client2.DataModel.GetLatestObjects().Should() + .BeEquivalentTo(_client1.DataModel.GetLatestObjects()); + } +} \ No newline at end of file From 0735fc4dd3d30f00ae4c530ce2de9c6d63830ec4 Mon Sep 17 00:00:00 2001 From: Tim Haasdyk Date: Tue, 15 Oct 2024 15:23:14 +0200 Subject: [PATCH 3/5] Fix adding intermediate snapshots that reference new entities --- src/SIL.Harmony/SnapshotWorker.cs | 24 ++++++++++++++++++++---- 1 file changed, 20 insertions(+), 4 deletions(-) diff --git a/src/SIL.Harmony/SnapshotWorker.cs b/src/SIL.Harmony/SnapshotWorker.cs index 67f5e91..10f8e69 100644 --- a/src/SIL.Harmony/SnapshotWorker.cs +++ b/src/SIL.Harmony/SnapshotWorker.cs @@ -16,6 +16,7 @@ internal class SnapshotWorker private readonly CrdtRepository _crdtRepository; private readonly CrdtConfig _crdtConfig; private readonly Dictionary _pendingSnapshots = []; + private readonly Dictionary _rootSnapshots = []; private readonly List _newIntermediateSnapshots = []; private SnapshotWorker(Dictionary snapshots, @@ -56,8 +57,11 @@ public async Task UpdateSnapshots(Commit oldestAddedCommit, Commit[] newCommits) var commits = await _crdtRepository.GetCommitsAfter(previousCommit); await ApplyCommitChanges(commits.UnionBy(newCommits, c => c.Id), true, previousCommit?.Hash ?? CommitBase.NullParentHash); - //intermediate snapshots should be added first, as the last snapshot added for an entity will be used in the projected tables + // First add any new entities/snapshots as they might be referenced by intermediate snapshots + await _crdtRepository.AddSnapshots(_rootSnapshots.Values); + // Then add any intermediate snapshots we're hanging onto for performance benefits await _crdtRepository.AddIfNew(_newIntermediateSnapshots); + // And finally the up-to-date snapshots, which will be used in the projected tables await _crdtRepository.AddSnapshots(_pendingSnapshots.Values); } @@ -117,7 +121,7 @@ private async ValueTask ApplyCommitChanges(IEnumerable commits, bool upd { //do nothing, will cause prevSnapshot to be overriden in _pendingSnapshots if it exists } - else if (commitIndex % 2 == 0 || prevSnapshot.IsRoot) + else if (commitIndex % 2 == 0 && !prevSnapshot.IsRoot) { intermediateSnapshots[prevSnapshot.Entity.Id] = prevSnapshot; } @@ -179,6 +183,11 @@ private async ValueTask MarkDeleted(Guid deletedEntityId, Commit commit) return snapshot; } + if (_rootSnapshots.TryGetValue(entityId, out var rootSnapshot)) + { + return rootSnapshot; + } + if (_snapshotLookup.TryGetValue(entityId, out var snapshotId)) { if (snapshotId is null) return null; @@ -193,7 +202,14 @@ private async ValueTask MarkDeleted(Guid deletedEntityId, Commit commit) private void AddSnapshot(ObjectSnapshot snapshot) { - //if there was already a pending snapshot there's no need to store it as both may point to the same commit - _pendingSnapshots[snapshot.Entity.Id] = snapshot; + if (snapshot.IsRoot) + { + _rootSnapshots[snapshot.Entity.Id] = snapshot; + } + else + { + //if there was already a pending snapshot there's no need to store it as both may point to the same commit + _pendingSnapshots[snapshot.Entity.Id] = snapshot; + } } } From 4f5b04e7b5232b521508a052c56a7589e383d9f8 Mon Sep 17 00:00:00 2001 From: Tim Haasdyk Date: Tue, 15 Oct 2024 15:24:16 +0200 Subject: [PATCH 4/5] Fix performance benchmark checking for poor comma-decimal cultures --- src/SIL.Harmony.Tests/DataModelPerformanceTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/SIL.Harmony.Tests/DataModelPerformanceTests.cs b/src/SIL.Harmony.Tests/DataModelPerformanceTests.cs index e65d836..d009691 100644 --- a/src/SIL.Harmony.Tests/DataModelPerformanceTests.cs +++ b/src/SIL.Harmony.Tests/DataModelPerformanceTests.cs @@ -31,7 +31,7 @@ public void AddingChangePerformance() ); foreach (var benchmarkCase in summary.BenchmarksCases.Where(b => !summary.IsBaseline(b))) { - var ratio = double.Parse(BaselineRatioColumn.RatioMean.GetValue(summary, benchmarkCase)); + var ratio = double.Parse(BaselineRatioColumn.RatioMean.GetValue(summary, benchmarkCase), System.Globalization.CultureInfo.InvariantCulture); //for now it just makes sure that no case is worse that 7x, this is based on the 10_000 test being 5 times worse. //it would be better to have this scale off the number of changes ratio.Should().BeInRange(0, 7, "performance should not get worse, benchmark " + benchmarkCase.DisplayInfo); From 1a188e87eeaa28c16bf755103856994fdf876813 Mon Sep 17 00:00:00 2001 From: Tim Haasdyk Date: Tue, 15 Oct 2024 15:24:22 +0200 Subject: [PATCH 5/5] Fix formatting --- src/SIL.Harmony.Tests/SyncTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/SIL.Harmony.Tests/SyncTests.cs b/src/SIL.Harmony.Tests/SyncTests.cs index 9d1fef6..214781d 100644 --- a/src/SIL.Harmony.Tests/SyncTests.cs +++ b/src/SIL.Harmony.Tests/SyncTests.cs @@ -100,7 +100,7 @@ public async Task SyncMultipleClientChanges(int clientCount) var clientSnapshot = await client.DataModel.GetProjectSnapshot(); var simpleSnapshot = clientSnapshot.Snapshots.Should().ContainKey(entitySnapshot.EntityId).WhoseValue; var entity = await client.DataModel.GetBySnapshotId(simpleSnapshot.Id); - entity.Should().BeEquivalentTo(serverEntity); + entity.Should().BeEquivalentTo(serverEntity); } } }