From d6a806360058d43b85fb870d911f319ef524ff73 Mon Sep 17 00:00:00 2001 From: Aaron Stannard Date: Thu, 8 Aug 2024 15:23:28 -0500 Subject: [PATCH] Akka.Persistence: Made `DateTime.UtcNow` the default timestamp for `SnapshotMetdata` (#7313) * Made `DateTime.UtcNow` the default timestamp for `SnapshotMetdata` * fixed all `SnapshotMetadata` calls * added API approvals * standardized on `Sys.Scheduler.Now.DateTime` * Fix SQL query error in QueryExecutor * Update API Approval list --------- Co-authored-by: Gregorius Soedharmo --- .../Snapshot/QueryExecutor.cs | 2 +- ...pec.ApprovePersistence.DotNet.verified.txt | 3 +- ...PISpec.ApprovePersistence.Net.verified.txt | 3 +- .../LocalSnapshotStoreSpec.cs | 2 +- .../Query/PersistenceIdsSpec.cs | 2 +- .../SnapshotStoreSerializationSpec.cs | 10 ++-- .../Snapshot/SnapshotStoreSpec.cs | 59 +++++++++++++++++-- src/core/Akka.Persistence/Eventsourced.cs | 4 +- src/core/Akka.Persistence/SnapshotProtocol.cs | 22 +++---- .../Akka/Actor/Scheduler/SchedulerBase.cs | 2 +- 10 files changed, 82 insertions(+), 27 deletions(-) diff --git a/src/contrib/persistence/Akka.Persistence.Sql.Common/Snapshot/QueryExecutor.cs b/src/contrib/persistence/Akka.Persistence.Sql.Common/Snapshot/QueryExecutor.cs index bafd503d06c..bb5a5d7ca2a 100644 --- a/src/contrib/persistence/Akka.Persistence.Sql.Common/Snapshot/QueryExecutor.cs +++ b/src/contrib/persistence/Akka.Persistence.Sql.Common/Snapshot/QueryExecutor.cs @@ -463,7 +463,7 @@ public virtual async Task DeleteAsync( await connection.ExecuteInTransaction(WriteIsolationLevel, cancellationToken, async (tx, token) => { var sql = timestamp.HasValue - ? DeleteSnapshotRangeSql + " AND { Configuration.TimestampColumnName} = @Timestamp" + ? DeleteSnapshotSql + $" AND {Configuration.TimestampColumnName} <= @Timestamp" : DeleteSnapshotSql; using var command = GetCommand(connection, sql); diff --git a/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.DotNet.verified.txt b/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.DotNet.verified.txt index e231291dc65..a19ffd58af1 100644 --- a/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.DotNet.verified.txt +++ b/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.DotNet.verified.txt @@ -587,7 +587,8 @@ namespace Akka.Persistence } public sealed class SnapshotMetadata : System.IEquatable { - public static System.DateTime TimestampNotSpecified; + [System.ObsoleteAttribute("This constructor is deprecated and will be removed in v1.6. Use the constructor w" + + "ith the timestamp parameter instead. Since v1.5.28", true)] public SnapshotMetadata(string persistenceId, long sequenceNr) { } [Newtonsoft.Json.JsonConstructorAttribute()] public SnapshotMetadata(string persistenceId, long sequenceNr, System.DateTime timestamp) { } diff --git a/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.Net.verified.txt b/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.Net.verified.txt index 4708abc85da..f48cd96cfa4 100644 --- a/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.Net.verified.txt +++ b/src/core/Akka.API.Tests/verify/CoreAPISpec.ApprovePersistence.Net.verified.txt @@ -587,7 +587,8 @@ namespace Akka.Persistence } public sealed class SnapshotMetadata : System.IEquatable { - public static System.DateTime TimestampNotSpecified; + [System.ObsoleteAttribute("This constructor is deprecated and will be removed in v1.6. Use the constructor w" + + "ith the timestamp parameter instead. Since v1.5.28", true)] public SnapshotMetadata(string persistenceId, long sequenceNr) { } [Newtonsoft.Json.JsonConstructorAttribute()] public SnapshotMetadata(string persistenceId, long sequenceNr, System.DateTime timestamp) { } diff --git a/src/core/Akka.Persistence.TCK.Tests/LocalSnapshotStoreSpec.cs b/src/core/Akka.Persistence.TCK.Tests/LocalSnapshotStoreSpec.cs index b728d354816..2f44b7c9bc3 100644 --- a/src/core/Akka.Persistence.TCK.Tests/LocalSnapshotStoreSpec.cs +++ b/src/core/Akka.Persistence.TCK.Tests/LocalSnapshotStoreSpec.cs @@ -42,7 +42,7 @@ protected override void AfterAll() public void LocalSnapshotStore_can_snapshot_actors_with_PersistenceId_containing_invalid_path_characters() { var pid = @"p\/:*?-1"; - SnapshotStore.Tell(new SaveSnapshot(new SnapshotMetadata(pid, 1), "sample data"), TestActor); + SnapshotStore.Tell(new SaveSnapshot(new SnapshotMetadata(pid, 1, Sys.Scheduler.Now.DateTime), "sample data"), TestActor); ExpectMsg(); SnapshotStore.Tell(new LoadSnapshot(pid, SnapshotSelectionCriteria.Latest, long.MaxValue), TestActor); diff --git a/src/core/Akka.Persistence.TCK/Query/PersistenceIdsSpec.cs b/src/core/Akka.Persistence.TCK/Query/PersistenceIdsSpec.cs index 1d810a0b33c..468fec21598 100644 --- a/src/core/Akka.Persistence.TCK/Query/PersistenceIdsSpec.cs +++ b/src/core/Akka.Persistence.TCK/Query/PersistenceIdsSpec.cs @@ -247,7 +247,7 @@ protected IActorRef WriteSnapshot(string persistenceId, int n) ExpectMsg($"{persistenceId}-{i}-done"); } - var metadata = new SnapshotMetadata(persistenceId, n + 10); + var metadata = new SnapshotMetadata(persistenceId, n + 10, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, $"s-{n}"), _senderProbe.Ref); _senderProbe.ExpectMsg(); diff --git a/src/core/Akka.Persistence.TCK/Serialization/SnapshotStoreSerializationSpec.cs b/src/core/Akka.Persistence.TCK/Serialization/SnapshotStoreSerializationSpec.cs index 909a9ff1d0c..e75c3a6500d 100644 --- a/src/core/Akka.Persistence.TCK/Serialization/SnapshotStoreSerializationSpec.cs +++ b/src/core/Akka.Persistence.TCK/Serialization/SnapshotStoreSerializationSpec.cs @@ -69,7 +69,7 @@ public virtual void SnapshotStore_should_serialize_Payload() var snapshot = new Test.MySnapshot("a"); - var metadata = new SnapshotMetadata(Pid, 1); + var metadata = new SnapshotMetadata(Pid, 1, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, snapshot), probe.Ref); probe.ExpectMsg(); @@ -85,7 +85,7 @@ public virtual void SnapshotStore_should_serialize_Payload_with_string_manifest( var snapshot = new Test.MySnapshot2("a"); - var metadata = new SnapshotMetadata(Pid, 1); + var metadata = new SnapshotMetadata(Pid, 1, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, snapshot), probe.Ref); probe.ExpectMsg(); @@ -107,7 +107,7 @@ public virtual void SnapshotStore_should_serialize_AtLeastOnceDeliverySnapshot() }; var atLeastOnceDeliverySnapshot = new AtLeastOnceDeliverySnapshot(17, unconfirmed); - var metadata = new SnapshotMetadata(Pid, 2); + var metadata = new SnapshotMetadata(Pid, 2, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, atLeastOnceDeliverySnapshot), probe.Ref); probe.ExpectMsg(); @@ -123,7 +123,7 @@ public virtual void SnapshotStore_should_serialize_AtLeastOnceDeliverySnapshot_w var unconfirmed = Array.Empty(); var atLeastOnceDeliverySnapshot = new AtLeastOnceDeliverySnapshot(13, unconfirmed); - var metadata = new SnapshotMetadata(Pid, 2); + var metadata = new SnapshotMetadata(Pid, 2, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, atLeastOnceDeliverySnapshot), probe.Ref); probe.ExpectMsg(); @@ -138,7 +138,7 @@ public virtual void SnapshotStore_should_serialize_PersistentFSMSnapshot() var persistentFSMSnapshot = new PersistentFSM.PersistentFSMSnapshot("mystate", "mydata", TimeSpan.FromDays(4)); - var metadata = new SnapshotMetadata(Pid, 2); + var metadata = new SnapshotMetadata(Pid, 2, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, persistentFSMSnapshot), probe.Ref); probe.ExpectMsg(); diff --git a/src/core/Akka.Persistence.TCK/Snapshot/SnapshotStoreSpec.cs b/src/core/Akka.Persistence.TCK/Snapshot/SnapshotStoreSpec.cs index 1106e79191c..431b541384b 100644 --- a/src/core/Akka.Persistence.TCK/Snapshot/SnapshotStoreSpec.cs +++ b/src/core/Akka.Persistence.TCK/Snapshot/SnapshotStoreSpec.cs @@ -14,6 +14,7 @@ using Akka.Persistence.Fsm; using Akka.Persistence.TCK.Serialization; using Akka.TestKit; +using FluentAssertions.Extensions; using Xunit; using Xunit.Abstractions; @@ -100,7 +101,7 @@ private IEnumerable WriteSnapshots() { for (int i = 1; i <= 5; i++) { - var metadata = new SnapshotMetadata(Pid, i + 10); + var metadata = new SnapshotMetadata(Pid, i + 10, Sys.Scheduler.Now.DateTime); SnapshotStore.Tell(new SaveSnapshot(metadata, $"s-{i}"), _senderProbe.Ref); yield return _senderProbe.ExpectMsg().Metadata; } @@ -177,11 +178,13 @@ public virtual void SnapshotStore_should_load_the_most_recent_snapshot_matching_ && result.Snapshot.Snapshot.ToString() == "s-3"); } + // Issue #7312 + // Backward compatibility check, SnapshotMetadata .ctor should work if we pass in UtcNow [Fact] public virtual void SnapshotStore_should_delete_a_single_snapshot_identified_by_SequenceNr_in_snapshot_metadata() { var md = Metadata[2]; - md = new SnapshotMetadata(md.PersistenceId, md.SequenceNr); // don't care about timestamp for delete of a single snap + md = new SnapshotMetadata(md.PersistenceId, md.SequenceNr, Sys.Scheduler.Now.DateTime); var command = new DeleteSnapshot(md); var sub = CreateTestProbe(); @@ -198,6 +201,54 @@ public virtual void SnapshotStore_should_delete_a_single_snapshot_identified_by_ && result.Snapshot.Snapshot.ToString() == "s-2"); } + // Issue #7312 + // Backward compatibility check, old SnapshotMetadata .ctor default value should work as expected + [Fact] + public virtual void SnapshotStore_should_delete_a_single_snapshot_identified_by_SequenceNr_in_snapshot_metadata_if_timestamp_is_MinValue() + { + var md = Metadata[2]; + // In previous incarnation, timestamp argument defaults to DateTime.MinValue + md = new SnapshotMetadata(md.PersistenceId, md.SequenceNr, DateTime.MinValue); + var command = new DeleteSnapshot(md); + var sub = CreateTestProbe(); + + Subscribe(sub.Ref); + SnapshotStore.Tell(command, _senderProbe.Ref); + sub.ExpectMsg(command); + _senderProbe.ExpectMsg(); + + SnapshotStore.Tell(new LoadSnapshot(Pid, new SnapshotSelectionCriteria(md.SequenceNr), long.MaxValue), _senderProbe.Ref); + _senderProbe.ExpectMsg(result => + result.ToSequenceNr == long.MaxValue + && result.Snapshot != null + && result.Snapshot.Metadata.Equals(Metadata[1]) + && result.Snapshot.Snapshot.ToString() == "s-2"); + } + + // Issue #7312, this is a side effect of the ctor signature changes + // DeleteSnapshot should not delete snapshot if timestamp value does not meet deletion criteria + [Fact] + public virtual void SnapshotStore_should_not_delete_snapshot_identified_by_SequenceNr_if_metadata_timestamp_is_less_than_stored_timestamp() + { + var md = Metadata[2]; + // timestamp argument is less than the actual metadata data stored in the database, no deletion occured + md = new SnapshotMetadata(md.PersistenceId, md.SequenceNr, md.Timestamp - 2.Seconds()); + var command = new DeleteSnapshot(md); + var sub = CreateTestProbe(); + + Subscribe(sub.Ref); + SnapshotStore.Tell(command, _senderProbe.Ref); + sub.ExpectMsg(command); + _senderProbe.ExpectMsg(); + + SnapshotStore.Tell(new LoadSnapshot(Pid, new SnapshotSelectionCriteria(md.SequenceNr), long.MaxValue), _senderProbe.Ref); + _senderProbe.ExpectMsg(result => + result.ToSequenceNr == long.MaxValue + && result.Snapshot != null + && result.Snapshot.Metadata.Equals(Metadata[2]) + && result.Snapshot.Snapshot.ToString() == "s-3"); + } + [Fact] public virtual void SnapshotStore_should_delete_all_snapshots_matching_upper_sequence_number_and_timestamp_bounds() { @@ -260,7 +311,7 @@ public virtual void SnapshotStore_should_save_and_overwrite_snapshot_with_same_s [Fact] public virtual void SnapshotStore_should_save_bigger_size_snapshot() { - var metadata = new SnapshotMetadata(Pid, 100); + var metadata = new SnapshotMetadata(Pid, 100, Sys.Scheduler.Now.DateTime); var bigSnapshot = new byte[SnapshotByteSizeLimit]; new Random().NextBytes(bigSnapshot); SnapshotStore.Tell(new SaveSnapshot(metadata, bigSnapshot), _senderProbe.Ref); @@ -274,7 +325,7 @@ public virtual void ShouldSerializeSnapshots() if (!SupportsSerialization) return; var probe = CreateTestProbe(); - var metadata = new SnapshotMetadata(Pid, 100L); + var metadata = new SnapshotMetadata(Pid, 100L, Sys.Scheduler.Now.DateTime); var snap = new TestPayload(probe.Ref); SnapshotStore.Tell(new SaveSnapshot(metadata, snap), _senderProbe.Ref); diff --git a/src/core/Akka.Persistence/Eventsourced.cs b/src/core/Akka.Persistence/Eventsourced.cs index a0befa7741b..74980d43553 100644 --- a/src/core/Akka.Persistence/Eventsourced.cs +++ b/src/core/Akka.Persistence/Eventsourced.cs @@ -218,7 +218,7 @@ public void LoadSnapshot(string persistenceId, SnapshotSelectionCriteria criteri /// TBD public void SaveSnapshot(object snapshot) { - SnapshotStore.Tell(new SaveSnapshot(new SnapshotMetadata(SnapshotterId, SnapshotSequenceNr), snapshot)); + SnapshotStore.Tell(new SaveSnapshot(new SnapshotMetadata(SnapshotterId, SnapshotSequenceNr, Context.System.Scheduler.Now.Date), snapshot)); } /// @@ -230,7 +230,7 @@ public void SaveSnapshot(object snapshot) /// TBD public void DeleteSnapshot(long sequenceNr) { - SnapshotStore.Tell(new DeleteSnapshot(new SnapshotMetadata(SnapshotterId, sequenceNr))); + SnapshotStore.Tell(new DeleteSnapshot(new SnapshotMetadata(SnapshotterId, sequenceNr, DateTime.MinValue))); } /// diff --git a/src/core/Akka.Persistence/SnapshotProtocol.cs b/src/core/Akka.Persistence/SnapshotProtocol.cs index df39b38de3c..c00d723afb8 100644 --- a/src/core/Akka.Persistence/SnapshotProtocol.cs +++ b/src/core/Akka.Persistence/SnapshotProtocol.cs @@ -52,32 +52,28 @@ public int Compare(SnapshotMetadata x, SnapshotMetadata y) } /// - /// TBD + /// The singleton comparer instance. /// public static IComparer Comparer { get; } = new SnapshotMetadataComparer(); - /// - /// TBD - /// - public static DateTime TimestampNotSpecified = DateTime.MinValue; - /// /// Initializes a new instance of the class. /// /// The id of the persistent actor fro which the snapshot was taken. /// The sequence number at which the snapshot was taken. + [Obsolete("This constructor is deprecated and will be removed in v1.6. Use the constructor with the timestamp parameter instead. Since v1.5.28", true)] public SnapshotMetadata(string persistenceId, long sequenceNr) - : this(persistenceId, sequenceNr, TimestampNotSpecified) + : this(persistenceId, sequenceNr, DateTime.UtcNow) { } /// /// Initializes a new instance of the class. /// - /// The id of the persistent actor fro mwhich the snapshot was taken. + /// The id of the persistent actor from which the snapshot was taken. /// The sequence number at which the snapshot was taken. /// The time at which the snapshot was saved. - [JsonConstructor] + [JsonConstructor] // TODO: remove this public SnapshotMetadata(string persistenceId, long sequenceNr, DateTime timestamp) { PersistenceId = persistenceId; @@ -100,7 +96,13 @@ public SnapshotMetadata(string persistenceId, long sequenceNr, DateTime timestam /// public DateTime Timestamp { get; } - + /// + /// We will probably use nullable in the future, but for the time being + /// we use to represent "no timestamp" + /// + internal static DateTime TimestampNotSpecified => DateTime.MinValue; + + public override bool Equals(object obj) => Equals(obj as SnapshotMetadata); diff --git a/src/core/Akka/Actor/Scheduler/SchedulerBase.cs b/src/core/Akka/Actor/Scheduler/SchedulerBase.cs index ff57f975964..e41d3f761b7 100644 --- a/src/core/Akka/Actor/Scheduler/SchedulerBase.cs +++ b/src/core/Akka/Actor/Scheduler/SchedulerBase.cs @@ -98,7 +98,7 @@ void IActionScheduler.ScheduleRepeatedly(TimeSpan initialDelay, TimeSpan interva DateTimeOffset ITimeProvider.Now { get { return TimeNow; } } /// - /// TBD + /// The current time in UTC. /// protected abstract DateTimeOffset TimeNow { get; }