Files
lmxopcua/tests/Server/ZB.MOM.WW.OtOpcUa.Host.IntegrationTests/FleetDiagnosticsRoundTripTests.cs
T
Joseph Doherty 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
docs: backfill XML documentation across 756 files
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.
2026-05-28 08:10:17 -04:00

84 lines
3.5 KiB
C#

using Microsoft.Extensions.DependencyInjection;
using Shouldly;
using Xunit;
using ZB.MOM.WW.OtOpcUa.Commons.Interfaces;
using ZB.MOM.WW.OtOpcUa.Commons.Messages.Admin;
namespace ZB.MOM.WW.OtOpcUa.Host.IntegrationTests;
/// <summary>
/// End-to-end <see cref="IFleetDiagnosticsClient"/> round-trip via the cluster:
/// admin node asks node-B's <c>DriverHostActor</c> for a snapshot via <c>ActorSelection</c>.
/// Verifies the cross-node Ask/Reply works and the snapshot reflects the target node's
/// view (NodeId + CurrentRevision after a deploy).
/// </summary>
public sealed class FleetDiagnosticsRoundTripTests
{
private static CancellationToken Ct => TestContext.Current.CancellationToken;
/// <summary>Verifies that get diagnostics returns a snapshot with the target node ID.</summary>
[Fact]
public async Task GetDiagnostics_returns_snapshot_with_target_NodeId()
{
await using var harness = await TwoNodeClusterHarness.StartAsync();
// Resolve target NodeId from the cluster — second member ordered by address.
var members = Akka.Cluster.Cluster.Get(harness.NodeASystem).State.Members
.OrderBy(m => m.Address.ToString())
.ToArray();
members.Length.ShouldBe(2);
var targetAddress = members[1].Address;
var targetNodeId = Commons.Types.NodeId.Parse($"{targetAddress.Host}:{targetAddress.Port}");
await using var scope = harness.NodeA.Services.CreateAsyncScope();
var client = scope.ServiceProvider.GetRequiredService<IFleetDiagnosticsClient>();
var snapshot = await client.GetDiagnosticsAsync(targetNodeId, Ct);
snapshot.NodeId.ShouldBe(targetNodeId);
snapshot.Drivers.ShouldBeEmpty(); // No driver children yet (F7).
snapshot.AsOfUtc.ShouldBeGreaterThan(DateTime.UtcNow.AddSeconds(-30));
}
/// <summary>Verifies that get diagnostics after deploy reports the current revision.</summary>
[Fact]
public async Task GetDiagnostics_after_deploy_reports_current_revision()
{
await using var harness = await TwoNodeClusterHarness.StartAsync();
await using var scope = harness.NodeA.Services.CreateAsyncScope();
var adminOps = scope.ServiceProvider.GetRequiredService<IAdminOperationsClient>();
var diagnostics = scope.ServiceProvider.GetRequiredService<IFleetDiagnosticsClient>();
var deploy = await adminOps.StartDeploymentAsync(createdBy: "alice@test", Ct);
deploy.Outcome.ShouldBe(StartDeploymentOutcome.Accepted);
var expectedRev = deploy.RevisionHash!.Value;
// Wait until both DriverHostActors have caught up to the deployed revision.
var members = Akka.Cluster.Cluster.Get(harness.NodeASystem).State.Members
.OrderBy(m => m.Address.ToString())
.Select(m => Commons.Types.NodeId.Parse($"{m.Address.Host}:{m.Address.Port}"))
.ToArray();
foreach (var nodeId in members)
{
await WaitForAsync(async () =>
{
var snap = await diagnostics.GetDiagnosticsAsync(nodeId, Ct);
return snap.CurrentRevision == expectedRev;
}, TimeSpan.FromSeconds(15));
}
}
private static async Task WaitForAsync(Func<Task<bool>> condition, TimeSpan timeout)
{
var deadline = DateTime.UtcNow + timeout;
while (DateTime.UtcNow < deadline)
{
if (await condition()) return;
await Task.Delay(200);
}
throw new TimeoutException($"Condition not met within {timeout}");
}
}