64e3fbe035
v2-ci / build (push) Failing after 1m43s
v2-ci / unit-tests (tests/Core/ZB.MOM.WW.OtOpcUa.Cluster.Tests) (push) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.ControlPlane.Tests) (push) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.OpcUaServer.Tests) (push) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.Runtime.Tests) (push) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.Security.Tests) (push) Has been skipped
v2-ci / integration (tests/Server/ZB.MOM.WW.OtOpcUa.Host.IntegrationTests) (push) Has been skipped
v2-ci / integration (tests/Server/ZB.MOM.WW.OtOpcUa.OpcUaServer.IntegrationTests) (push) Has been skipped
Adds <summary>, <param>, <typeparam>, and <inheritdoc/> tags to public members surfaced by commentchecker — resolves 5,847 of 5,869 issues (99.6%) across three /fixdocs passes.
120 lines
4.1 KiB
C#
120 lines
4.1 KiB
C#
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Core.Resilience;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Core.Tests.Resilience;
|
|
|
|
[Trait("Category", "Unit")]
|
|
public sealed class DriverResilienceStatusTrackerTests
|
|
{
|
|
private static readonly DateTime Now = new(2026, 4, 19, 12, 0, 0, DateTimeKind.Utc);
|
|
|
|
/// <summary>Verifies that TryGet returns null before any write operations.</summary>
|
|
[Fact]
|
|
public void TryGet_Returns_Null_Before_AnyWrite()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.TryGet("drv", "host").ShouldBeNull();
|
|
}
|
|
|
|
/// <summary>Verifies that RecordFailure accumulates consecutive failures.</summary>
|
|
[Fact]
|
|
public void RecordFailure_Accumulates_ConsecutiveFailures()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.RecordFailure("drv", "host", Now);
|
|
tracker.RecordFailure("drv", "host", Now.AddSeconds(1));
|
|
tracker.RecordFailure("drv", "host", Now.AddSeconds(2));
|
|
|
|
tracker.TryGet("drv", "host")!.ConsecutiveFailures.ShouldBe(3);
|
|
}
|
|
|
|
/// <summary>Verifies that RecordSuccess resets consecutive failures to zero.</summary>
|
|
[Fact]
|
|
public void RecordSuccess_Resets_ConsecutiveFailures()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
tracker.RecordFailure("drv", "host", Now);
|
|
tracker.RecordFailure("drv", "host", Now.AddSeconds(1));
|
|
|
|
tracker.RecordSuccess("drv", "host", Now.AddSeconds(2));
|
|
|
|
tracker.TryGet("drv", "host")!.ConsecutiveFailures.ShouldBe(0);
|
|
}
|
|
|
|
/// <summary>Verifies that RecordBreakerOpen populates the LastBreakerOpenUtc timestamp.</summary>
|
|
[Fact]
|
|
public void RecordBreakerOpen_Populates_LastBreakerOpenUtc()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.RecordBreakerOpen("drv", "host", Now);
|
|
|
|
tracker.TryGet("drv", "host")!.LastBreakerOpenUtc.ShouldBe(Now);
|
|
}
|
|
|
|
/// <summary>Verifies that RecordRecycle populates the LastRecycleUtc timestamp.</summary>
|
|
[Fact]
|
|
public void RecordRecycle_Populates_LastRecycleUtc()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.RecordRecycle("drv", "host", Now);
|
|
|
|
tracker.TryGet("drv", "host")!.LastRecycleUtc.ShouldBe(Now);
|
|
}
|
|
|
|
/// <summary>Verifies that RecordFootprint captures baseline and current memory usage.</summary>
|
|
[Fact]
|
|
public void RecordFootprint_CapturesBaselineAndCurrent()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.RecordFootprint("drv", "host", baselineBytes: 100_000_000, currentBytes: 150_000_000, Now);
|
|
|
|
var snap = tracker.TryGet("drv", "host")!;
|
|
snap.BaselineFootprintBytes.ShouldBe(100_000_000);
|
|
snap.CurrentFootprintBytes.ShouldBe(150_000_000);
|
|
}
|
|
|
|
/// <summary>Verifies that different hosts are tracked independently.</summary>
|
|
[Fact]
|
|
public void DifferentHosts_AreIndependent()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
|
|
tracker.RecordFailure("drv", "host-a", Now);
|
|
tracker.RecordFailure("drv", "host-b", Now);
|
|
tracker.RecordSuccess("drv", "host-a", Now.AddSeconds(1));
|
|
|
|
tracker.TryGet("drv", "host-a")!.ConsecutiveFailures.ShouldBe(0);
|
|
tracker.TryGet("drv", "host-b")!.ConsecutiveFailures.ShouldBe(1);
|
|
}
|
|
|
|
/// <summary>Verifies that Snapshot returns all tracked driver-host pairs.</summary>
|
|
[Fact]
|
|
public void Snapshot_ReturnsAll_TrackedPairs()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
tracker.RecordFailure("drv-1", "host-a", Now);
|
|
tracker.RecordFailure("drv-1", "host-b", Now);
|
|
tracker.RecordFailure("drv-2", "host-a", Now);
|
|
|
|
var snapshot = tracker.Snapshot();
|
|
|
|
snapshot.Count.ShouldBe(3);
|
|
}
|
|
|
|
/// <summary>Verifies that concurrent writes do not lose failure records.</summary>
|
|
[Fact]
|
|
public void ConcurrentWrites_DoNotLose_Failures()
|
|
{
|
|
var tracker = new DriverResilienceStatusTracker();
|
|
Parallel.For(0, 500, _ => tracker.RecordFailure("drv", "host", Now));
|
|
|
|
tracker.TryGet("drv", "host")!.ConsecutiveFailures.ShouldBe(500);
|
|
}
|
|
}
|