using Shouldly; using Xunit; using ZB.MOM.WW.OtOpcUa.Core.Abstractions; namespace ZB.MOM.WW.OtOpcUa.Driver.OpcUaClient.IntegrationTests; /// /// End-to-end smoke against a live opc-plc (task #215). Drives the real /// OPC UA Secure Channel + Session + MonitoredItem exchange — no mocks. Every /// test here proves a capability surface that loopback against our own server /// couldn't exercise cleanly: real cert negotiation, real endpoint descriptions, /// real simulated nodes that change without a write. /// [Collection(OpcPlcCollection.Name)] [Trait("Category", "Integration")] [Trait("Simulator", "opc-plc")] public sealed class OpcUaClientSmokeTests(OpcPlcFixture sim) { [Fact] public async Task Client_connects_and_reads_StepUp_node_through_real_OPC_UA_stack() { if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason); var options = OpcPlcProfile.BuildOptions(sim.EndpointUrl); await using var drv = new OpcUaClientDriver(options, driverInstanceId: "opcua-smoke-read"); await drv.InitializeAsync("{}", TestContext.Current.CancellationToken); var snapshots = await drv.ReadAsync( [OpcPlcProfile.StepUp], TestContext.Current.CancellationToken); snapshots.Count.ShouldBe(1); snapshots[0].StatusCode.ShouldBe(0u, "opc-plc StepUp read must succeed end-to-end"); snapshots[0].Value.ShouldNotBeNull("StepUp always has a current value"); } [Fact] public async Task Client_reads_batch_of_varied_types_from_live_simulator() { if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason); var options = OpcPlcProfile.BuildOptions(sim.EndpointUrl); await using var drv = new OpcUaClientDriver(options, driverInstanceId: "opcua-smoke-batch"); await drv.InitializeAsync("{}", TestContext.Current.CancellationToken); var snapshots = await drv.ReadAsync( [OpcPlcProfile.StepUp, OpcPlcProfile.RandomSignedInt32, OpcPlcProfile.AlternatingBoolean], TestContext.Current.CancellationToken); snapshots.Count.ShouldBe(3); foreach (var s in snapshots) { s.StatusCode.ShouldBe(0u); s.Value.ShouldNotBeNull(); } // AlternatingBoolean should decode as a bool specifically — catches a common // attribute-mapping regression where the driver stringifies variant values. snapshots[2].Value.ShouldBeOfType(); } [Fact] public async Task Client_subscribe_receives_StepUp_data_changes_from_live_server() { if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason); var options = OpcPlcProfile.BuildOptions(sim.EndpointUrl); await using var drv = new OpcUaClientDriver(options, driverInstanceId: "opcua-smoke-sub"); await drv.InitializeAsync("{}", TestContext.Current.CancellationToken); var observed = new List(); var gate = new SemaphoreSlim(0); drv.OnDataChange += (_, e) => { lock (observed) observed.Add(e); gate.Release(); }; var handle = await drv.SubscribeAsync( [OpcPlcProfile.FastUInt1], TimeSpan.FromMilliseconds(250), TestContext.Current.CancellationToken); // FastUInt1 ticks every 100 ms — one publishing interval (250 ms) should deliver. // Wait up to 3 s to tolerate container warm-up + first-publish delay. var got = await gate.WaitAsync(TimeSpan.FromSeconds(3), TestContext.Current.CancellationToken); got.ShouldBeTrue("opc-plc FastUInt1 must publish at least one data change within 3s"); int observedCount; lock (observed) observedCount = observed.Count; observedCount.ShouldBeGreaterThan(0); await drv.UnsubscribeAsync(handle, TestContext.Current.CancellationToken); } }