using Opc.Ua; using Opc.Ua.Server; using ZB.MOM.WW.OtOpcUa.Commons.OpcUa; using ZB.MOM.WW.OtOpcUa.Core.Abstractions; namespace ZB.MOM.WW.OtOpcUa.OpcUaServer; /// /// subclass that wires in the v2 . /// Exposes the live node manager after start so callers (, /// the fused Host's DI binding) can wrap it in a and hand /// it to OpcUaPublishActor. /// public sealed class OtOpcUaSdkServer : StandardServer { private OtOpcUaNodeManager? _otOpcUaNodeManager; /// The custom node manager once StartAsync has called /// . Null until the SDK has bootstrapped. public OtOpcUaNodeManager? NodeManager => _otOpcUaNodeManager; /// /// Wire the reverse-path sink for inbound Part 9 alarm method calls onto the created /// . The host calls this after start with a non-blocking /// mediator.Tell that publishes each onto the /// alarm-commands DistributedPubSub topic. No-op (returns false) when the node /// manager has not been created yet, so the caller can detect a too-early call. /// /// The router invoked by the condition handlers once the AlarmAck /// gate passes; may be null to clear it. /// true when the router was set on a live node manager; false when no node /// manager exists yet. public bool SetAlarmCommandRouter(Action? router) { if (_otOpcUaNodeManager is null) return false; _otOpcUaNodeManager.AlarmCommandRouter = router; return true; } /// /// Wire the reverse-path router for inbound Part 9 Acknowledge of a NATIVE (driver-fed, e.g. Galaxy) /// condition onto the created . The host calls this after start with a /// non-blocking router that Tells a DriverHostActor.RouteNativeAlarmAck into the local /// DriverHostActor, which (Primary-gated) routes the acknowledge to the owning driver child. /// Native conditions are not owned by the scripted-alarm engine, so the node manager branches their /// inbound Acknowledge to this seam instead of the scripted path. /// Passing null clears it. No-op (returns false) when the node manager has not been /// created yet, so the caller can detect a too-early call (mirrors ). /// /// The router invoked by the native condition's Acknowledge handler once the /// AlarmAck gate passes; may be null to clear it. /// true when the router was set on a live node manager; false when no node /// manager exists yet. public bool SetNativeAlarmAckRouter(Action? router) { if (_otOpcUaNodeManager is null) return false; _otOpcUaNodeManager.NativeAlarmAckRouter = router; return true; } /// /// Wire the reverse-path gateway for inbound operator writes to writable equipment-tag nodes onto the /// created . The host calls this after start with an /// ActorNodeWriteGateway whose WriteAsync kicks off a bounded Ask of the local /// DriverHostActor (RouteNodeWrite); the node manager calls it fire-and-forget and uses /// the resolved NodeWriteOutcome to self-correct the node on a failed device write (mirrors /// ). Passing null restores the /// NullOpcUaNodeWriteGateway default (writes unavailable). No-op (returns false) when the /// node manager has not been created yet, so the caller can detect a too-early call. /// /// The gateway invoked by the writable node's OnWriteValue handler once the /// WriteOperate gate passes; may be null to restore the Null default. /// true when the gateway was set on a live node manager; false when no node /// manager exists yet. public bool SetNodeWriteGateway(IOpcUaNodeWriteGateway? gateway) { if (_otOpcUaNodeManager is null) return false; // The NodeWriteGateway setter null-coalesces to the Null default, so a null gateway is intentional // (restores "writes unavailable"); forgive the nullable-in here. _otOpcUaNodeManager.NodeWriteGateway = gateway!; return true; } /// /// Wire the server-side HistoryRead backend (the the node /// manager's HistoryRead overrides block-bridge to) onto the created /// . The host calls this after start with the DI-resolved source — /// the NullHistorianDataSource default (GoodNoData-empty reads) or the configured Wonderware /// read client. Passing null restores the Null default (the property setter null-coalesces), /// i.e. "no historian". No-op (returns false) when the node manager has not been created yet, /// so the caller can detect a too-early call (mirrors ). /// /// The read backend invoked by the node manager's HistoryRead overrides; may be /// null to restore the Null default (GoodNoData-empty reads). /// true when the source was set on a live node manager; false when no node /// manager exists yet. public bool SetHistorianDataSource(IHistorianDataSource? source) { if (_otOpcUaNodeManager is null) return false; // The HistorianDataSource setter null-coalesces to the Null default, so a null source is intentional // (restores GoodNoData-empty reads); forgive the nullable-in here. _otOpcUaNodeManager.HistorianDataSource = source!; return true; } /// protected override MasterNodeManager CreateMasterNodeManager( IServerInternal server, ApplicationConfiguration configuration) { _otOpcUaNodeManager = new OtOpcUaNodeManager(server, configuration); return new MasterNodeManager(server, configuration, dynamicNamespaceUri: null, _otOpcUaNodeManager); } }