370 lines
16 KiB
C#
Raw Normal View History

using DeviceDetectorNET.Parser.Device;
using JiShe.CollectBus.Common.BuildSendDatas;
using JiShe.CollectBus.Common.Consts;
using JiShe.CollectBus.Common.Enums;
2024-10-29 16:28:14 +08:00
using JiShe.CollectBus.Common.Extensions;
using JiShe.CollectBus.Common.Helpers;
2024-10-29 16:28:14 +08:00
using JiShe.CollectBus.Common.Models;
using JiShe.CollectBus.Enums;
using JiShe.CollectBus.IotSystems.Devices;
2025-03-14 14:28:04 +08:00
using JiShe.CollectBus.IotSystems.MessageReceiveds;
using JiShe.CollectBus.IotSystems.Protocols;
using JiShe.CollectBus.Kafka.Producer;
2024-10-21 13:30:53 +08:00
using JiShe.CollectBus.Protocol.Contracts.Abstracts;
using JiShe.CollectBus.Protocol.Contracts.Interfaces;
2025-02-27 17:02:50 +08:00
using JiShe.CollectBus.Protocol.Contracts.Models;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
2025-02-27 17:02:50 +08:00
using Newtonsoft.Json.Linq;
using TouchSocket.Sockets;
using Volo.Abp.Domain.Repositories;
2024-09-30 17:10:43 +08:00
namespace JiShe.CollectBus.Protocol
{
public class StandardProtocolPlugin : ProtocolPlugin
2024-09-30 17:10:43 +08:00
{
private readonly ILogger<StandardProtocolPlugin> _logger;
private readonly IProducerService _producerService;
private readonly IRepository<Device, Guid> _deviceRepository;
2024-12-19 16:07:07 +08:00
/// <summary>
/// Initializes a new instance of the <see cref="StandardProtocolPlugin"/> class.
/// </summary>
2025-02-24 13:47:12 +08:00
/// <param name="serviceProvider">The service provider.</param>
public StandardProtocolPlugin(IServiceProvider serviceProvider,ILogger<StandardProtocolPlugin> logger) : base(serviceProvider, logger)
2024-09-30 17:53:14 +08:00
{
_logger= logger;
//_logger = serviceProvider.GetRequiredService<ILogger<StandardProtocolPlugin>>();
_producerService = serviceProvider.GetRequiredService<IProducerService>();
_deviceRepository = serviceProvider.GetRequiredService<IRepository<Device, Guid>>();
2024-09-30 17:10:43 +08:00
}
2024-12-19 16:07:07 +08:00
public sealed override ProtocolInfo Info => new(nameof(StandardProtocolPlugin), "376.1", "TCP", "376.1协议", "DTS1980");
2024-10-21 13:30:53 +08:00
public override async Task<T> AnalyzeAsync<T>(ITcpSessionClient client, string messageReceived, Action<T>? sendAction = null)
2024-10-29 16:28:14 +08:00
{
TB3761? tB3761 = Analysis3761(messageReceived);
if (tB3761 != null)
2025-02-27 17:02:50 +08:00
{
if (tB3761.AFN_FC?.AFN == (int)AFN.)
{
if (tB3761.A == null || tB3761.A.Code.IsNullOrWhiteSpace() || tB3761.A.A3?.D1_D7 == null || tB3761.SEQ?.PSEQ == null)
2025-03-12 13:39:26 +08:00
{
_logger.LogError($"解析AFN.链路接口检测报文失败,报文:{messageReceived},TB3761:{tB3761.Serialize()}");
2025-03-12 13:39:26 +08:00
}
else
2025-03-12 13:39:26 +08:00
{
2025-04-22 09:34:59 +08:00
if (tB3761.DT?.Fn == (int)FN.)
{
2025-04-22 09:34:59 +08:00
// 登录回复
if (tB3761.SEQ.CON == (int)CON.)
await LoginAsync(client, messageReceived, tB3761.A.Code, tB3761.A.A3?.D1_D7, tB3761.SEQ?.PSEQ);
2025-04-22 09:34:59 +08:00
}
else if (tB3761.DT?.Fn == (int)FN.)
{
// 心跳回复
//心跳帧有两种情况:
//1. 集中器先有登录帧,再有心跳帧
//2. 集中器没有登录帧,只有心跳帧
await HeartbeatAsync(client, messageReceived, tB3761.A.Code, tB3761.A.A3?.D1_D7, tB3761.SEQ?.PSEQ);
}
2025-03-12 13:39:26 +08:00
}
}
2025-02-27 17:02:50 +08:00
}
return (tB3761 as T)!;
2024-10-21 13:30:53 +08:00
}
/// <summary>
/// 登录回复
/// </summary>
/// <param name="client"></param>
/// <param name="code"></param>
/// <param name="msa"></param>
/// <param name="pseq"></param>
/// <returns></returns>
public async Task LoginAsync(ITcpSessionClient client,string messageReceived, string code, int? msa, int? pseq)
{
string oldClientId = $"{client.Id}";
await client.ResetIdAsync(code);
var deviceInfoList = await _deviceRepository.GetListAsync(a => a.Number == code);
if (deviceInfoList != null && deviceInfoList.Count > 1)
{
//todo 推送集中器编号重复预警
_logger.LogError($"集中器编号:{code},存在多个集中器,请检查集中器编号是否重复");
return;
}
var entity = deviceInfoList?.FirstOrDefault(a => a.Number == code);
if (entity == null)
{
await _deviceRepository.InsertAsync(new Device(code, oldClientId, DateTime.Now, DateTime.Now, DeviceStatus.Online));
}
else
{
entity.UpdateByLoginAndHeartbeat(oldClientId);
await _deviceRepository.UpdateAsync(entity);
}
var messageReceivedLoginEvent = new MessageReceivedLogin
{
ClientId = code,
ClientIp = client.IP,
ClientPort = client.Port,
MessageHexString = messageReceived,
DeviceNo = code,
MessageId = Guid.NewGuid().ToString()
};
//await _producerBus.PublishAsync(ProtocolConst.SubscriberLoginReceivedEventName, messageReceivedLoginEvent);
await _producerService.ProduceAsync(ProtocolConst.SubscriberLoginReceivedEventName, messageReceivedLoginEvent);
//await _producerBus.Publish( messageReceivedLoginEvent);
2025-04-22 09:34:59 +08:00
//var aTuple = (Tuple<string, int>)messageReceived.StringToPairs().GetAnalyzeValue(CommandChunkEnum.A);
//var seq = (Seq)messageReceived.StringToPairs().GetAnalyzeValue(CommandChunkEnum.SEQ);
var reqParam = new ReqParameter2
{
AFN = AFN.,
FunCode = (int)CFromStationFunCode.,
PRM = PRM.,
A =code,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = pseq!.Value
},
MSA = msa!.Value,
Pn = 0,
Fn = 1
};
var bytes = Build3761SendData.BuildSendCommandBytes(reqParam);
//await _producerBus.PublishAsync(ProtocolConst.SubscriberLoginIssuedEventName, new IssuedEventMessage { ClientId = messageReceived.ClientId, DeviceNo = messageReceived.DeviceNo, Message = bytes, Type = IssuedEventType.Login, MessageId = messageReceived.MessageId });
await _producerService.ProduceAsync(ProtocolConst.SubscriberLoginIssuedEventName, new IssuedEventMessage { ClientId = messageReceivedLoginEvent.ClientId, DeviceNo = messageReceivedLoginEvent.DeviceNo, Message = bytes, Type = IssuedEventType.Login, MessageId = messageReceivedLoginEvent.MessageId });
//await _producerBus.Publish(new IssuedEventMessage { ClientId = messageReceived.ClientId, DeviceNo = messageReceived.DeviceNo, Message = bytes, Type = IssuedEventType.Login, MessageId = messageReceived.MessageId });
}
/// <summary>
/// 心跳帧解析
/// </summary>
/// <param name="client"></param>
/// <param name="code"></param>
/// <param name="msa"></param>
/// <param name="pseq"></param>
/// <returns></returns>
public async Task HeartbeatAsync(ITcpSessionClient client,string messageReceived, string code, int? msa, int? pseq)
{
string clientId = code;
string oldClientId = $"{client.Id}";
var deviceInfoList = await _deviceRepository.GetListAsync(a => a.Number == code);
if (deviceInfoList != null && deviceInfoList.Count > 1)
{
//todo 推送集中器编号重复预警
_logger.LogError($"集中器编号:{code},存在多个集中器,请检查集中器编号是否重复");
return;
}
var entity = deviceInfoList?.FirstOrDefault(a => a.Number == code);
if (entity == null) //没有登录帧的设备,只有心跳帧
{
await client.ResetIdAsync(clientId);
await _deviceRepository.InsertAsync(new Device(code, oldClientId, DateTime.Now, DateTime.Now, DeviceStatus.Online));
}
else
{
if (clientId != oldClientId)
{
entity.UpdateByLoginAndHeartbeat(oldClientId);
}
else
{
entity.UpdateByLoginAndHeartbeat();
}
await _deviceRepository.UpdateAsync(entity);
}
var messageReceivedHeartbeatEvent = new MessageReceivedHeartbeat
{
ClientId = clientId,
ClientIp = client.IP,
ClientPort = client.Port,
MessageHexString = messageReceived,
DeviceNo = code,
MessageId = Guid.NewGuid().ToString()
};
//await _producerBus.PublishAsync(ProtocolConst.SubscriberHeartbeatReceivedEventName, messageReceivedHeartbeatEvent);
await _producerService.ProduceAsync(ProtocolConst.SubscriberHeartbeatReceivedEventName, messageReceivedHeartbeatEvent);
//await _producerBus.Publish(messageReceivedHeartbeatEvent);
var reqParam = new ReqParameter2()
{
AFN = AFN.,
FunCode = (int)CFromStationFunCode.,
PRM = PRM.,
A = code,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = pseq!.Value,
},
MSA = msa!.Value,
Pn = 0,
Fn = 1
};
var bytes = Build3761SendData.BuildSendCommandBytes(reqParam);
//await _producerBus.PublishAsync(ProtocolConst.SubscriberHeartbeatIssuedEventName, new IssuedEventMessage { ClientId = messageReceived.ClientId, DeviceNo = messageReceived.DeviceNo, Message = bytes, Type = IssuedEventType.Heartbeat, MessageId = messageReceived.MessageId });
await _producerService.ProduceAsync(ProtocolConst.SubscriberHeartbeatIssuedEventName, new IssuedEventMessage { ClientId = messageReceivedHeartbeatEvent.ClientId, DeviceNo = messageReceivedHeartbeatEvent.DeviceNo, Message = bytes, Type = IssuedEventType.Heartbeat, MessageId = messageReceivedHeartbeatEvent.MessageId });
//await _producerBus.Publish(new IssuedEventMessage { ClientId = messageReceived.ClientId, DeviceNo = messageReceived.DeviceNo, Message = bytes, Type = IssuedEventType.Heartbeat, MessageId = messageReceived.MessageId });
}
2024-10-21 16:24:29 +08:00
#region
//68
//32 00
//32 00
//68
//C9 1100'1001. 控制域C。
// D7=1, (终端发送)上行方向。
// D6=1, 此帧来自启动站。
// D5=0, (上行方向)要求访问位。表示终端无事件数据等待访问。
// D4=0, 保留
// D3~D0=9, 功能码。链路测试
//20 32 行政区划码
//90 26 终端地址
//00 主站地址和组地址标志。终端为单地址。 //3220 09 87 2
// 终端启动的发送帧的 MSA 应为 0, 其主站响应帧的 MSA 也应为 0.
//02 应用层功能码。AFN=2, 链路接口检测
//70 0111'0000. 帧序列域。无时间标签、单帧、需要确认。
//00 00 信息点。DA1和DA2全为“0”时表示终端信息点。
//01 00 信息类。F1, 登录。
//44 帧尾,包含用户区数据校验和
//16 帧结束标志
2024-10-21 13:30:53 +08:00
/// <summary>
2024-10-21 16:24:29 +08:00
/// 解析上行命令
2024-10-21 13:30:53 +08:00
/// </summary>
2024-10-21 16:24:29 +08:00
/// <param name="cmd"></param>
2024-10-21 13:30:53 +08:00
/// <returns></returns>
2024-10-21 16:24:29 +08:00
public CommandReulst? AnalysisCmd(string cmd)
2024-10-21 13:30:53 +08:00
{
2024-10-21 16:24:29 +08:00
CommandReulst? commandReulst = null;
2024-10-29 16:28:14 +08:00
var hexStringList = cmd.StringToPairs();
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
if (hexStringList.Count < hearderLen)
2024-10-21 13:30:53 +08:00
{
2024-10-21 16:24:29 +08:00
return commandReulst;
2024-10-21 13:30:53 +08:00
}
2024-10-21 16:24:29 +08:00
//验证起始字符
2024-11-08 14:53:36 +08:00
if (!hexStringList[0].IsStartStr() || !hexStringList[5].IsStartStr())
2024-10-21 13:30:53 +08:00
{
2024-10-21 16:24:29 +08:00
return commandReulst;
2024-10-21 13:30:53 +08:00
}
2024-10-21 16:24:29 +08:00
var lenHexStr = $"{hexStringList[2]}{hexStringList[1]}";
2024-10-29 16:28:14 +08:00
var lenBin = lenHexStr.HexToBin();
var len = lenBin.Remove(lenBin.Length - 2).BinToDec();
2024-10-21 16:24:29 +08:00
//验证长度
if (hexStringList.Count - 2 != hearderLen + len)
return commandReulst;
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
var userDataIndex = hearderLen;
var c = hexStringList[userDataIndex];//控制域 1字节
userDataIndex += 1;
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
var aHexList = hexStringList.Skip(userDataIndex).Take(5).ToList();//地址域 5字节
var a = AnalysisA(aHexList);
2024-10-29 16:28:14 +08:00
var a3Bin = aHexList[4].HexToBin().PadLeft(8, '0');
var mSA = a3Bin.Substring(0, 7).BinToDec();
2024-10-21 16:24:29 +08:00
userDataIndex += 5;
2024-10-21 13:30:53 +08:00
2024-10-29 16:28:14 +08:00
var aFN = (AFN)hexStringList[userDataIndex].HexToDec();//1字节
2024-10-21 16:24:29 +08:00
userDataIndex += 1;
2024-10-21 13:30:53 +08:00
2024-10-29 16:28:14 +08:00
var seq = hexStringList[userDataIndex].HexToBin().PadLeft(8, '0');
2024-10-21 16:24:29 +08:00
var tpV = (TpV)Convert.ToInt32(seq.Substring(0, 1));
var fIRFIN = (FIRFIN)Convert.ToInt32(seq.Substring(1, 2));
var cON = (CON)Convert.ToInt32(seq.Substring(3, 1));
var prseqBin = seq.Substring(4, 4);
userDataIndex += 1;
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
// (DA2 - 1) * 8 + DA1 = pn
2024-10-29 16:28:14 +08:00
var da1Bin = hexStringList[userDataIndex].HexToBin();
2024-10-21 16:24:29 +08:00
var da1 = da1Bin == "0" ? 0 : da1Bin.Length;
userDataIndex += 1;
2024-10-29 16:28:14 +08:00
var da2 = hexStringList[userDataIndex].HexToDec();
2024-10-21 16:24:29 +08:00
var pn = da2 == 0 ? 0 : (da2 - 1) * 8 + da1;
2024-10-29 16:28:14 +08:00
userDataIndex += 1;
2024-10-21 16:24:29 +08:00
//(DT2*8)+DT1=fn
2024-10-29 16:28:14 +08:00
var dt1Bin = hexStringList[userDataIndex].HexToBin();
2024-10-21 16:24:29 +08:00
var dt1 = dt1Bin != "0" ? dt1Bin.Length : 0;
userDataIndex += 1;
2024-10-29 16:28:14 +08:00
var dt2 = hexStringList[userDataIndex].HexToDec();
2024-10-22 20:57:26 +08:00
var fn = dt2 * 8 + dt1;
2024-10-29 16:28:14 +08:00
userDataIndex += 1;
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
//数据单元
var datas = hexStringList.Skip(userDataIndex).Take(len + hearderLen - userDataIndex).ToList();
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
//EC
//Tp
commandReulst = new CommandReulst()
{
A = a,
MSA = mSA,
AFN = aFN,
Seq = new Seq()
{
TpV = tpV,
FIRFIN = fIRFIN,
CON = cON,
2024-10-29 16:28:14 +08:00
PRSEQ = prseqBin.BinToDec(),
2024-10-21 16:24:29 +08:00
},
CmdLength = len,
Pn = pn,
Fn = fn,
HexDatas = datas
};
2024-10-21 13:30:53 +08:00
2024-10-21 16:24:29 +08:00
return commandReulst;
2024-10-21 13:30:53 +08:00
}
/// <summary>
2024-10-21 16:24:29 +08:00
/// 解析地址
2024-10-21 13:30:53 +08:00
/// </summary>
2024-10-21 16:24:29 +08:00
/// <param name="aHexList"></param>
2024-10-21 13:30:53 +08:00
/// <returns></returns>
2024-10-21 16:24:29 +08:00
private string AnalysisA(List<string> aHexList)
2024-10-21 13:30:53 +08:00
{
2024-10-21 16:24:29 +08:00
var a1 = aHexList[1] + aHexList[0];
var a2 = aHexList[3] + aHexList[2];
2024-10-29 16:28:14 +08:00
var a2Dec = a2.HexToDec();
2024-10-21 16:24:29 +08:00
var a3 = aHexList[4];
var a = $"{a1}{a2Dec.ToString().PadLeft(5, '0')}";
return a;
2024-10-21 13:30:53 +08:00
}
2024-10-21 16:24:29 +08:00
#endregion
2024-09-30 17:10:43 +08:00
}
}