Group all 69 projects into category subfolders under src/ and tests/ so the Rider Solution Explorer mirrors the module structure. Folders: Core, Server, Drivers (with a nested Driver CLIs subfolder), Client, Tooling. - Move every project folder on disk with git mv (history preserved as renames). - Recompute relative paths in 57 .csproj files: cross-category ProjectReferences, the lib/ HintPath+None refs in Driver.Historian.Wonderware, and the external mxaccessgw refs in Driver.Galaxy and its test project. - Rebuild ZB.MOM.WW.OtOpcUa.slnx with nested solution folders. - Re-prefix project paths in functional scripts (e2e, compliance, smoke SQL, integration, install). Build green (0 errors); unit tests pass. Docs left for a separate pass. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
119 lines
3.5 KiB
C#
119 lines
3.5 KiB
C#
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Server.Redundancy;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Server.Tests;
|
|
|
|
[Trait("Category", "Unit")]
|
|
public sealed class ApplyLeaseRegistryTests
|
|
{
|
|
private static readonly DateTime T0 = new(2026, 4, 19, 12, 0, 0, DateTimeKind.Utc);
|
|
|
|
private sealed class FakeTimeProvider : TimeProvider
|
|
{
|
|
public DateTime Utc { get; set; } = T0;
|
|
public override DateTimeOffset GetUtcNow() => new(Utc, TimeSpan.Zero);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task EmptyRegistry_NotInProgress()
|
|
{
|
|
var reg = new ApplyLeaseRegistry();
|
|
reg.IsApplyInProgress.ShouldBeFalse();
|
|
await Task.Yield();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task BeginAndDispose_ClosesLease()
|
|
{
|
|
var reg = new ApplyLeaseRegistry();
|
|
|
|
await using (reg.BeginApplyLease(1, Guid.NewGuid()))
|
|
{
|
|
reg.IsApplyInProgress.ShouldBeTrue();
|
|
reg.OpenLeaseCount.ShouldBe(1);
|
|
}
|
|
|
|
reg.IsApplyInProgress.ShouldBeFalse();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Dispose_OnException_StillCloses()
|
|
{
|
|
var reg = new ApplyLeaseRegistry();
|
|
var publishId = Guid.NewGuid();
|
|
|
|
await Should.ThrowAsync<InvalidOperationException>(async () =>
|
|
{
|
|
await using var lease = reg.BeginApplyLease(1, publishId);
|
|
throw new InvalidOperationException("publish failed");
|
|
});
|
|
|
|
reg.IsApplyInProgress.ShouldBeFalse("await-using semantics must close the lease on exception");
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Dispose_TwiceIsSafe()
|
|
{
|
|
var reg = new ApplyLeaseRegistry();
|
|
var lease = reg.BeginApplyLease(1, Guid.NewGuid());
|
|
|
|
await lease.DisposeAsync();
|
|
await lease.DisposeAsync();
|
|
|
|
reg.IsApplyInProgress.ShouldBeFalse();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task MultipleLeases_Concurrent_StayIsolated()
|
|
{
|
|
var reg = new ApplyLeaseRegistry();
|
|
var id1 = Guid.NewGuid();
|
|
var id2 = Guid.NewGuid();
|
|
|
|
await using var lease1 = reg.BeginApplyLease(1, id1);
|
|
await using var lease2 = reg.BeginApplyLease(2, id2);
|
|
|
|
reg.OpenLeaseCount.ShouldBe(2);
|
|
await lease1.DisposeAsync();
|
|
reg.IsApplyInProgress.ShouldBeTrue("lease2 still open");
|
|
await lease2.DisposeAsync();
|
|
reg.IsApplyInProgress.ShouldBeFalse();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Watchdog_ClosesStaleLeases()
|
|
{
|
|
var clock = new FakeTimeProvider();
|
|
var reg = new ApplyLeaseRegistry(applyMaxDuration: TimeSpan.FromMinutes(10), timeProvider: clock);
|
|
|
|
_ = reg.BeginApplyLease(1, Guid.NewGuid()); // intentional leak; not awaited / disposed
|
|
|
|
// Lease still young → no-op.
|
|
clock.Utc = T0.AddMinutes(5);
|
|
reg.PruneStale().ShouldBe(0);
|
|
reg.IsApplyInProgress.ShouldBeTrue();
|
|
|
|
// Past the watchdog horizon → force-close.
|
|
clock.Utc = T0.AddMinutes(11);
|
|
var closed = reg.PruneStale();
|
|
|
|
closed.ShouldBe(1);
|
|
reg.IsApplyInProgress.ShouldBeFalse("ServiceLevel can't stick at mid-apply after a crashed publisher");
|
|
await Task.Yield();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Watchdog_LeavesRecentLeaseAlone()
|
|
{
|
|
var clock = new FakeTimeProvider();
|
|
var reg = new ApplyLeaseRegistry(applyMaxDuration: TimeSpan.FromMinutes(10), timeProvider: clock);
|
|
|
|
await using var lease = reg.BeginApplyLease(1, Guid.NewGuid());
|
|
clock.Utc = T0.AddMinutes(3);
|
|
|
|
reg.PruneStale().ShouldBe(0);
|
|
reg.IsApplyInProgress.ShouldBeTrue();
|
|
}
|
|
}
|