2982cc4bb5
v2-ci / build (pull_request) Failing after 39s
v2-ci / unit-tests (tests/Core/ZB.MOM.WW.OtOpcUa.Cluster.Tests) (pull_request) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.ControlPlane.Tests) (pull_request) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.OpcUaServer.Tests) (pull_request) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.Runtime.Tests) (pull_request) Has been skipped
v2-ci / unit-tests (tests/Server/ZB.MOM.WW.OtOpcUa.Security.Tests) (pull_request) Has been skipped
v2-ci / integration (tests/Server/ZB.MOM.WW.OtOpcUa.Host.IntegrationTests) (pull_request) Has been skipped
v2-ci / integration (tests/Server/ZB.MOM.WW.OtOpcUa.OpcUaServer.IntegrationTests) (pull_request) Has been skipped
The ContinuousHistorizationRecorder was spawned with an EMPTY historized-ref set, so it registered interest in nothing and historized nothing. This feeds it the currently-historized tag refs on every address-space deploy/redeploy so its DependencyMuxActor interest converges to exactly the historized set (the same refs the EnsureTags provisioning hook resolves: override-or-FullName). Design — delta convergence (the plan is a pure DIFF): - New seam IHistorizedTagSubscriptionSink (Core.Abstractions/Historian) with a Null no-op singleton, mirroring how IHistorianProvisioning decouples the T15 hook. AddressSpaceApplier gains a DEFAULTED ctor param (Null sink) so all ~80 existing call sites + the production site compile unchanged. - Apply() only ever sees a plan diff (an incremental/surgical apply carries a delta, not the full set), so the applier feeds an add/remove DELTA computed from AddedEquipmentTags / RemovedEquipmentTags / ChangedEquipmentTags. The recorder keeps the full set and re-registers it. The feed is a single non-blocking Tell behind the sink, wrapped in try/catch so a faulting feed never blocks or breaks a deploy (same discipline as the provisioning hook). - Recorder.UpdateHistorizedRefs(added, removed) converges the tracked set, then — only when it actually changed — sends ONE RegisterInterest with the full set (the mux's RegisterInterest is a full-REPLACE) or one UnregisterInterest when it drains to empty (the mux has no per-ref unregister). An unchanged delta is a no-op (no mux churn). - DI: the recorder is now spawned BEFORE the applier so the adapter (ActorHistorizedTagSubscriptionSink) can wrap its IActorRef; the Null sink is used when continuous historization is off/unwired. Tests: recorder convergence (add-from-empty, add+remove converge, idempotent, drain-to-empty unregisters); applier feeds resolved added refs, removed+renamed deltas, and survives a throwing sink. Build clean (0 warnings on touched projects); Runtime/OpcUaServer/Gateway/AdminUI suites green. Claude-Session: https://claude.ai/code/session_012SDSQ3AcaXqPcBtDESBRii
263 lines
13 KiB
C#
263 lines
13 KiB
C#
using Microsoft.Extensions.Logging.Abstractions;
|
|
using Shouldly;
|
|
using Xunit;
|
|
using ZB.MOM.WW.OtOpcUa.Commons.OpcUa;
|
|
using ZB.MOM.WW.OtOpcUa.Core.Abstractions;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.OpcUaServer.Tests;
|
|
|
|
/// <summary>
|
|
/// T15 — verifies the non-blocking historian-provisioning hook in
|
|
/// <see cref="AddressSpaceApplier.Apply"/>. The hook fires AFTER the address-space work and
|
|
/// dispatches <see cref="IHistorianProvisioning.EnsureTagsAsync"/> fire-and-forget, so a slow or
|
|
/// throwing provisioner can never block or break a deploy on the OPC UA publish actor's pinned
|
|
/// thread.
|
|
/// </summary>
|
|
public sealed class AddressSpaceApplierProvisioningTests
|
|
{
|
|
/// <summary>Capturing <see cref="IHistorianProvisioning"/> double. Records the requests it was
|
|
/// handed and signals a <see cref="TaskCompletionSource"/> when invoked, so a test can await the
|
|
/// fire-and-forget dispatch deterministically (never poll/sleep). A <see cref="Throw"/> flag
|
|
/// simulates a synchronous provisioner fault.</summary>
|
|
private sealed class CapturingProvisioner : IHistorianProvisioning
|
|
{
|
|
private readonly TaskCompletionSource _called = new(TaskCreationOptions.RunContinuationsAsynchronously);
|
|
|
|
/// <summary>Gets the requests the hook handed to <see cref="EnsureTagsAsync"/>.</summary>
|
|
public List<HistorianTagProvisionRequest> Seen { get; } = new();
|
|
|
|
/// <summary>When true, <see cref="EnsureTagsAsync"/> throws synchronously (a fault before any await).</summary>
|
|
public bool Throw { get; init; }
|
|
|
|
/// <summary>Completes once <see cref="EnsureTagsAsync"/> has been invoked.</summary>
|
|
public Task Called => _called.Task;
|
|
|
|
/// <inheritdoc />
|
|
public Task<HistorianProvisionResult> EnsureTagsAsync(
|
|
IReadOnlyList<HistorianTagProvisionRequest> requests, CancellationToken ct)
|
|
{
|
|
if (Throw)
|
|
{
|
|
_called.TrySetResult();
|
|
throw new InvalidOperationException("boom");
|
|
}
|
|
|
|
Seen.AddRange(requests);
|
|
_called.TrySetResult();
|
|
return Task.FromResult(new HistorianProvisionResult(requests.Count, requests.Count, 0, 0));
|
|
}
|
|
}
|
|
|
|
/// <summary>The hook provisions ONLY historized added tags, with the resolved historian name
|
|
/// (override when set, else the driver-side FullName).</summary>
|
|
[Fact]
|
|
public async Task Apply_provisions_only_historized_added_tags()
|
|
{
|
|
var prov = new CapturingProvisioner();
|
|
var applier = new AddressSpaceApplier(NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance, prov);
|
|
|
|
// Leaf display name "Temp"; historian override "Pump1.Temp".
|
|
var plan = PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Temp", historianName: "Pump1.Temp", dataType: "Float32"),
|
|
NonHistorizedTag(displayName: "Run", dataType: "Boolean"));
|
|
|
|
var outcome = applier.Apply(plan);
|
|
|
|
outcome.RebuildCalled.ShouldBeTrue();
|
|
|
|
// Fire-and-forget: await the capturing double's signal so the assertion is deterministic.
|
|
await prov.Called.WaitAsync(TimeSpan.FromSeconds(5), TestContext.Current.CancellationToken);
|
|
prov.Seen.Count.ShouldBe(1);
|
|
prov.Seen[0].TagName.ShouldBe("Pump1.Temp"); // resolved historian name (override)
|
|
prov.Seen[0].DataType.ShouldBe(DriverDataType.Float32);
|
|
prov.Seen[0].Description.ShouldBe("Temp"); // leaf display name
|
|
}
|
|
|
|
/// <summary>A null/blank historian-name override resolves to the driver-side FullName — mirroring
|
|
/// the materialiser's resolution exactly.</summary>
|
|
[Fact]
|
|
public async Task Apply_resolves_historian_name_from_fullname_when_override_blank()
|
|
{
|
|
var prov = new CapturingProvisioner();
|
|
var applier = new AddressSpaceApplier(NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance, prov);
|
|
|
|
// IsHistorized but no override → historian name falls back to FullName ("40001").
|
|
var plan = PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Speed", historianName: null, dataType: "Int32", fullName: "40001"));
|
|
|
|
applier.Apply(plan);
|
|
|
|
await prov.Called.WaitAsync(TimeSpan.FromSeconds(5), TestContext.Current.CancellationToken);
|
|
prov.Seen.Count.ShouldBe(1);
|
|
prov.Seen[0].TagName.ShouldBe("40001");
|
|
prov.Seen[0].DataType.ShouldBe(DriverDataType.Int32);
|
|
}
|
|
|
|
/// <summary>A synchronously-throwing provisioner must NOT block or break the publish: the
|
|
/// synchronous <see cref="AddressSpaceApplier.Apply"/> still completes its address-space work and
|
|
/// returns its normal outcome.</summary>
|
|
[Fact]
|
|
public void Provisioner_throw_does_not_block_publish()
|
|
{
|
|
var applier = new AddressSpaceApplier(
|
|
NullOpcUaAddressSpaceSink.Instance,
|
|
NullLogger<AddressSpaceApplier>.Instance,
|
|
new CapturingProvisioner { Throw = true });
|
|
|
|
var outcome = applier.Apply(PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Temp", historianName: "Pump1.Temp", dataType: "Float32")));
|
|
|
|
outcome.RebuildCalled.ShouldBeTrue(); // address-space work still completed
|
|
}
|
|
|
|
/// <summary>The default ctor (no provisioner) binds the no-op <see cref="NullHistorianProvisioning"/>
|
|
/// and never faults a deploy — preserving every existing call site.</summary>
|
|
[Fact]
|
|
public void Default_ctor_uses_null_provisioning_and_does_not_throw()
|
|
{
|
|
var applier = new AddressSpaceApplier(NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance);
|
|
|
|
var outcome = applier.Apply(PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Temp", historianName: "Pump1.Temp", dataType: "Float32")));
|
|
|
|
outcome.RebuildCalled.ShouldBeTrue();
|
|
}
|
|
|
|
/// <summary>An added historized tag whose DataType string is not a <see cref="DriverDataType"/> is
|
|
/// skipped (no request) — the hook never throws on an unparseable type.</summary>
|
|
[Fact]
|
|
public async Task Apply_skips_added_tag_with_unparseable_datatype()
|
|
{
|
|
var prov = new CapturingProvisioner();
|
|
var applier = new AddressSpaceApplier(NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance, prov);
|
|
|
|
// "Float" is NOT a DriverDataType member (the members are Float32/Float64); it must be skipped.
|
|
var plan = PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Bad", historianName: "Pump1.Bad", dataType: "Float"),
|
|
HistorizedTag(displayName: "Good", historianName: "Pump1.Good", dataType: "Float32"));
|
|
|
|
applier.Apply(plan);
|
|
|
|
await prov.Called.WaitAsync(TimeSpan.FromSeconds(5), TestContext.Current.CancellationToken);
|
|
prov.Seen.Count.ShouldBe(1);
|
|
prov.Seen[0].TagName.ShouldBe("Pump1.Good");
|
|
}
|
|
|
|
/// <summary>Capturing <see cref="IHistorizedTagSubscriptionSink"/> double. Records the add/remove
|
|
/// ref deltas the applier feeds it. A <see cref="Throw"/> flag simulates a faulting feed.</summary>
|
|
private sealed class CapturingSubscriptionSink : IHistorizedTagSubscriptionSink
|
|
{
|
|
/// <summary>Refs the applier fed as ADDED.</summary>
|
|
public List<string> Added { get; } = new();
|
|
|
|
/// <summary>Refs the applier fed as REMOVED.</summary>
|
|
public List<string> Removed { get; } = new();
|
|
|
|
/// <summary>When true, <see cref="UpdateHistorizedRefs"/> throws synchronously.</summary>
|
|
public bool Throw { get; init; }
|
|
|
|
/// <inheritdoc />
|
|
public void UpdateHistorizedRefs(IReadOnlyList<string> added, IReadOnlyList<string> removed)
|
|
{
|
|
if (Throw) throw new InvalidOperationException("boom");
|
|
Added.AddRange(added);
|
|
Removed.AddRange(removed);
|
|
}
|
|
}
|
|
|
|
/// <summary>The feed pushes ONLY historized added refs, resolved (override-or-FullName) exactly like
|
|
/// the provisioning hook — non-historized tags never reach the recorder.</summary>
|
|
[Fact]
|
|
public void Apply_feeds_historized_added_refs_to_the_subscription_sink()
|
|
{
|
|
var sink = new CapturingSubscriptionSink();
|
|
var applier = new AddressSpaceApplier(
|
|
NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance,
|
|
historizedSubscriptions: sink);
|
|
|
|
var plan = PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Temp", historianName: "Pump1.Temp", dataType: "Float32"),
|
|
HistorizedTag(displayName: "Speed", historianName: null, dataType: "Int32", fullName: "40001"),
|
|
NonHistorizedTag(displayName: "Run", dataType: "Boolean"));
|
|
|
|
applier.Apply(plan);
|
|
|
|
sink.Added.ShouldBe(new[] { "Pump1.Temp", "40001" }, ignoreOrder: true); // override + FullName fallback
|
|
sink.Removed.ShouldBeEmpty();
|
|
}
|
|
|
|
/// <summary>Removed historized tags are fed as REMOVED refs; a changed tag whose historian override is
|
|
/// renamed feeds the old ref removed + the new ref added (the recorder converges the full set).</summary>
|
|
[Fact]
|
|
public void Apply_feeds_removed_and_renamed_historized_refs()
|
|
{
|
|
var sink = new CapturingSubscriptionSink();
|
|
var applier = new AddressSpaceApplier(
|
|
NullOpcUaAddressSpaceSink.Instance, NullLogger<AddressSpaceApplier>.Instance,
|
|
historizedSubscriptions: sink);
|
|
|
|
var removedTag = HistorizedTag(displayName: "Old", historianName: "Pump1.Old", dataType: "Float32");
|
|
// Same TagId ("tag-T"), historian override renamed A → B (both historized) → remove A, add B.
|
|
var prev = HistorizedTag(displayName: "T", historianName: "Pump1.A", dataType: "Float32");
|
|
var cur = HistorizedTag(displayName: "T", historianName: "Pump1.B", dataType: "Float32");
|
|
|
|
var plan = new AddressSpacePlan(
|
|
AddedEquipment: Array.Empty<EquipmentNode>(),
|
|
RemovedEquipment: Array.Empty<EquipmentNode>(),
|
|
ChangedEquipment: Array.Empty<AddressSpacePlan.EquipmentDelta>(),
|
|
AddedDrivers: Array.Empty<DriverInstancePlan>(),
|
|
RemovedDrivers: Array.Empty<DriverInstancePlan>(),
|
|
ChangedDrivers: Array.Empty<AddressSpacePlan.DriverDelta>(),
|
|
AddedAlarms: Array.Empty<ScriptedAlarmPlan>(),
|
|
RemovedAlarms: Array.Empty<ScriptedAlarmPlan>(),
|
|
ChangedAlarms: Array.Empty<AddressSpacePlan.AlarmDelta>())
|
|
{
|
|
RemovedEquipmentTags = new[] { removedTag },
|
|
ChangedEquipmentTags = new[] { new AddressSpacePlan.EquipmentTagDelta(prev, cur) },
|
|
};
|
|
|
|
applier.Apply(plan);
|
|
|
|
sink.Added.ShouldBe(new[] { "Pump1.B" }, ignoreOrder: true);
|
|
sink.Removed.ShouldBe(new[] { "Pump1.Old", "Pump1.A" }, ignoreOrder: true);
|
|
}
|
|
|
|
/// <summary>A synchronously-throwing subscription sink must NOT block or break the publish — the
|
|
/// address-space work still completes and <see cref="AddressSpaceApplier.Apply"/> returns its outcome.</summary>
|
|
[Fact]
|
|
public void Subscription_sink_throw_does_not_block_publish()
|
|
{
|
|
var applier = new AddressSpaceApplier(
|
|
NullOpcUaAddressSpaceSink.Instance,
|
|
NullLogger<AddressSpaceApplier>.Instance,
|
|
historizedSubscriptions: new CapturingSubscriptionSink { Throw = true });
|
|
|
|
var outcome = applier.Apply(PlanWithAddedTags(
|
|
HistorizedTag(displayName: "Temp", historianName: "Pump1.Temp", dataType: "Float32")));
|
|
|
|
outcome.RebuildCalled.ShouldBeTrue(); // address-space work still completed
|
|
}
|
|
|
|
private static EquipmentTagPlan HistorizedTag(string displayName, string? historianName, string dataType, string fullName = "ref")
|
|
=> new("tag-" + displayName, "eq-1", "drv", FolderPath: "", Name: displayName, DataType: dataType, FullName: fullName,
|
|
Writable: false, Alarm: null, IsHistorized: true, HistorianTagname: historianName);
|
|
|
|
private static EquipmentTagPlan NonHistorizedTag(string displayName, string dataType)
|
|
=> new("tag-" + displayName, "eq-1", "drv", FolderPath: "", Name: displayName, DataType: dataType, FullName: "ref",
|
|
Writable: false, Alarm: null, IsHistorized: false, HistorianTagname: null);
|
|
|
|
private static AddressSpacePlan PlanWithAddedTags(params EquipmentTagPlan[] tags) => new(
|
|
AddedEquipment: Array.Empty<EquipmentNode>(),
|
|
RemovedEquipment: Array.Empty<EquipmentNode>(),
|
|
ChangedEquipment: Array.Empty<AddressSpacePlan.EquipmentDelta>(),
|
|
AddedDrivers: Array.Empty<DriverInstancePlan>(),
|
|
RemovedDrivers: Array.Empty<DriverInstancePlan>(),
|
|
ChangedDrivers: Array.Empty<AddressSpacePlan.DriverDelta>(),
|
|
AddedAlarms: Array.Empty<ScriptedAlarmPlan>(),
|
|
RemovedAlarms: Array.Empty<ScriptedAlarmPlan>(),
|
|
ChangedAlarms: Array.Empty<AddressSpacePlan.AlarmDelta>())
|
|
{
|
|
AddedEquipmentTags = tags,
|
|
};
|
|
}
|