using ScadaLink.Commons.Messages.Health; namespace ScadaLink.HealthMonitoring; /// /// In-memory state for a single site's health, stored by the central aggregator. /// Immutable: every state transition produces a new instance which the aggregator /// installs into its ConcurrentDictionary via an atomic compare-and-swap. /// This makes handing the reference straight to UI callers safe — a consumer can /// never observe a torn or half-applied update. /// public sealed record SiteHealthState { public required string SiteId { get; init; } /// /// The latest full received for the site, or /// null if the site is known only via heartbeats and has not yet sent /// a report. /// public SiteHealthReport? LatestReport { get; init; } /// /// Time the latest full was processed, or /// null if the site is known only via heartbeats and has not yet sent /// a report. Used by the UI to surface report staleness during failover; /// the null case must be rendered as "no report yet" rather than as a /// timestamp (a default sentinel would display as year-0001). /// public DateTimeOffset? LastReportReceivedAt { get; init; } /// /// Time the most recent signal of any kind (full report OR heartbeat) was /// received. Drives offline detection — heartbeats from the standby keep the /// site marked online even when the active node is unable to produce a report /// (mid-failover, brief stalls). Heartbeat cadence is owned by the Cluster /// Infrastructure / SiteCommunicationActor (every /// CommunicationOptions.TransportHeartbeatInterval — 5s by default). /// public DateTimeOffset LastHeartbeatAt { get; init; } public long LastSequenceNumber { get; init; } public bool IsOnline { get; init; } }