|
29 | 29 |
|
30 | 30 | using NUnit.Framework; |
31 | 31 | using Opc.Ua.Tests; |
| 32 | +using System; |
32 | 33 |
|
33 | 34 | namespace Opc.Ua.Core.Tests.Stack.State |
34 | 35 | { |
@@ -64,6 +65,196 @@ protected void OneTimeTearDown() |
64 | 65 | Utils.SilentDispose(m_context); |
65 | 66 | } |
66 | 67 |
|
| 68 | + /// <summary> |
| 69 | + /// Test that SetEnableState updates the timestamp and clears change masks. |
| 70 | + /// </summary> |
| 71 | + [Test] |
| 72 | + public void SetEnableStateUpdatesTimestampAndClearsChangeMasks() |
| 73 | + { |
| 74 | + var condition = new ConditionState(null); |
| 75 | + condition.Create(m_context, new NodeId(1), "Condition", null, true); |
| 76 | + |
| 77 | + // Set initial state |
| 78 | + var beforeTime = DateTime.UtcNow; |
| 79 | + condition.SetEnableState(m_context, true); |
| 80 | + var afterTime = DateTime.UtcNow; |
| 81 | + |
| 82 | + // Verify timestamp is updated |
| 83 | + Assert.That(condition.EnabledState.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 84 | + Assert.That(condition.EnabledState.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 85 | + |
| 86 | + // Verify change masks are cleared (all should be None) |
| 87 | + Assert.That(condition.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 88 | + Assert.That(condition.EnabledState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 89 | + } |
| 90 | + |
| 91 | + /// <summary> |
| 92 | + /// Test that SetSeverity updates the timestamp and clears change masks. |
| 93 | + /// </summary> |
| 94 | + [Test] |
| 95 | + public void SetSeverityUpdatesTimestampAndClearsChangeMasks() |
| 96 | + { |
| 97 | + var condition = new ConditionState(null); |
| 98 | + condition.Create(m_context, new NodeId(1), "Condition", null, true); |
| 99 | + |
| 100 | + var beforeTime = DateTime.UtcNow; |
| 101 | + condition.SetSeverity(m_context, EventSeverity.High); |
| 102 | + var afterTime = DateTime.UtcNow; |
| 103 | + |
| 104 | + // Verify timestamps are updated |
| 105 | + Assert.That(condition.Severity.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 106 | + Assert.That(condition.Severity.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 107 | + Assert.That(condition.LastSeverity.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 108 | + Assert.That(condition.LastSeverity.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 109 | + |
| 110 | + // Verify change masks are cleared |
| 111 | + Assert.That(condition.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 112 | + } |
| 113 | + |
| 114 | + /// <summary> |
| 115 | + /// Test that SetActiveState updates the timestamp and clears change masks. |
| 116 | + /// </summary> |
| 117 | + [Test] |
| 118 | + public void SetActiveStateUpdatesTimestampAndClearsChangeMasks() |
| 119 | + { |
| 120 | + var alarm = new AlarmConditionState(m_telemetry, null); |
| 121 | + alarm.Create(m_context, new NodeId(1), "Alarm", null, true); |
| 122 | + |
| 123 | + var beforeTime = DateTime.UtcNow; |
| 124 | + alarm.SetActiveState(m_context, true); |
| 125 | + var afterTime = DateTime.UtcNow; |
| 126 | + |
| 127 | + // Verify timestamp is updated |
| 128 | + Assert.That(alarm.ActiveState.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 129 | + Assert.That(alarm.ActiveState.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 130 | + |
| 131 | + // Verify change masks are cleared |
| 132 | + Assert.That(alarm.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 133 | + Assert.That(alarm.ActiveState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 134 | + } |
| 135 | + |
| 136 | + /// <summary> |
| 137 | + /// Test that SetSuppressedState updates the timestamp and clears change masks. |
| 138 | + /// </summary> |
| 139 | + [Test] |
| 140 | + public void SetSuppressedStateUpdatesTimestampAndClearsChangeMasks() |
| 141 | + { |
| 142 | + var alarm = new AlarmConditionState(m_telemetry, null); |
| 143 | + alarm.Create(m_context, new NodeId(1), "Alarm", null, true); |
| 144 | + alarm.SuppressedState = new TwoStateVariableState(alarm); |
| 145 | + alarm.SuppressedState.Create(m_context, null, BrowseNames.SuppressedState, null, false); |
| 146 | + |
| 147 | + var beforeTime = DateTime.UtcNow; |
| 148 | + alarm.SetSuppressedState(m_context, true); |
| 149 | + var afterTime = DateTime.UtcNow; |
| 150 | + |
| 151 | + // Verify timestamp is updated |
| 152 | + Assert.That(alarm.SuppressedState.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 153 | + Assert.That(alarm.SuppressedState.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 154 | + |
| 155 | + // Verify change masks are cleared |
| 156 | + Assert.That(alarm.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 157 | + Assert.That(alarm.SuppressedState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 158 | + } |
| 159 | + |
| 160 | + /// <summary> |
| 161 | + /// Test that SetAcknowledgedState updates the timestamp and clears change masks. |
| 162 | + /// </summary> |
| 163 | + [Test] |
| 164 | + public void SetAcknowledgedStateUpdatesTimestampAndClearsChangeMasks() |
| 165 | + { |
| 166 | + var condition = new AcknowledgeableConditionState(null); |
| 167 | + condition.Create(m_context, new NodeId(1), "AckCondition", null, true); |
| 168 | + |
| 169 | + var beforeTime = DateTime.UtcNow; |
| 170 | + condition.SetAcknowledgedState(m_context, true); |
| 171 | + var afterTime = DateTime.UtcNow; |
| 172 | + |
| 173 | + // Verify timestamp is updated |
| 174 | + Assert.That(condition.AckedState.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 175 | + Assert.That(condition.AckedState.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 176 | + |
| 177 | + // Verify change masks are cleared |
| 178 | + Assert.That(condition.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 179 | + Assert.That(condition.AckedState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 180 | + } |
| 181 | + |
| 182 | + /// <summary> |
| 183 | + /// Test that SetConfirmedState updates the timestamp and clears change masks. |
| 184 | + /// </summary> |
| 185 | + [Test] |
| 186 | + public void SetConfirmedStateUpdatesTimestampAndClearsChangeMasks() |
| 187 | + { |
| 188 | + var condition = new AcknowledgeableConditionState(null); |
| 189 | + condition.Create(m_context, new NodeId(1), "AckCondition", null, true); |
| 190 | + condition.ConfirmedState = new TwoStateVariableState(condition); |
| 191 | + condition.ConfirmedState.Create(m_context, null, BrowseNames.ConfirmedState, null, false); |
| 192 | + |
| 193 | + var beforeTime = DateTime.UtcNow; |
| 194 | + condition.SetConfirmedState(m_context, true); |
| 195 | + var afterTime = DateTime.UtcNow; |
| 196 | + |
| 197 | + // Verify timestamp is updated |
| 198 | + Assert.That(condition.ConfirmedState.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 199 | + Assert.That(condition.ConfirmedState.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 200 | + |
| 201 | + // Verify change masks are cleared |
| 202 | + Assert.That(condition.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 203 | + Assert.That(condition.ConfirmedState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 204 | + } |
| 205 | + |
| 206 | + /// <summary> |
| 207 | + /// Test that SetShelvingState updates the timestamp and clears change masks. |
| 208 | + /// </summary> |
| 209 | + [Test] |
| 210 | + public void SetShelvingStateUpdatesTimestampAndClearsChangeMasks() |
| 211 | + { |
| 212 | + var alarm = new AlarmConditionState(m_telemetry, null); |
| 213 | + alarm.Create(m_context, new NodeId(1), "Alarm", null, true); |
| 214 | + alarm.ShelvingState = new ShelvedStateMachineState(alarm); |
| 215 | + alarm.ShelvingState.Create(m_context, null, BrowseNames.ShelvingState, null, false); |
| 216 | + alarm.ShelvingState.UnshelveTime = new PropertyState<double>(alarm.ShelvingState); |
| 217 | + |
| 218 | + var beforeTime = DateTime.UtcNow; |
| 219 | + alarm.SetShelvingState(m_context, true, false, 1000); |
| 220 | + var afterTime = DateTime.UtcNow; |
| 221 | + |
| 222 | + // Verify timestamp is updated on UnshelveTime |
| 223 | + Assert.That(alarm.ShelvingState.UnshelveTime.Timestamp, Is.GreaterThanOrEqualTo(beforeTime)); |
| 224 | + Assert.That(alarm.ShelvingState.UnshelveTime.Timestamp, Is.LessThanOrEqualTo(afterTime)); |
| 225 | + |
| 226 | + // Verify change masks are cleared |
| 227 | + Assert.That(alarm.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 228 | + } |
| 229 | + |
| 230 | + /// <summary> |
| 231 | + /// Test that subscribed clients are notified when SetActiveState is called. |
| 232 | + /// This simulates the issue scenario. |
| 233 | + /// </summary> |
| 234 | + [Test] |
| 235 | + public void SetActiveStateNotifiesSubscribers() |
| 236 | + { |
| 237 | + var alarm = new AlarmConditionState(m_telemetry, null); |
| 238 | + alarm.Create(m_context, new NodeId(1), "Alarm", null, true); |
| 239 | + |
| 240 | + // Initially inactive |
| 241 | + alarm.SetActiveState(m_context, false); |
| 242 | + var initialTimestamp = alarm.ActiveState.Timestamp; |
| 243 | + |
| 244 | + // Now activate the alarm - timestamp should be greater than or equal to the initial timestamp |
| 245 | + // since both could be set to DateTime.UtcNow which has limited precision |
| 246 | + var beforeActivation = DateTime.UtcNow; |
| 247 | + alarm.SetActiveState(m_context, true); |
| 248 | + var afterActivation = DateTime.UtcNow; |
| 249 | + |
| 250 | + // Verify that the timestamp is within the expected range |
| 251 | + Assert.That(alarm.ActiveState.Timestamp, Is.GreaterThanOrEqualTo(beforeActivation)); |
| 252 | + Assert.That(alarm.ActiveState.Timestamp, Is.LessThanOrEqualTo(afterActivation)); |
| 253 | + |
| 254 | + // Verify that change masks were cleared (indicating subscribers were notified) |
| 255 | + Assert.That(alarm.ActiveState.ChangeMasks, Is.EqualTo(NodeStateChangeMasks.None)); |
| 256 | + } |
| 257 | + |
67 | 258 | /// <summary> |
68 | 259 | /// Test that UpdateStateAfterEnable calls EvaluateRetainStateOnEnable |
69 | 260 | /// and that the default implementation sets Retain based on GetRetainState. |
|
0 commit comments