From 62b7cc8f990c77f8430923a76b707d015244b479 Mon Sep 17 00:00:00 2001 From: Robin Munn Date: Mon, 25 Nov 2024 13:23:20 +0700 Subject: [PATCH 1/4] Add Update(before, after) API for senses --- .../FwDataMiniLcmBridge/Api/FwDataMiniLcmApi.cs | 17 +++++++++++++++++ .../FwLiteProjectSync/DryRunMiniLcmApi.cs | 12 ++++++++++++ backend/FwLite/LcmCrdt/CrdtMiniLcmApi.cs | 16 ++++++++++++++++ backend/FwLite/MiniLcm/IMiniLcmReadApi.cs | 1 + backend/FwLite/MiniLcm/IMiniLcmWriteApi.cs | 1 + backend/LfClassicData/LfClassicMiniLcmApi.cs | 9 +++++++++ 6 files changed, 56 insertions(+) diff --git a/backend/FwLite/FwDataMiniLcmBridge/Api/FwDataMiniLcmApi.cs b/backend/FwLite/FwDataMiniLcmBridge/Api/FwDataMiniLcmApi.cs index a60f206e2..c5ffc2974 100644 --- a/backend/FwLite/FwDataMiniLcmBridge/Api/FwDataMiniLcmApi.cs +++ b/backend/FwLite/FwDataMiniLcmBridge/Api/FwDataMiniLcmApi.cs @@ -911,6 +911,12 @@ private void ApplySenseToLexSense(Sense sense, ILexSense lexSense) } } + public Task GetSense(Guid entryId, Guid id) + { + var lcmSense = SenseRepository.GetObject(id); + return Task.FromResult(lcmSense is null ? null : FromLexSense(lcmSense)); + } + public Task CreateSense(Guid entryId, Sense sense) { if (sense.Id == default) sense.Id = Guid.NewGuid(); @@ -938,6 +944,17 @@ public Task UpdateSense(Guid entryId, Guid senseId, UpdateObjectInput UpdateSense(Guid entryId, Sense before, Sense after) + { + await Cache.DoUsingNewOrCurrentUOW("Update Sense", + "Revert Sense", + async () => + { + await SenseSync.Sync(entryId, after, before, this); + }); + return await GetSense(entryId, after.Id) ?? throw new NullReferenceException("unable to find sense with id " + after.Id); + } + public Task AddSemanticDomainToSense(Guid senseId, SemanticDomain semanticDomain) { UndoableUnitOfWorkHelper.DoUsingNewOrCurrentUOW("Add Semantic Domain to Sense", diff --git a/backend/FwLite/FwLiteProjectSync/DryRunMiniLcmApi.cs b/backend/FwLite/FwLiteProjectSync/DryRunMiniLcmApi.cs index ff925b4c5..82d5190ca 100644 --- a/backend/FwLite/FwLiteProjectSync/DryRunMiniLcmApi.cs +++ b/backend/FwLite/FwLiteProjectSync/DryRunMiniLcmApi.cs @@ -190,6 +190,11 @@ public async Task RemoveComplexFormType(Guid entryId, Guid complexFormTypeId) await Task.CompletedTask; } + public Task GetSense(Guid entryId, Guid id) + { + return api.GetSense(entryId, id); + } + public Task CreateSense(Guid entryId, Sense sense) { DryRunRecords.Add(new DryRunRecord(nameof(CreateSense), $"Create sense {sense.Gloss}")); @@ -206,6 +211,13 @@ public async Task UpdateSense(Guid entryId, Guid senseId, UpdateObjectInp return sense; } + public async Task UpdateSense(Guid entryId, Sense before, Sense after) + { + DryRunRecords.Add(new DryRunRecord(nameof(UpdateSense), + $"Update sense {after.Id}")); + return await GetSense(entryId, after.Id) ?? throw new NullReferenceException($"unable to find sense with id {after.Id}"); + } + public Task DeleteSense(Guid entryId, Guid senseId) { DryRunRecords.Add(new DryRunRecord(nameof(DeleteSense), $"Delete sense {senseId}")); diff --git a/backend/FwLite/LcmCrdt/CrdtMiniLcmApi.cs b/backend/FwLite/LcmCrdt/CrdtMiniLcmApi.cs index 387dd8a0e..3955ab957 100644 --- a/backend/FwLite/LcmCrdt/CrdtMiniLcmApi.cs +++ b/backend/FwLite/LcmCrdt/CrdtMiniLcmApi.cs @@ -464,6 +464,16 @@ private async IAsyncEnumerable CreateSenseChanges(Guid entryId, Sense s } } + public async Task GetSense(Guid entryId, Guid id) + { + var entry = await Entries.AsTracking(false) + .LoadWith(e => e.Senses) + .ThenLoad(s => s.ExampleSentences) + .AsQueryable() + .SingleOrDefaultAsync(e => e.Id == entryId); + return entry?.Senses.FirstOrDefault(s => s.Id == id); + } + public async Task CreateSense(Guid entryId, Sense sense) { await dataModel.AddChanges(ClientId, await CreateSenseChanges(entryId, sense).ToArrayAsync()); @@ -480,6 +490,12 @@ public async Task UpdateSense(Guid entryId, return await dataModel.GetLatest(senseId) ?? throw new NullReferenceException(); } + public async Task UpdateSense(Guid entryId, Sense before, Sense after) + { + await SenseSync.Sync(entryId, after, before, this); + return await GetSense(entryId, after.Id) ?? throw new NullReferenceException("unable to find sense with id " + after.Id); + } + public async Task DeleteSense(Guid entryId, Guid senseId) { await dataModel.AddChange(ClientId, new DeleteChange(senseId)); diff --git a/backend/FwLite/MiniLcm/IMiniLcmReadApi.cs b/backend/FwLite/MiniLcm/IMiniLcmReadApi.cs index 03d1aa440..ff8a5cfba 100644 --- a/backend/FwLite/MiniLcm/IMiniLcmReadApi.cs +++ b/backend/FwLite/MiniLcm/IMiniLcmReadApi.cs @@ -13,6 +13,7 @@ public interface IMiniLcmReadApi IAsyncEnumerable GetEntries(QueryOptions? options = null); IAsyncEnumerable SearchEntries(string query, QueryOptions? options = null); Task GetEntry(Guid id); + Task GetSense(Guid entryId, Guid id); Task GetPartOfSpeech(Guid id); Task GetSemanticDomain(Guid id); Task GetExampleSentence(Guid entryId, Guid senseId, Guid id); diff --git a/backend/FwLite/MiniLcm/IMiniLcmWriteApi.cs b/backend/FwLite/MiniLcm/IMiniLcmWriteApi.cs index 1c5301fb8..b41040cb1 100644 --- a/backend/FwLite/MiniLcm/IMiniLcmWriteApi.cs +++ b/backend/FwLite/MiniLcm/IMiniLcmWriteApi.cs @@ -48,6 +48,7 @@ Task UpdateWritingSystem(WritingSystemId id, #region Sense Task CreateSense(Guid entryId, Sense sense); Task UpdateSense(Guid entryId, Guid senseId, UpdateObjectInput update); + Task UpdateSense(Guid entryId, Sense before, Sense after); Task DeleteSense(Guid entryId, Guid senseId); Task AddSemanticDomainToSense(Guid senseId, SemanticDomain semanticDomain); Task RemoveSemanticDomainFromSense(Guid senseId, Guid semanticDomainId); diff --git a/backend/LfClassicData/LfClassicMiniLcmApi.cs b/backend/LfClassicData/LfClassicMiniLcmApi.cs index c8c25c8fa..b2377bce6 100644 --- a/backend/LfClassicData/LfClassicMiniLcmApi.cs +++ b/backend/LfClassicData/LfClassicMiniLcmApi.cs @@ -315,6 +315,15 @@ private static SemanticDomain ToSemanticDomain(Entities.OptionListItem item) return ToEntry(entry); } + public async Task GetSense(Guid entryId, Guid id) + { + var entry = await Entries.Find(e => e.Guid == entryId).FirstOrDefaultAsync(); + if (entry is null) return null; + var sense = entry.Senses?.FirstOrDefault(s => s?.Guid == id); + if (sense is null) return null; + return ToSense(entryId, sense); + } + public async Task GetExampleSentence(Guid entryId, Guid senseId, Guid id) { var entry = await Entries.Find(e => e.Guid == entryId).FirstOrDefaultAsync(); From db8aff0b0df308513e2b6ca39195a22bc953d6e5 Mon Sep 17 00:00:00 2001 From: Robin Munn Date: Fri, 29 Nov 2024 13:33:47 +0700 Subject: [PATCH 2/4] Address review comments --- .../FwLite/MiniLcm/SyncHelpers/SenseSync.cs | 37 +++++++++---------- 1 file changed, 18 insertions(+), 19 deletions(-) diff --git a/backend/FwLite/MiniLcm/SyncHelpers/SenseSync.cs b/backend/FwLite/MiniLcm/SyncHelpers/SenseSync.cs index 1b3b7c437..181107105 100644 --- a/backend/FwLite/MiniLcm/SyncHelpers/SenseSync.cs +++ b/backend/FwLite/MiniLcm/SyncHelpers/SenseSync.cs @@ -17,6 +17,24 @@ public static async Task Sync(Guid entryId, afterSense.ExampleSentences, beforeSense.ExampleSentences, api); + changes += await DiffCollection.Diff(api, + beforeSense.SemanticDomains, + afterSense.SemanticDomains, + async (api, domain) => + { + await api.AddSemanticDomainToSense(beforeSense.Id, domain); + return 1; + }, + async (api, beforeDomain) => + { + await api.RemoveSemanticDomainFromSense(beforeSense.Id, beforeDomain.Id); + return 1; + }, + (_, beforeDomain, afterDomain) => + { + //do nothing, semantic domains are not editable here + return Task.FromResult(0); + }); return changes + (updateObjectInput is null ? 0 : 1); } @@ -38,25 +56,6 @@ public static async Task Sync(Guid entryId, patchDocument.Replace(sense => sense.PartOfSpeechId, afterSense.PartOfSpeechId); } - await DiffCollection.Diff(null!, - beforeSense.SemanticDomains, - afterSense.SemanticDomains, - (_, domain) => - { - patchDocument.Add(sense => sense.SemanticDomains, domain); - return Task.FromResult(1); - }, - (_, beforeDomain) => - { - patchDocument.Remove(sense => sense.SemanticDomains, - beforeSense.SemanticDomains.IndexOf(beforeDomain)); - return Task.FromResult(1); - }, - (_, beforeDomain, afterDomain) => - { - //do nothing, semantic domains are not editable here - return Task.FromResult(0); - }); if (patchDocument.Operations.Count == 0) return null; return new UpdateObjectInput(patchDocument); } From fc78b2344c88f42547f0e4606070049962e2e3b8 Mon Sep 17 00:00:00 2001 From: Robin Munn Date: Wed, 4 Dec 2024 12:35:33 +0700 Subject: [PATCH 3/4] Add new interface implementation needed after rebase Rebasing into develop pulled in a change that LfClassicMiniLcmApi didn't yet implement. Since LF Classic doesn't handle complex forms, the implementation is dead simple: just return null. --- backend/LfClassicData/LfClassicMiniLcmApi.cs | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/backend/LfClassicData/LfClassicMiniLcmApi.cs b/backend/LfClassicData/LfClassicMiniLcmApi.cs index b2377bce6..f4aef4e22 100644 --- a/backend/LfClassicData/LfClassicMiniLcmApi.cs +++ b/backend/LfClassicData/LfClassicMiniLcmApi.cs @@ -20,6 +20,11 @@ public IAsyncEnumerable GetComplexFormTypes() return AsyncEnumerable.Empty(); } + public Task GetComplexFormType(Guid id) + { + return Task.FromResult(null); + } + public async Task GetWritingSystems() { var inputSystems = await systemDbContext.Projects.AsQueryable() From 10e672a57151d2f3505c6f25050743fff17874c5 Mon Sep 17 00:00:00 2001 From: Robin Munn Date: Wed, 4 Dec 2024 12:38:04 +0700 Subject: [PATCH 4/4] Update test expectations so it won't fail wrongly --- backend/FwLite/FwLiteProjectSync.Tests/UpdateDiffTests.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/backend/FwLite/FwLiteProjectSync.Tests/UpdateDiffTests.cs b/backend/FwLite/FwLiteProjectSync.Tests/UpdateDiffTests.cs index 2d1eb2058..37d4b7026 100644 --- a/backend/FwLite/FwLiteProjectSync.Tests/UpdateDiffTests.cs +++ b/backend/FwLite/FwLiteProjectSync.Tests/UpdateDiffTests.cs @@ -37,7 +37,7 @@ public async Task SenseDiffShouldUpdateAllFields() var senseDiffToUpdate = await SenseSync.SenseDiffToUpdate(before, after); ArgumentNullException.ThrowIfNull(senseDiffToUpdate); senseDiffToUpdate.Apply(before); - before.Should().BeEquivalentTo(after, options => options.Excluding(x => x.Id).Excluding(x => x.EntryId).Excluding(x => x.DeletedAt).Excluding(x => x.ExampleSentences)); + before.Should().BeEquivalentTo(after, options => options.Excluding(x => x.Id).Excluding(x => x.EntryId).Excluding(x => x.DeletedAt).Excluding(x => x.ExampleSentences).Excluding(x => x.SemanticDomains)); } [Fact]