diff --git a/src/ZB.MOM.WW.ScadaBridge.SiteEventLogging/ISiteEventLogger.cs b/src/ZB.MOM.WW.ScadaBridge.SiteEventLogging/ISiteEventLogger.cs
index 07fc9ea4..630822ef 100644
--- a/src/ZB.MOM.WW.ScadaBridge.SiteEventLogging/ISiteEventLogger.cs
+++ b/src/ZB.MOM.WW.ScadaBridge.SiteEventLogging/ISiteEventLogger.cs
@@ -11,7 +11,7 @@ public interface ISiteEventLogger
/// completes once the event is durably persisted and faults if
/// the write fails, so callers that await it observe success or failure.
///
- /// Category: script, alarm, deployment, connection, store_and_forward, instance_lifecycle
+ /// Category: script, alarm, deployment, connection, store_and_forward, instance_lifecycle, notification
/// Info, Warning, or Error
/// Optional instance ID associated with the event
/// Source identifier, e.g., "ScriptActor:MonitorSpeed"
diff --git a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ServiceCollectionExtensions.cs b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ServiceCollectionExtensions.cs
index e0cf2a46..163a0cb8 100644
--- a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ServiceCollectionExtensions.cs
+++ b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ServiceCollectionExtensions.cs
@@ -2,6 +2,7 @@ using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Options;
using ZB.MOM.WW.ScadaBridge.Commons.Interfaces.Services;
+using ZB.MOM.WW.ScadaBridge.SiteEventLogging;
namespace ZB.MOM.WW.ScadaBridge.StoreAndForward;
@@ -49,13 +50,19 @@ public static class ServiceCollectionExtensions
// observable in the central audit log instead of producing a
// silent empty-string SourceSite.
var siteId = siteContext?.SiteId ?? string.Empty;
+ // M1.7: optional site operational-event log. Resolved through
+ // GetService so a host (or test) that has not called
+ // AddSiteEventLogging simply gets null and the S&F activity stays
+ // a no-op for site-event purposes.
+ var siteEventLogger = sp.GetService();
return new StoreAndForwardService(
storage,
options,
logger,
replication,
cachedCallObserver,
- siteId);
+ siteId,
+ siteEventLogger);
});
services.AddSingleton(sp =>
diff --git a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/StoreAndForwardService.cs b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/StoreAndForwardService.cs
index 6b082579..e335ea01 100644
--- a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/StoreAndForwardService.cs
+++ b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/StoreAndForwardService.cs
@@ -3,6 +3,7 @@ using ZB.MOM.WW.ScadaBridge.Commons.Interfaces.Services;
using ZB.MOM.WW.ScadaBridge.Commons.Observability;
using ZB.MOM.WW.ScadaBridge.Commons.Types;
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
+using ZB.MOM.WW.ScadaBridge.SiteEventLogging;
namespace ZB.MOM.WW.ScadaBridge.StoreAndForward;
@@ -44,6 +45,15 @@ public class StoreAndForwardService
///
private readonly ICachedCallLifecycleObserver? _cachedCallObserver;
///
+ /// M1.7: optional site operational-event log. When non-null the service maps
+ /// its own buffer/retry/park activity (the same activity that drives
+ /// ) onto site events — store_and_forward for the
+ /// cached-call categories and notification for the site's
+ /// forward-to-central notification path. Best-effort and fire-and-forget so a
+ /// failing logger never affects delivery bookkeeping.
+ ///
+ private readonly ISiteEventLogger? _siteEventLogger;
+ ///
/// Audit Log #23 (M3 Bundle E — Task E4): site id stamped onto the
/// cached-call attempt context so the audit bridge can build the
/// half of the telemetry packet.
@@ -173,13 +183,20 @@ public class StoreAndForwardService
/// Optional replication service for standby synchronization.
/// Optional observer for cached call lifecycle events.
/// The site identifier this service belongs to.
+ ///
+ /// M1.7: optional site operational-event log. When non-null, buffer/retry/park
+ /// activity is mirrored to site events (store_and_forward /
+ /// notification by category). Optional with a null default so the
+ /// many direct-construction tests still compile unchanged.
+ ///
public StoreAndForwardService(
StoreAndForwardStorage storage,
StoreAndForwardOptions options,
ILogger logger,
ReplicationService? replication = null,
ICachedCallLifecycleObserver? cachedCallObserver = null,
- string siteId = "")
+ string siteId = "",
+ ISiteEventLogger? siteEventLogger = null)
{
_storage = storage;
_options = options;
@@ -191,6 +208,91 @@ public class StoreAndForwardService
// audit pipeline keying off SourceSite) never see an empty string and
// a misconfigured host is recognisable in the central log.
_siteId = string.IsNullOrWhiteSpace(siteId) ? UnknownSiteSentinel : siteId;
+ _siteEventLogger = siteEventLogger;
+
+ // M1.7: ride the existing activity hook to emit site operational events.
+ // RaiseActivity already isolates a throwing subscriber, so a failing
+ // event log can never be misclassified as a transient delivery failure
+ // (StoreAndForward-009). Only subscribe when a logger is wired so the
+ // legacy (test/central) construction path stays a no-op.
+ if (_siteEventLogger != null)
+ {
+ OnActivity += EmitSiteEvent;
+ }
+ }
+
+ ///
+ /// M1.7: maps one store-and-forward activity to a site operational event,
+ /// following the Site Event Logging spec's per-category scope
+ /// (Component-SiteEventLogging.md §"Events Logged"):
+ ///
+ /// - Cached-call categories
+ /// ( /
+ /// ) log under
+ /// store_and_forward for queued / retried / parked / retry-delivered
+ /// activity.
+ /// - The site's notification forward-to-central path
+ /// () logs under
+ /// notification ONLY on a forward FAILURE (buffered after the
+ /// immediate forward threw) or a park (long-buffered / retries exhausted).
+ /// Routine enqueue and forward-success are deliberately NOT logged — central's
+ /// Notifications table is the record of audit; the site only fills the
+ /// in-transit blind spot when central is unreachable.
+ ///
+ /// A successful immediate cached-call Delivered is the normal hot path and
+ /// is not logged.
+ ///
+ private void EmitSiteEvent(string action, StoreAndForwardCategory category, string detail)
+ {
+ var logger = _siteEventLogger;
+ if (logger == null)
+ {
+ return;
+ }
+
+ // An immediate-delivery success is the normal hot path, not an
+ // operational event. A retry-loop success (detail "Delivered to … after
+ // N retries") IS logged for cached calls — it records a recovery.
+ if (action == "Delivered" && detail.StartsWith("Immediate", StringComparison.Ordinal))
+ {
+ return;
+ }
+
+ if (category == StoreAndForwardCategory.Notification)
+ {
+ // Spec: log only forward-failure (the immediate forward threw and the
+ // notification was buffered for retry — detail "Buffered for retry:")
+ // and park. A routine "No handler registered, buffered" enqueue and a
+ // forward-success "Delivered" are deliberately NOT logged.
+ var isForwardFailure = action == "Queued"
+ && detail.StartsWith("Buffered for retry", StringComparison.Ordinal);
+ if (!isForwardFailure && action != "Parked")
+ {
+ return;
+ }
+
+ var notifSeverity = action == "Parked" ? "Error" : "Warning";
+ _ = logger.LogEventAsync(
+ "notification", notifSeverity, instanceId: null,
+ source: "StoreAndForwardService",
+ message: $"Notification {action.ToLowerInvariant()}: {detail}");
+ return;
+ }
+
+ // Cached-call categories: queued / retried / parked / retry-delivered.
+ // Severity: parking is an Error (delivery abandoned for retry purposes);
+ // queue/retry/requeue are Warning; a retry-loop Delivered is Info.
+ var severity = action switch
+ {
+ "Parked" => "Error",
+ "Delivered" => "Info",
+ _ => "Warning",
+ };
+
+ _ = logger.LogEventAsync(
+ "store_and_forward", severity, instanceId: null,
+ source: "StoreAndForwardService",
+ message: $"Operation {action.ToLowerInvariant()}: {detail}");
}
///
diff --git a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ZB.MOM.WW.ScadaBridge.StoreAndForward.csproj b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ZB.MOM.WW.ScadaBridge.StoreAndForward.csproj
index c997e571..4f568eb4 100644
--- a/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ZB.MOM.WW.ScadaBridge.StoreAndForward.csproj
+++ b/src/ZB.MOM.WW.ScadaBridge.StoreAndForward/ZB.MOM.WW.ScadaBridge.StoreAndForward.csproj
@@ -17,6 +17,7 @@
+
diff --git a/tests/ZB.MOM.WW.ScadaBridge.StoreAndForward.Tests/StoreAndForwardSiteEventTests.cs b/tests/ZB.MOM.WW.ScadaBridge.StoreAndForward.Tests/StoreAndForwardSiteEventTests.cs
new file mode 100644
index 00000000..2c3dcea7
--- /dev/null
+++ b/tests/ZB.MOM.WW.ScadaBridge.StoreAndForward.Tests/StoreAndForwardSiteEventTests.cs
@@ -0,0 +1,168 @@
+using System.Collections.Concurrent;
+using Microsoft.Data.Sqlite;
+using Microsoft.Extensions.Logging.Abstractions;
+using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
+using ZB.MOM.WW.ScadaBridge.SiteEventLogging;
+
+namespace ZB.MOM.WW.ScadaBridge.StoreAndForward.Tests;
+
+///
+/// M1.7: the StoreAndForwardService emits site operational events for its own
+/// buffer/park activity — store_and_forward for cached-call categories
+/// (ExternalSystem / CachedDbWrite) and notification for the site's
+/// notification forward-to-central path. Emission rides the existing
+/// OnActivity hook and is best-effort (a failing logger never affects
+/// delivery bookkeeping).
+///
+public class StoreAndForwardSiteEventTests : IAsyncLifetime, IDisposable
+{
+ private sealed record Entry(string EventType, string Severity, string Source, string Message);
+
+ private sealed class FakeSiteEventLogger : ISiteEventLogger
+ {
+ private readonly ConcurrentQueue _entries = new();
+ public IReadOnlyList Entries => _entries.ToArray();
+ public IReadOnlyList OfType(string t) => _entries.Where(e => e.EventType == t).ToArray();
+
+ public Task LogEventAsync(string eventType, string severity, string? instanceId,
+ string source, string message, string? details = null)
+ {
+ _entries.Enqueue(new Entry(eventType, severity, source, message));
+ return Task.CompletedTask;
+ }
+
+ public long FailedWriteCount => 0;
+ }
+
+ private readonly SqliteConnection _keepAlive;
+ private readonly StoreAndForwardStorage _storage;
+ private readonly StoreAndForwardOptions _options;
+ private readonly FakeSiteEventLogger _siteLog = new();
+ private readonly StoreAndForwardService _service;
+
+ public StoreAndForwardSiteEventTests()
+ {
+ var dbName = $"SiteEvt_{Guid.NewGuid():N}";
+ var connStr = $"Data Source={dbName};Mode=Memory;Cache=Shared";
+ _keepAlive = new SqliteConnection(connStr);
+ _keepAlive.Open();
+
+ _storage = new StoreAndForwardStorage(connStr, NullLogger.Instance);
+ _options = new StoreAndForwardOptions
+ {
+ DefaultRetryInterval = TimeSpan.Zero,
+ DefaultMaxRetries = 1,
+ RetryTimerInterval = TimeSpan.FromMinutes(10)
+ };
+
+ _service = new StoreAndForwardService(
+ _storage, _options, NullLogger.Instance,
+ replication: null, cachedCallObserver: null, siteId: "site-a",
+ siteEventLogger: _siteLog);
+ }
+
+ public async Task InitializeAsync() => await _storage.InitializeAsync();
+ public Task DisposeAsync() => Task.CompletedTask;
+ public void Dispose() => _keepAlive.Dispose();
+
+ [Fact]
+ public async Task BufferForRetry_ExternalSystem_EmitsStoreAndForwardSiteEvent()
+ {
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.ExternalSystem,
+ _ => throw new HttpRequestException("transient"));
+
+ await _service.EnqueueAsync(StoreAndForwardCategory.ExternalSystem, "api.example.com", """{}""", "Pump1");
+
+ var rows = _siteLog.OfType("store_and_forward");
+ Assert.Contains(rows, r => r.Severity == "Warning" &&
+ r.Source == "StoreAndForwardService" &&
+ r.Message.Contains("queued", StringComparison.OrdinalIgnoreCase));
+ // The cached-call categories must NOT surface as notification events.
+ Assert.Empty(_siteLog.OfType("notification"));
+ }
+
+ [Fact]
+ public async Task ForwardFailure_Notification_EmitsNotificationSiteEvent()
+ {
+ // The site's notification role is forward-to-central. When the immediate
+ // forward to central throws (central unreachable), the notification is
+ // buffered for retry — a forward FAILURE, which the spec says to log as a
+ // `notification` site event (filling the in-transit blind spot).
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.Notification,
+ _ => throw new HttpRequestException("central unreachable"));
+
+ await _service.EnqueueAsync(StoreAndForwardCategory.Notification, "list-a", """{}""", "Pump1");
+
+ var rows = _siteLog.OfType("notification");
+ Assert.Contains(rows, r => r.Severity == "Warning" &&
+ r.Source == "StoreAndForwardService" &&
+ r.Message.Contains("queued", StringComparison.OrdinalIgnoreCase));
+ // A notification forward-failure is not a store_and_forward (cached-call) event.
+ Assert.Empty(_siteLog.OfType("store_and_forward"));
+ }
+
+ [Fact]
+ public async Task RoutineEnqueue_Notification_DoesNotEmitSiteEvent()
+ {
+ // Spec: routine enqueue / forward-success on the notification path are
+ // deliberately NOT logged — central's Notifications table is the audit
+ // record of record. A successful immediate forward emits no site event.
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.Notification,
+ _ => Task.FromResult(true));
+
+ await _service.EnqueueAsync(StoreAndForwardCategory.Notification, "list-a", """{}""", "Pump1");
+
+ Assert.Empty(_siteLog.OfType("notification"));
+ }
+
+ [Fact]
+ public async Task Park_Notification_EmitsErrorNotificationSiteEvent()
+ {
+ // A long-buffered notification that exhausts retries is parked — the spec
+ // logs this as a `notification` event (Error severity).
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.Notification,
+ _ => throw new HttpRequestException("central unreachable"));
+
+ await _service.EnqueueAsync(
+ StoreAndForwardCategory.Notification, "list-a", """{}""", "Pump1",
+ attemptImmediateDelivery: false, maxRetries: 1);
+
+ await _service.RetryPendingMessagesAsync();
+
+ var rows = _siteLog.OfType("notification");
+ Assert.Contains(rows, r => r.Severity == "Error" &&
+ r.Message.Contains("parked", StringComparison.OrdinalIgnoreCase));
+ }
+
+ [Fact]
+ public async Task Park_ExternalSystem_EmitsErrorStoreAndForwardSiteEvent()
+ {
+ // MaxRetries = 1 → the first sweep retry parks the message.
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.ExternalSystem,
+ _ => throw new HttpRequestException("transient"));
+
+ await _service.EnqueueAsync(
+ StoreAndForwardCategory.ExternalSystem, "api.example.com", """{}""", "Pump1",
+ attemptImmediateDelivery: false, maxRetries: 1);
+
+ await _service.RetryPendingMessagesAsync();
+
+ var rows = _siteLog.OfType("store_and_forward");
+ Assert.Contains(rows, r => r.Severity == "Error" &&
+ r.Message.Contains("parked", StringComparison.OrdinalIgnoreCase));
+ }
+
+ [Fact]
+ public async Task DeliveredImmediately_DoesNotEmitSiteEvent()
+ {
+ // A successful immediate delivery is the normal hot path — it is not a
+ // store-and-forward buffering event, so no operational event is logged.
+ _service.RegisterDeliveryHandler(StoreAndForwardCategory.ExternalSystem,
+ _ => Task.FromResult(true));
+
+ await _service.EnqueueAsync(StoreAndForwardCategory.ExternalSystem, "api", """{}""", "Pump1");
+
+ Assert.Empty(_siteLog.OfType("store_and_forward"));
+ Assert.Empty(_siteLog.OfType("notification"));
+ }
+}