feat(communication): map enriched alarm fields across gRPC (server + client)
This commit is contained in:
@@ -67,7 +67,24 @@ public class StreamRelayActor : ReceiveActor
|
|||||||
Priority = msg.Priority,
|
Priority = msg.Priority,
|
||||||
Timestamp = Timestamp.FromDateTimeOffset(msg.Timestamp),
|
Timestamp = Timestamp.FromDateTimeOffset(msg.Timestamp),
|
||||||
Level = MapAlarmLevel(msg.Level),
|
Level = MapAlarmLevel(msg.Level),
|
||||||
Message = msg.Message ?? string.Empty
|
Message = msg.Message ?? string.Empty,
|
||||||
|
// Native alarm enrichment (additive — computed alarms map their default condition).
|
||||||
|
Kind = msg.Kind.ToString(),
|
||||||
|
Active = msg.Condition.Active,
|
||||||
|
Acknowledged = msg.Condition.Acknowledged,
|
||||||
|
Confirmed = msg.Condition.Confirmed ?? false,
|
||||||
|
ShelveState = AlarmShelveStateCodec.ToWire(msg.Condition.Shelve),
|
||||||
|
Suppressed = msg.Condition.Suppressed,
|
||||||
|
SourceReference = msg.SourceReference ?? string.Empty,
|
||||||
|
AlarmTypeName = msg.AlarmTypeName ?? string.Empty,
|
||||||
|
Category = msg.Category ?? string.Empty,
|
||||||
|
OperatorUser = msg.OperatorUser ?? string.Empty,
|
||||||
|
OperatorComment = msg.OperatorComment ?? string.Empty,
|
||||||
|
OriginalRaiseTime = msg.OriginalRaiseTime.HasValue
|
||||||
|
? Timestamp.FromDateTimeOffset(msg.OriginalRaiseTime.Value)
|
||||||
|
: null,
|
||||||
|
CurrentValue = msg.CurrentValue ?? string.Empty,
|
||||||
|
LimitValue = msg.LimitValue ?? string.Empty
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
||||||
|
|
||||||
|
namespace ZB.MOM.WW.ScadaBridge.Communication.Grpc;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Maps <see cref="AlarmShelveState"/> to/from the wire string carried in the
|
||||||
|
/// <c>AlarmStateUpdate.shelve_state</c> proto field. The wire form is the enum
|
||||||
|
/// member name; an empty or unrecognized string parses back to
|
||||||
|
/// <see cref="AlarmShelveState.Unshelved"/> (the safe default).
|
||||||
|
/// </summary>
|
||||||
|
public static class AlarmShelveStateCodec
|
||||||
|
{
|
||||||
|
/// <summary>Returns the wire string for a shelve state (the enum member name).</summary>
|
||||||
|
public static string ToWire(AlarmShelveState state) => state.ToString();
|
||||||
|
|
||||||
|
/// <summary>Parses a wire string back to a shelve state; defaults to <see cref="AlarmShelveState.Unshelved"/>.</summary>
|
||||||
|
public static AlarmShelveState Parse(string? wire) =>
|
||||||
|
Enum.TryParse<AlarmShelveState>(wire, ignoreCase: true, out var state)
|
||||||
|
? state
|
||||||
|
: AlarmShelveState.Unshelved;
|
||||||
|
}
|
||||||
@@ -3,6 +3,7 @@ using Grpc.Core;
|
|||||||
using Grpc.Net.Client;
|
using Grpc.Net.Client;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using ZB.MOM.WW.ScadaBridge.Commons.Messages.Streaming;
|
using ZB.MOM.WW.ScadaBridge.Commons.Messages.Streaming;
|
||||||
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Alarms;
|
||||||
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
||||||
using Google.Protobuf.WellKnownTypes;
|
using Google.Protobuf.WellKnownTypes;
|
||||||
|
|
||||||
@@ -233,11 +234,32 @@ public class SiteStreamGrpcClient : IAsyncDisposable, IDisposable
|
|||||||
evt.AlarmChanged.Timestamp.ToDateTimeOffset())
|
evt.AlarmChanged.Timestamp.ToDateTimeOffset())
|
||||||
{
|
{
|
||||||
Level = MapAlarmLevel(evt.AlarmChanged.Level),
|
Level = MapAlarmLevel(evt.AlarmChanged.Level),
|
||||||
Message = evt.AlarmChanged.Message ?? string.Empty
|
Message = evt.AlarmChanged.Message ?? string.Empty,
|
||||||
|
// Native alarm enrichment (additive — computed alarms carry their default condition).
|
||||||
|
Kind = ParseAlarmKind(evt.AlarmChanged.Kind),
|
||||||
|
Condition = new AlarmConditionState(
|
||||||
|
Active: evt.AlarmChanged.Active,
|
||||||
|
Acknowledged: evt.AlarmChanged.Acknowledged,
|
||||||
|
Confirmed: evt.AlarmChanged.Confirmed,
|
||||||
|
Shelve: AlarmShelveStateCodec.Parse(evt.AlarmChanged.ShelveState),
|
||||||
|
Suppressed: evt.AlarmChanged.Suppressed,
|
||||||
|
Severity: evt.AlarmChanged.Priority),
|
||||||
|
SourceReference = evt.AlarmChanged.SourceReference ?? string.Empty,
|
||||||
|
AlarmTypeName = evt.AlarmChanged.AlarmTypeName ?? string.Empty,
|
||||||
|
Category = evt.AlarmChanged.Category ?? string.Empty,
|
||||||
|
OperatorUser = evt.AlarmChanged.OperatorUser ?? string.Empty,
|
||||||
|
OperatorComment = evt.AlarmChanged.OperatorComment ?? string.Empty,
|
||||||
|
OriginalRaiseTime = evt.AlarmChanged.OriginalRaiseTime?.ToDateTimeOffset(),
|
||||||
|
CurrentValue = evt.AlarmChanged.CurrentValue ?? string.Empty,
|
||||||
|
LimitValue = evt.AlarmChanged.LimitValue ?? string.Empty
|
||||||
},
|
},
|
||||||
_ => null
|
_ => null
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// <summary>Parses the wire "kind" string back to <see cref="AlarmKind"/>; defaults to Computed.</summary>
|
||||||
|
internal static AlarmKind ParseAlarmKind(string? kind) =>
|
||||||
|
System.Enum.TryParse<AlarmKind>(kind, ignoreCase: true, out var k) ? k : AlarmKind.Computed;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Maps proto Quality enum to domain string. Internal for testability.
|
/// Maps proto Quality enum to domain string. Internal for testability.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|||||||
@@ -40,16 +40,31 @@ public class SiteStreamGrpcClientTests
|
|||||||
public void ConvertToDomainEvent_AlarmChanged_MapsCorrectly()
|
public void ConvertToDomainEvent_AlarmChanged_MapsCorrectly()
|
||||||
{
|
{
|
||||||
var ts = DateTimeOffset.UtcNow;
|
var ts = DateTimeOffset.UtcNow;
|
||||||
|
var raiseTime = ts.AddMinutes(-30);
|
||||||
var evt = new SiteStreamEvent
|
var evt = new SiteStreamEvent
|
||||||
{
|
{
|
||||||
CorrelationId = "corr-2",
|
CorrelationId = "corr-2",
|
||||||
AlarmChanged = new AlarmStateUpdate
|
AlarmChanged = new AlarmStateUpdate
|
||||||
{
|
{
|
||||||
InstanceUniqueName = "Site1.Motor01",
|
InstanceUniqueName = "Site1.Motor01",
|
||||||
AlarmName = "OverTemp",
|
AlarmName = "T01.Hi",
|
||||||
State = AlarmStateEnum.AlarmStateActive,
|
State = AlarmStateEnum.AlarmStateActive,
|
||||||
Priority = 3,
|
Priority = 850,
|
||||||
Timestamp = Timestamp.FromDateTimeOffset(ts)
|
Timestamp = Timestamp.FromDateTimeOffset(ts),
|
||||||
|
Kind = "NativeOpcUa",
|
||||||
|
Active = true,
|
||||||
|
Acknowledged = false,
|
||||||
|
Confirmed = false,
|
||||||
|
ShelveState = "TimedShelved",
|
||||||
|
Suppressed = true,
|
||||||
|
SourceReference = "T01.Hi",
|
||||||
|
AlarmTypeName = "AnalogLimit.Hi",
|
||||||
|
Category = "Process",
|
||||||
|
OperatorUser = "op2",
|
||||||
|
OperatorComment = "shelved",
|
||||||
|
OriginalRaiseTime = Timestamp.FromDateTimeOffset(raiseTime),
|
||||||
|
CurrentValue = "120",
|
||||||
|
LimitValue = "100"
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -57,10 +72,26 @@ public class SiteStreamGrpcClientTests
|
|||||||
|
|
||||||
var alarm = Assert.IsType<AlarmStateChanged>(result);
|
var alarm = Assert.IsType<AlarmStateChanged>(result);
|
||||||
Assert.Equal("Site1.Motor01", alarm.InstanceUniqueName);
|
Assert.Equal("Site1.Motor01", alarm.InstanceUniqueName);
|
||||||
Assert.Equal("OverTemp", alarm.AlarmName);
|
Assert.Equal("T01.Hi", alarm.AlarmName);
|
||||||
Assert.Equal(AlarmState.Active, alarm.State);
|
Assert.Equal(AlarmState.Active, alarm.State);
|
||||||
Assert.Equal(3, alarm.Priority);
|
Assert.Equal(850, alarm.Priority);
|
||||||
Assert.Equal(ts, alarm.Timestamp);
|
Assert.Equal(ts, alarm.Timestamp);
|
||||||
|
|
||||||
|
// Native enrichment mapped back.
|
||||||
|
Assert.Equal(AlarmKind.NativeOpcUa, alarm.Kind);
|
||||||
|
Assert.True(alarm.Condition.Active);
|
||||||
|
Assert.False(alarm.Condition.Acknowledged);
|
||||||
|
Assert.Equal(AlarmShelveState.TimedShelved, alarm.Condition.Shelve);
|
||||||
|
Assert.True(alarm.Condition.Suppressed);
|
||||||
|
Assert.Equal(850, alarm.Condition.Severity);
|
||||||
|
Assert.Equal("T01.Hi", alarm.SourceReference);
|
||||||
|
Assert.Equal("AnalogLimit.Hi", alarm.AlarmTypeName);
|
||||||
|
Assert.Equal("Process", alarm.Category);
|
||||||
|
Assert.Equal("op2", alarm.OperatorUser);
|
||||||
|
Assert.Equal("shelved", alarm.OperatorComment);
|
||||||
|
Assert.Equal(raiseTime, alarm.OriginalRaiseTime);
|
||||||
|
Assert.Equal("120", alarm.CurrentValue);
|
||||||
|
Assert.Equal("100", alarm.LimitValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ using Akka.Actor;
|
|||||||
using Akka.TestKit.Xunit2;
|
using Akka.TestKit.Xunit2;
|
||||||
using Google.Protobuf.WellKnownTypes;
|
using Google.Protobuf.WellKnownTypes;
|
||||||
using ZB.MOM.WW.ScadaBridge.Commons.Messages.Streaming;
|
using ZB.MOM.WW.ScadaBridge.Commons.Messages.Streaming;
|
||||||
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Alarms;
|
||||||
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
using ZB.MOM.WW.ScadaBridge.Commons.Types.Enums;
|
||||||
using ZB.MOM.WW.ScadaBridge.Communication.Actors;
|
using ZB.MOM.WW.ScadaBridge.Communication.Actors;
|
||||||
using ZB.MOM.WW.ScadaBridge.Communication.Grpc;
|
using ZB.MOM.WW.ScadaBridge.Communication.Grpc;
|
||||||
@@ -57,8 +58,23 @@ public class StreamRelayActorTests : TestKit
|
|||||||
new StreamRelayActor(correlationId, channel.Writer)));
|
new StreamRelayActor(correlationId, channel.Writer)));
|
||||||
|
|
||||||
var timestamp = new DateTimeOffset(2026, 3, 21, 11, 0, 0, TimeSpan.Zero);
|
var timestamp = new DateTimeOffset(2026, 3, 21, 11, 0, 0, TimeSpan.Zero);
|
||||||
|
var raiseTime = new DateTimeOffset(2026, 3, 21, 10, 0, 0, TimeSpan.Zero);
|
||||||
var domainEvent = new AlarmStateChanged(
|
var domainEvent = new AlarmStateChanged(
|
||||||
"Site1.Pump01", "HighPressure", AlarmState.Active, 2, timestamp);
|
"Site1.Pump01", "T01.Hi", AlarmState.Active, 900, timestamp)
|
||||||
|
{
|
||||||
|
Kind = AlarmKind.NativeMxAccess,
|
||||||
|
SourceReference = "T01.Hi",
|
||||||
|
AlarmTypeName = "AnalogLimit.Hi",
|
||||||
|
Category = "Process",
|
||||||
|
OperatorUser = "op1",
|
||||||
|
OperatorComment = "ack",
|
||||||
|
OriginalRaiseTime = raiseTime,
|
||||||
|
CurrentValue = "92",
|
||||||
|
LimitValue = "90",
|
||||||
|
Condition = new AlarmConditionState(
|
||||||
|
Active: true, Acknowledged: true, Confirmed: null,
|
||||||
|
Shelve: AlarmShelveState.OneShotShelved, Suppressed: false, Severity: 900)
|
||||||
|
};
|
||||||
|
|
||||||
actor.Tell(domainEvent);
|
actor.Tell(domainEvent);
|
||||||
|
|
||||||
@@ -76,10 +92,25 @@ public class StreamRelayActorTests : TestKit
|
|||||||
|
|
||||||
var alarm = protoEvent.AlarmChanged;
|
var alarm = protoEvent.AlarmChanged;
|
||||||
Assert.Equal("Site1.Pump01", alarm.InstanceUniqueName);
|
Assert.Equal("Site1.Pump01", alarm.InstanceUniqueName);
|
||||||
Assert.Equal("HighPressure", alarm.AlarmName);
|
Assert.Equal("T01.Hi", alarm.AlarmName);
|
||||||
Assert.Equal(AlarmStateEnum.AlarmStateActive, alarm.State);
|
Assert.Equal(AlarmStateEnum.AlarmStateActive, alarm.State);
|
||||||
Assert.Equal(2, alarm.Priority);
|
Assert.Equal(900, alarm.Priority);
|
||||||
Assert.Equal(Timestamp.FromDateTimeOffset(timestamp), alarm.Timestamp);
|
Assert.Equal(Timestamp.FromDateTimeOffset(timestamp), alarm.Timestamp);
|
||||||
|
|
||||||
|
// Native enrichment mapped out.
|
||||||
|
Assert.Equal("NativeMxAccess", alarm.Kind);
|
||||||
|
Assert.True(alarm.Active);
|
||||||
|
Assert.True(alarm.Acknowledged);
|
||||||
|
Assert.Equal("OneShotShelved", alarm.ShelveState);
|
||||||
|
Assert.False(alarm.Suppressed);
|
||||||
|
Assert.Equal("T01.Hi", alarm.SourceReference);
|
||||||
|
Assert.Equal("AnalogLimit.Hi", alarm.AlarmTypeName);
|
||||||
|
Assert.Equal("Process", alarm.Category);
|
||||||
|
Assert.Equal("op1", alarm.OperatorUser);
|
||||||
|
Assert.Equal("ack", alarm.OperatorComment);
|
||||||
|
Assert.Equal(Timestamp.FromDateTimeOffset(raiseTime), alarm.OriginalRaiseTime);
|
||||||
|
Assert.Equal("92", alarm.CurrentValue);
|
||||||
|
Assert.Equal("90", alarm.LimitValue);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Fact]
|
[Fact]
|
||||||
|
|||||||
Reference in New Issue
Block a user