Modbus — deletes tests/.../Modbus.IntegrationTests/Pymodbus/ (serve.ps1, standard.json, dl205.json, mitsubishi.json, s7_1500.json, README.md). Profile JSONs live only under Docker/profiles/ now. Docker/README.md loses its "Native-Python fallback" section; docs/drivers/Modbus-Test-Fixture.md "What the fixture is" bullet flipped from "primary launcher is Docker, native fallback under Pymodbus/" to "Docker is the only supported launch path". S7 — deletes tests/.../S7.IntegrationTests/PythonSnap7/ (server.py, s7_1500.json, serve.ps1, README.md). Docker/README.md loses "Native-Python fallback"; docs/drivers/S7-Test-Fixture.md updated to match. AB CIP — the biggest simplification because the native-binary spawn had the most code. AbServerFixture.cs rewrites: drops Process management (no more Process _proc + Kill/WaitForExit), drops LocateBinary() PATH lookup, drops the IAsyncLifetime initialize-spawns-server behavior. Fixture is now a thin TCP probe against localhost:44818 (or AB_SERVER_ENDPOINT override) — same shape as Snap7ServerFixture / ModbusSimulatorFixture / OpcPlcFixture. IsServerAvailable() simplifies to a single 500 ms probe. AbServerProfile.cs drops AbServerPlcArg + SeedTags + BuildCliArgs + ToCliSpec + the entire AbServerSeedTag record — the compose file is the canonical source of truth for which tags + which --plc mode each family gets; the profile record now carries just Family + ComposeProfile (matches the docker-compose service key) + Notes. KnownProfiles.ForFamily + .All stay for tests that iterate families. AbServerProfileTests.cs rewrites to match: drops BuildCliArgs_* + ToCliSpec_* + SeedTags_* tests; keeps the family-coverage contract tests + verifies the ComposeProfile strings match compose-file service names (a typo in either surfaces as a unit-test failure, not a silent "wrong family booted" at runtime). Docker/README.md loses "Native-binary fallback" section; docs/drivers/AbServer-Test-Fixture.md "What the fixture is" flipped to Docker-only with clearer skip rules. dev-environment.md §Docker fixtures — the "Native fallbacks" subsection goes away; replaced with a one-line note that Docker is the only supported path for these four fixtures + a fresh clone needs Docker Desktop and nothing else. Verified: whole-solution build 0 errors, AB CIP profile unit tests 6/6, AB CIP Docker smoke 4/4 (all family theory rows), S7 Docker smoke 3/3. Container lifecycle clean. The deleted native code surface was already redundant — every fixture the native paths served is now covered by Docker; keeping them invited drift between the two paths (the original AB CIP native profile had three undetected bugs per the #162 commit message: case-sensitive --plc, bracket tag notation, --path=1,0 requirement — noise the Docker path now avoids by never running the buggy code). Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
56 lines
2.7 KiB
C#
56 lines
2.7 KiB
C#
using ZB.MOM.WW.OtOpcUa.Driver.AbCip;
|
||
|
||
namespace ZB.MOM.WW.OtOpcUa.Driver.AbCip.IntegrationTests;
|
||
|
||
/// <summary>
|
||
/// Per-family marker for the <c>ab_server</c> Docker compose profile a given test
|
||
/// targets. The compose file (<c>Docker/docker-compose.yml</c>) is the canonical
|
||
/// source of truth for which tags a family seeds + which <c>--plc</c> mode the
|
||
/// simulator boots in; this record just ties a family enum to operator-facing
|
||
/// notes so fixture + test code can filter / branch by family.
|
||
/// </summary>
|
||
/// <param name="Family">OtOpcUa driver family this profile targets.</param>
|
||
/// <param name="ComposeProfile">The <c>docker compose --profile</c> name that brings
|
||
/// this family's ab_server up. Matches the service key in the compose file.</param>
|
||
/// <param name="Notes">Operator-facing description of coverage + any quirks.</param>
|
||
public sealed record AbServerProfile(
|
||
AbCipPlcFamily Family,
|
||
string ComposeProfile,
|
||
string Notes)
|
||
{
|
||
/// <summary>Default ab_server port — matches the compose-file port-map + the
|
||
/// CIP / EtherNet/IP standard.</summary>
|
||
public const int DefaultPort = 44818;
|
||
}
|
||
|
||
/// <summary>Canonical profiles covering every AB CIP family shipped in PRs 9–12.</summary>
|
||
public static class KnownProfiles
|
||
{
|
||
public static readonly AbServerProfile ControlLogix = new(
|
||
Family: AbCipPlcFamily.ControlLogix,
|
||
ComposeProfile: "controllogix",
|
||
Notes: "Widest-coverage profile — PR 9 baseline. UDTs unit-tested via golden Template Object buffers; ab_server lacks full UDT emulation.");
|
||
|
||
public static readonly AbServerProfile CompactLogix = new(
|
||
Family: AbCipPlcFamily.CompactLogix,
|
||
ComposeProfile: "compactlogix",
|
||
Notes: "ab_server doesn't enforce the narrower ConnectionSize; driver-side profile caps it per PR 10.");
|
||
|
||
public static readonly AbServerProfile Micro800 = new(
|
||
Family: AbCipPlcFamily.Micro800,
|
||
ComposeProfile: "micro800",
|
||
Notes: "--plc=Micro800 mode (unconnected-only, empty path). Driver-side enforcement verified in the unit suite.");
|
||
|
||
public static readonly AbServerProfile GuardLogix = new(
|
||
Family: AbCipPlcFamily.GuardLogix,
|
||
ComposeProfile: "guardlogix",
|
||
Notes: "ab_server has no safety subsystem — _S-suffixed seed tag triggers driver-side ViewOnly classification only.");
|
||
|
||
public static IReadOnlyList<AbServerProfile> All { get; } =
|
||
[ControlLogix, CompactLogix, Micro800, GuardLogix];
|
||
|
||
public static AbServerProfile ForFamily(AbCipPlcFamily family) =>
|
||
All.FirstOrDefault(p => p.Family == family)
|
||
?? throw new ArgumentOutOfRangeException(nameof(family), family, "No integration profile for this family.");
|
||
}
|