Deployment Manager (WP-1–8, WP-16): - DeploymentService: full pipeline (flatten→validate→send→track→audit) - OperationLockManager: per-instance concurrency control - StateTransitionValidator: Enabled/Disabled/NotDeployed transition matrix - ArtifactDeploymentService: broadcast to all sites with per-site results - Deployment identity (GUID + revision hash), idempotency, staleness detection - Instance lifecycle commands (disable/enable/delete) with deduplication Store-and-Forward (WP-9–15): - StoreAndForwardStorage: SQLite persistence, 3 categories, no max buffer - StoreAndForwardService: fixed-interval retry, transient-only buffering, parking - ReplicationService: async best-effort to standby (fire-and-forget) - Parked message management (query/retry/discard from central) - Messages survive instance deletion, S&F drains on disable 620 tests pass (+79 new), zero warnings.
137 lines
4.2 KiB
C#
137 lines
4.2 KiB
C#
using Microsoft.Extensions.Logging;
|
|
using ScadaLink.Commons.Types.Enums;
|
|
|
|
namespace ScadaLink.StoreAndForward;
|
|
|
|
/// <summary>
|
|
/// WP-11: Async replication of buffer operations to standby node.
|
|
///
|
|
/// - Forwards add/remove/park operations to standby via a replication handler.
|
|
/// - No ack wait (fire-and-forget per design).
|
|
/// - Standby applies operations to its own SQLite.
|
|
/// - On failover, standby resumes delivery from its replicated state.
|
|
/// </summary>
|
|
public class ReplicationService
|
|
{
|
|
private readonly StoreAndForwardOptions _options;
|
|
private readonly ILogger<ReplicationService> _logger;
|
|
private Func<ReplicationOperation, Task>? _replicationHandler;
|
|
|
|
public ReplicationService(
|
|
StoreAndForwardOptions options,
|
|
ILogger<ReplicationService> logger)
|
|
{
|
|
_options = options;
|
|
_logger = logger;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the handler for forwarding replication operations to the standby node.
|
|
/// Typically wraps Akka Tell to the standby's replication actor.
|
|
/// </summary>
|
|
public void SetReplicationHandler(Func<ReplicationOperation, Task> handler)
|
|
{
|
|
_replicationHandler = handler;
|
|
}
|
|
|
|
/// <summary>
|
|
/// WP-11: Replicates an enqueue operation to standby (fire-and-forget).
|
|
/// </summary>
|
|
public void ReplicateEnqueue(StoreAndForwardMessage message)
|
|
{
|
|
if (!_options.ReplicationEnabled || _replicationHandler == null) return;
|
|
|
|
FireAndForget(new ReplicationOperation(
|
|
ReplicationOperationType.Add,
|
|
message.Id,
|
|
message));
|
|
}
|
|
|
|
/// <summary>
|
|
/// WP-11: Replicates a remove operation to standby (fire-and-forget).
|
|
/// </summary>
|
|
public void ReplicateRemove(string messageId)
|
|
{
|
|
if (!_options.ReplicationEnabled || _replicationHandler == null) return;
|
|
|
|
FireAndForget(new ReplicationOperation(
|
|
ReplicationOperationType.Remove,
|
|
messageId,
|
|
null));
|
|
}
|
|
|
|
/// <summary>
|
|
/// WP-11: Replicates a park operation to standby (fire-and-forget).
|
|
/// </summary>
|
|
public void ReplicatePark(StoreAndForwardMessage message)
|
|
{
|
|
if (!_options.ReplicationEnabled || _replicationHandler == null) return;
|
|
|
|
FireAndForget(new ReplicationOperation(
|
|
ReplicationOperationType.Park,
|
|
message.Id,
|
|
message));
|
|
}
|
|
|
|
/// <summary>
|
|
/// WP-11: Applies a replicated operation received from the active node.
|
|
/// Used by the standby node to keep its SQLite in sync.
|
|
/// </summary>
|
|
public async Task ApplyReplicatedOperationAsync(
|
|
ReplicationOperation operation,
|
|
StoreAndForwardStorage storage)
|
|
{
|
|
switch (operation.OperationType)
|
|
{
|
|
case ReplicationOperationType.Add when operation.Message != null:
|
|
await storage.EnqueueAsync(operation.Message);
|
|
break;
|
|
|
|
case ReplicationOperationType.Remove:
|
|
await storage.RemoveMessageAsync(operation.MessageId);
|
|
break;
|
|
|
|
case ReplicationOperationType.Park when operation.Message != null:
|
|
operation.Message.Status = StoreAndForwardMessageStatus.Parked;
|
|
await storage.UpdateMessageAsync(operation.Message);
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void FireAndForget(ReplicationOperation operation)
|
|
{
|
|
Task.Run(async () =>
|
|
{
|
|
try
|
|
{
|
|
await _replicationHandler!.Invoke(operation);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
// WP-11: No ack wait — log and move on
|
|
_logger.LogDebug(ex,
|
|
"Replication of {OpType} for message {MessageId} failed (best-effort)",
|
|
operation.OperationType, operation.MessageId);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// WP-11: Represents a buffer operation to be replicated to standby.
|
|
/// </summary>
|
|
public record ReplicationOperation(
|
|
ReplicationOperationType OperationType,
|
|
string MessageId,
|
|
StoreAndForwardMessage? Message);
|
|
|
|
/// <summary>
|
|
/// WP-11: Types of buffer operations that are replicated.
|
|
/// </summary>
|
|
public enum ReplicationOperationType
|
|
{
|
|
Add,
|
|
Remove,
|
|
Park
|
|
}
|