ac5db0a9f8
Important 1: ShelveAlarmAsync Timed branch now multiplies shelvingTimeSeconds × 1000.0
before passing to CallMethodAsync — OPC UA Part 9 TimedShelve ShelvingTime is a Duration
in milliseconds, not seconds. IOpcUaClientService XML doc and ShelveCommand --duration
description updated to document the seconds-in / ms-out contract.
Important 2: ShelveAlarmAsync builds shelvingStateNodeId with the same
EndsWith(".ShelvingState") guard already used by the .Condition suffix in
AcknowledgeAlarmAsync / ConfirmAlarmAsync, preventing double-append.
Important 3: Add 6 service-layer tests to OpcUaClientServiceTests —
ConfirmAlarmAsync_OnSuccess_ReturnsGood
ConfirmAlarmAsync_OnServiceResultException_ReturnsBadStatusCode
ShelveAlarmAsync_OneShot_CallsMethodWithNoArgs
ShelveAlarmAsync_Timed_PassesDurationInMilliseconds (regression guard for Important 1)
ShelveAlarmAsync_Unshelve_CallsMethodWithNoArgs
ShelveAlarmAsync_OnServiceResultException_ReturnsBadStatusCode
FakeSessionAdapter extended with CallMethodInputArgs list to record per-call input
arguments so the Timed test can assert the ms value.
Minor 4: ShelveCommand output changed from "Shelve (OneShot) successful" to
"{shelveKind} successful/failed" so Unshelve reads "Unshelve successful: …".
Minor 6: ShelveCommand --duration description updated to "(must be > 0; in seconds,
converted to milliseconds for the OPC UA call; required for --kind Timed)".
182 lines
10 KiB
C#
182 lines
10 KiB
C#
using Opc.Ua;
|
|
using ZB.MOM.WW.OtOpcUa.Client.Shared.Models;
|
|
using BrowseResult = ZB.MOM.WW.OtOpcUa.Client.Shared.Models.BrowseResult;
|
|
|
|
namespace ZB.MOM.WW.OtOpcUa.Client.Shared;
|
|
|
|
/// <summary>
|
|
/// Shared OPC UA client service contract for CLI and UI consumers.
|
|
/// </summary>
|
|
public interface IOpcUaClientService : IDisposable
|
|
{
|
|
/// <summary>
|
|
/// Gets a value indicating whether the client is currently connected to an OPC UA endpoint.
|
|
/// </summary>
|
|
bool IsConnected { get; }
|
|
|
|
/// <summary>
|
|
/// Gets the current connection metadata shown to CLI and UI operators after a successful connect or failover.
|
|
/// </summary>
|
|
ConnectionInfo? CurrentConnectionInfo { get; }
|
|
|
|
/// <summary>
|
|
/// Connects the client to the configured OPC UA endpoint set, including failover-capable endpoints when provided.
|
|
/// </summary>
|
|
/// <param name="settings">The endpoint, security, and authentication settings used to establish the session.</param>
|
|
/// <param name="ct">The cancellation token that aborts the connect workflow.</param>
|
|
Task<ConnectionInfo> ConnectAsync(ConnectionSettings settings, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Disconnects from the active OPC UA endpoint and tears down subscriptions owned by the client.
|
|
/// </summary>
|
|
/// <param name="ct">The cancellation token that aborts disconnect cleanup.</param>
|
|
Task DisconnectAsync(CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Reads the current value of an OPC UA node.
|
|
/// </summary>
|
|
/// <param name="nodeId">The node whose value should be retrieved.</param>
|
|
/// <param name="ct">The cancellation token that aborts the read request.</param>
|
|
Task<DataValue> ReadValueAsync(NodeId nodeId, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Writes an operator-supplied value to an OPC UA node after applying client-side type conversion when needed.
|
|
/// </summary>
|
|
/// <param name="nodeId">The node whose value should be updated.</param>
|
|
/// <param name="value">The raw value supplied by the CLI or UI workflow.</param>
|
|
/// <param name="ct">The cancellation token that aborts the write request.</param>
|
|
Task<StatusCode> WriteValueAsync(NodeId nodeId, object value, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Browses the children of a node so the client can build an address-space tree for operators.
|
|
/// </summary>
|
|
/// <param name="parentNodeId">The node to browse, or <see cref="ObjectIds.ObjectsFolder"/> when omitted.</param>
|
|
/// <param name="ct">The cancellation token that aborts the browse request.</param>
|
|
Task<IReadOnlyList<BrowseResult>> BrowseAsync(NodeId? parentNodeId = null, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Subscribes to live data changes for a node.
|
|
/// </summary>
|
|
/// <param name="nodeId">The node whose value changes should be monitored.</param>
|
|
/// <param name="intervalMs">The monitored-item sampling and publishing interval in milliseconds.</param>
|
|
/// <param name="ct">The cancellation token that aborts subscription creation.</param>
|
|
Task SubscribeAsync(NodeId nodeId, int intervalMs = 1000, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Removes a previously created live-data subscription for a node.
|
|
/// </summary>
|
|
/// <param name="nodeId">The node whose live-data subscription should be removed.</param>
|
|
/// <param name="ct">The cancellation token that aborts the unsubscribe request.</param>
|
|
Task UnsubscribeAsync(NodeId nodeId, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Subscribes to OPC UA alarm and condition events for a source node or the server root.
|
|
/// </summary>
|
|
/// <param name="sourceNodeId">The event source to monitor, or the server object when omitted.</param>
|
|
/// <param name="intervalMs">The publishing interval in milliseconds for the alarm subscription.</param>
|
|
/// <param name="ct">The cancellation token that aborts alarm subscription creation.</param>
|
|
Task SubscribeAlarmsAsync(NodeId? sourceNodeId = null, int intervalMs = 1000, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Removes the active alarm subscription.
|
|
/// </summary>
|
|
/// <param name="ct">The cancellation token that aborts alarm subscription cleanup.</param>
|
|
Task UnsubscribeAlarmsAsync(CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Requests retained alarm conditions again so a client can repopulate its alarm list after reconnecting.
|
|
/// </summary>
|
|
/// <param name="ct">The cancellation token that aborts the condition refresh request.</param>
|
|
Task RequestConditionRefreshAsync(CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Acknowledges an active condition using the event identifier returned by an alarm notification.
|
|
/// </summary>
|
|
/// <param name="conditionNodeId">The condition node associated with the alarm event being acknowledged.</param>
|
|
/// <param name="eventId">The event identifier returned by the OPC UA server for the alarm event.</param>
|
|
/// <param name="comment">The operator acknowledgment comment to write with the method call.</param>
|
|
/// <param name="ct">The cancellation token that aborts the acknowledgment request.</param>
|
|
/// <returns>
|
|
/// <see cref="StatusCodes.Good"/> on success, or the server's bad <see cref="StatusCode"/>
|
|
/// (from the underlying <see cref="ServiceResultException"/>) when the acknowledge call
|
|
/// returns a bad result. Other transport-level failures still surface as exceptions.
|
|
/// </returns>
|
|
Task<StatusCode> AcknowledgeAlarmAsync(string conditionNodeId, byte[] eventId, string comment, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Confirms an acknowledged condition (Part 9 two-stage acknowledgment) using the event identifier returned by an alarm notification.
|
|
/// </summary>
|
|
/// <param name="conditionNodeId">The condition node associated with the alarm event being confirmed.</param>
|
|
/// <param name="eventId">The event identifier returned by the OPC UA server for the alarm event.</param>
|
|
/// <param name="comment">The operator confirmation comment to write with the method call.</param>
|
|
/// <param name="ct">The cancellation token that aborts the confirmation request.</param>
|
|
/// <returns>
|
|
/// <see cref="StatusCodes.Good"/> on success, or the server's bad <see cref="StatusCode"/>
|
|
/// (from the underlying <see cref="ServiceResultException"/>) when the confirm call
|
|
/// returns a bad result. Other transport-level failures still surface as exceptions.
|
|
/// </returns>
|
|
Task<StatusCode> ConfirmAlarmAsync(string conditionNodeId, byte[] eventId, string comment, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Shelves or unshelves an active alarm condition (OPC UA Part 9 ShelvedStateMachine).
|
|
/// </summary>
|
|
/// <param name="conditionNodeId">The condition node associated with the alarm being shelved.</param>
|
|
/// <param name="kind">The shelve operation: <c>OneShot</c>, <c>Timed</c>, or <c>Unshelve</c>.</param>
|
|
/// <param name="shelvingTimeSeconds">
|
|
/// For <c>Timed</c> shelving: the shelving duration in seconds (must be > 0).
|
|
/// This value is converted to milliseconds when passed to the OPC UA <c>TimedShelve</c> method
|
|
/// (<c>Duration</c> is milliseconds per OPC UA Part 9). Ignored for <c>OneShot</c> and <c>Unshelve</c>.
|
|
/// </param>
|
|
/// <param name="ct">The cancellation token that aborts the shelve request.</param>
|
|
/// <returns>
|
|
/// <see cref="StatusCodes.Good"/> on success, or the server's bad <see cref="StatusCode"/>
|
|
/// (from the underlying <see cref="ServiceResultException"/>) when the shelve call
|
|
/// returns a bad result. Other transport-level failures still surface as exceptions.
|
|
/// </returns>
|
|
Task<StatusCode> ShelveAlarmAsync(string conditionNodeId, ShelveKind kind, double shelvingTimeSeconds = 0, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Reads raw historical samples for a historized node.
|
|
/// </summary>
|
|
/// <param name="nodeId">The historized node whose samples should be read.</param>
|
|
/// <param name="startTime">The inclusive start of the requested history range.</param>
|
|
/// <param name="endTime">The inclusive end of the requested history range.</param>
|
|
/// <param name="maxValues">The maximum number of raw values to return.</param>
|
|
/// <param name="ct">The cancellation token that aborts the history read.</param>
|
|
Task<IReadOnlyList<DataValue>> HistoryReadRawAsync(NodeId nodeId, DateTime startTime, DateTime endTime,
|
|
int maxValues = 1000, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Reads aggregate historical values for a historized node using an OPC UA aggregate function.
|
|
/// </summary>
|
|
/// <param name="nodeId">The historized node whose processed values should be read.</param>
|
|
/// <param name="startTime">The inclusive start of the requested processed-history range.</param>
|
|
/// <param name="endTime">The inclusive end of the requested processed-history range.</param>
|
|
/// <param name="aggregate">The aggregate function the operator selected for processed history.</param>
|
|
/// <param name="intervalMs">The processing interval, in milliseconds, for each aggregate bucket.</param>
|
|
/// <param name="ct">The cancellation token that aborts the processed history request.</param>
|
|
Task<IReadOnlyList<DataValue>> HistoryReadAggregateAsync(NodeId nodeId, DateTime startTime, DateTime endTime,
|
|
AggregateType aggregate, double intervalMs = 3600000, CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Reads redundancy status data such as redundancy mode, service level, and partner endpoint URIs.
|
|
/// </summary>
|
|
/// <param name="ct">The cancellation token that aborts redundancy inspection.</param>
|
|
Task<RedundancyInfo> GetRedundancyInfoAsync(CancellationToken ct = default);
|
|
|
|
/// <summary>
|
|
/// Raised when a subscribed node produces a new live data value.
|
|
/// </summary>
|
|
event EventHandler<DataChangedEventArgs>? DataChanged;
|
|
|
|
/// <summary>
|
|
/// Raised when an alarm or condition event is received from the server.
|
|
/// </summary>
|
|
event EventHandler<AlarmEventArgs>? AlarmEvent;
|
|
|
|
/// <summary>
|
|
/// Raised when the client changes connection state during connect, disconnect, or failover.
|
|
/// </summary>
|
|
event EventHandler<ConnectionStateChangedEventArgs>? ConnectionStateChanged;
|
|
}
|