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; }
}