New project tests/ZB.MOM.WW.OtOpcUa.Driver.TwinCAT.IntegrationTests/ with four pieces. TwinCATXarFixture — TCP probe against the ADS-over-TCP port 48898 on the host from TWINCAT_TARGET_HOST env var, requires TWINCAT_TARGET_NETID for the target AmsNetId, optional TWINCAT_TARGET_PORT for runtime 2+ (default 851 = PLC runtime 1). Doesn't own a lifecycle — XAR can't run in Docker because it bypasses the Windows kernel scheduler to hit real-time cycles, so the VM stays operator-managed. Explicit skip reasons surface the setup steps (start VM, set env vars, reactivate trial license) instead of a confusing hang. TwinCATFactAttribute + TwinCATTheoryAttribute — xunit skip gate matching AbServerFactAttribute / OpcPlcCollection patterns. TwinCAT3SmokeTests — three smoke tests through the real AdsTwinCATClient + real ADS over TCP. Driver_reads_seeded_DINT_through_real_ADS reads GVL_Fixture.nCounter, asserts >= 1234 (MAIN increments every cycle so an exact match would race). Driver_write_then_read_round_trip_on_scratch_REAL writes 42.5 to GVL_Fixture.rSetpoint + reads back, catches the ADS write path regression that unit tests can't see. Driver_subscribe_receives_native_ADS_notifications_on_counter_changes validates the #189 native-notification path end-to-end — AddDeviceNotification fires OnDataChange at the PLC cycle boundary, the test observes one firing within 3 s. All three gated on TWINCAT_TARGET_HOST + NETID; skip via TwinCATFactAttribute when unset, verified in this commit with 3 clean [SKIP] results. TwinCatProject/README.md — the tsproj state the smoke tests depend on. GVL_Fixture with nCounter:DINT:=1234 + rSetpoint:REAL:=0.0 + bFlag:BOOL:=TRUE; MAIN program with the single-line ladder `GVL_Fixture.nCounter := GVL_Fixture.nCounter + 1;`; PlcTask cyclic @ 10 ms priority 20; PLC runtime 1 (AMS port 851). Explains why tsproj over the compiled bootproject (text-diffable, rebuildable, no per-install state). Full XAR VM setup walkthrough — Hyper-V Gen 2 VM, TC3 XAE+XAR install, noting the AmsNetId from the tray icon, bilateral route configuration (VM System Manager → Routes + dev box StaticRoutes.xml), project import, Activate Configuration + Run Mode. License-rotation section walks through two options — scheduled TcActivate.exe /reactivate via Task Scheduler (not officially Beckhoff-supported, reportedly works on current builds) or paid runtime license (~$1k one-time per runtime per CPU). Final section shows the exact env-var recipe + dotnet test command on the dev box. docs/drivers/TwinCAT-Test-Fixture.md — flipped TL;DR from "there is no integration fixture" to "scaffolding lives at tests/..., remaining operational work is VM + tsproj + license rotation". "What the fixture is" gains an Integration section describing the XAR VM target. "What it actually covers" gains an Integration subsection listing the three named smoke tests. Follow-up candidates rewritten — the #1 item used to be "TwinCAT 3 runtime on CI" as a speculative option; now it's concrete "XAR VM live-population" with a link to #221 + the project README for the operational walkthrough. License rotation becomes #2 with both automation paths. Key fixture / config files list adds the three new files + the project README. docs/drivers/README.md coverage-map row updated from "no integration fixture" to "XAR-VM integration scaffolding". Solution file picks up the new tests/ZB.MOM.WW.OtOpcUa.Driver.TwinCAT.IntegrationTests entry alongside the existing TwinCAT.Tests. xunit CollectionDefinition added to TwinCATXarFixture after the first build revealed the [Collection("TwinCATXar")] reference on TwinCAT3SmokeTests had no matching registration. Build 0 errors; 3 skip-clean test outcomes verified. #221 stays open as in_progress until the VM + tsproj land. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
136 lines
5.7 KiB
C#
136 lines
5.7 KiB
C#
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Core.Abstractions;
|
|
using ZB.MOM.WW.OtOpcUa.Driver.TwinCAT;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Driver.TwinCAT.IntegrationTests;
|
|
|
|
/// <summary>
|
|
/// End-to-end smoke tests against a live TwinCAT 3 XAR runtime. Skipped via
|
|
/// <see cref="TwinCATFactAttribute"/> when the VM isn't reachable / the AmsNetId
|
|
/// isn't set. Proves the driver's AMS route setup, ADS read/write, symbol browse,
|
|
/// and native <c>AddDeviceNotification</c> subscription all work on the wire —
|
|
/// coverage the <c>FakeTwinCATClient</c>-backed unit suite can only contract-test.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// <para><b>Required VM project state</b> (see <c>TwinCatProject/README.md</c>):</para>
|
|
/// <list type="bullet">
|
|
/// <item>GVL <c>GVL_Fixture</c> with <c>nCounter : DINT</c> (seed <c>1234</c>),
|
|
/// <c>rSetpoint : REAL</c> (scratch; smoke writes + reads), <c>bFlag : BOOL</c>
|
|
/// (seed <c>TRUE</c>).</item>
|
|
/// <item>PLC program <c>MAIN</c> that increments <c>GVL_Fixture.nCounter</c>
|
|
/// every cycle (so the native-notification test can observe monotonic changes
|
|
/// without writing).</item>
|
|
/// </list>
|
|
/// </remarks>
|
|
[Collection("TwinCATXar")]
|
|
[Trait("Category", "Integration")]
|
|
[Trait("Simulator", "TwinCAT-XAR")]
|
|
public sealed class TwinCAT3SmokeTests(TwinCATXarFixture sim)
|
|
{
|
|
[TwinCATFact]
|
|
public async Task Driver_reads_seeded_DINT_through_real_ADS()
|
|
{
|
|
if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason);
|
|
|
|
var options = BuildOptions(sim);
|
|
await using var drv = new TwinCATDriver(options, driverInstanceId: "tc3-smoke-read");
|
|
await drv.InitializeAsync("{}", TestContext.Current.CancellationToken);
|
|
|
|
var snapshots = await drv.ReadAsync(
|
|
["Counter"], TestContext.Current.CancellationToken);
|
|
|
|
snapshots.Count.ShouldBe(1);
|
|
snapshots[0].StatusCode.ShouldBe(0u,
|
|
"ADS read against GVL_Fixture.nCounter must succeed end-to-end");
|
|
// MAIN increments the counter every cycle, so the seed value (1234) is only the
|
|
// minimum we can assert — value grows monotonically.
|
|
Convert.ToInt32(snapshots[0].Value).ShouldBeGreaterThanOrEqualTo(1234);
|
|
}
|
|
|
|
[TwinCATFact]
|
|
public async Task Driver_write_then_read_round_trip_on_scratch_REAL()
|
|
{
|
|
if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason);
|
|
|
|
var options = BuildOptions(sim);
|
|
await using var drv = new TwinCATDriver(options, driverInstanceId: "tc3-smoke-write");
|
|
await drv.InitializeAsync("{}", TestContext.Current.CancellationToken);
|
|
|
|
const float probe = 42.5f;
|
|
var writeResults = await drv.WriteAsync(
|
|
[new WriteRequest("Setpoint", probe)],
|
|
TestContext.Current.CancellationToken);
|
|
writeResults.Count.ShouldBe(1);
|
|
writeResults[0].StatusCode.ShouldBe(0u);
|
|
|
|
var readResults = await drv.ReadAsync(
|
|
["Setpoint"], TestContext.Current.CancellationToken);
|
|
readResults.Count.ShouldBe(1);
|
|
readResults[0].StatusCode.ShouldBe(0u);
|
|
Convert.ToSingle(readResults[0].Value).ShouldBe(probe, tolerance: 0.001f);
|
|
}
|
|
|
|
[TwinCATFact]
|
|
public async Task Driver_subscribe_receives_native_ADS_notifications_on_counter_changes()
|
|
{
|
|
if (sim.SkipReason is not null) Assert.Skip(sim.SkipReason);
|
|
|
|
var options = BuildOptions(sim);
|
|
await using var drv = new TwinCATDriver(options, driverInstanceId: "tc3-smoke-sub");
|
|
await drv.InitializeAsync("{}", TestContext.Current.CancellationToken);
|
|
|
|
var observed = new List<DataChangeEventArgs>();
|
|
var gate = new SemaphoreSlim(0);
|
|
drv.OnDataChange += (_, e) =>
|
|
{
|
|
lock (observed) observed.Add(e);
|
|
gate.Release();
|
|
};
|
|
|
|
var handle = await drv.SubscribeAsync(
|
|
["Counter"], TimeSpan.FromMilliseconds(250),
|
|
TestContext.Current.CancellationToken);
|
|
|
|
// MAIN increments the counter every PLC cycle (default 10 ms task tick).
|
|
// Native ADS notifications fire on cycle boundaries so 3 s is generous for
|
|
// at least one OnDataChange to land.
|
|
var got = await gate.WaitAsync(TimeSpan.FromSeconds(3), TestContext.Current.CancellationToken);
|
|
got.ShouldBeTrue("native ADS notification on GVL_Fixture.nCounter must fire within 3 s of subscribe");
|
|
|
|
int observedCount;
|
|
lock (observed) observedCount = observed.Count;
|
|
observedCount.ShouldBeGreaterThan(0);
|
|
|
|
await drv.UnsubscribeAsync(handle, TestContext.Current.CancellationToken);
|
|
}
|
|
|
|
private static TwinCATDriverOptions BuildOptions(TwinCATXarFixture sim) => new()
|
|
{
|
|
Devices = [
|
|
new TwinCATDeviceOptions(
|
|
HostAddress: $"ads://{sim.TargetNetId}:{sim.AmsPort}",
|
|
DeviceName: "XAR-VM"),
|
|
],
|
|
Tags = [
|
|
new TwinCATTagDefinition(
|
|
Name: "Counter",
|
|
DeviceHostAddress: $"ads://{sim.TargetNetId}:{sim.AmsPort}",
|
|
SymbolPath: "GVL_Fixture.nCounter",
|
|
DataType: TwinCATDataType.DInt),
|
|
new TwinCATTagDefinition(
|
|
Name: "Setpoint",
|
|
DeviceHostAddress: $"ads://{sim.TargetNetId}:{sim.AmsPort}",
|
|
SymbolPath: "GVL_Fixture.rSetpoint",
|
|
DataType: TwinCATDataType.Real,
|
|
Writable: true),
|
|
],
|
|
UseNativeNotifications = true,
|
|
Timeout = TimeSpan.FromSeconds(5),
|
|
// Disable the probe loop — the smoke tests run their own reads; a background
|
|
// probe against GVL_Fixture.nCounter would race with them for the ADS client
|
|
// gate + inject flakiness unrelated to the code under test.
|
|
Probe = new TwinCATProbeOptions { Enabled = false },
|
|
};
|
|
}
|