2025-04-27 09:40:31 +08:00
|
|
|
|
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.Models;
|
2025-04-21 23:47:11 +08:00
|
|
|
|
using JiShe.CollectBus.IotSystems.Devices;
|
2025-03-14 14:28:04 +08:00
|
|
|
|
using JiShe.CollectBus.IotSystems.Protocols;
|
2025-04-21 23:47:11 +08:00
|
|
|
|
using JiShe.CollectBus.Kafka.Producer;
|
2024-10-21 13:30:53 +08:00
|
|
|
|
using JiShe.CollectBus.Protocol.Contracts.Abstracts;
|
2025-04-24 17:48:20 +08:00
|
|
|
|
using JiShe.CollectBus.Protocol.Contracts.SendData;
|
|
|
|
|
|
using JiShe.CollectBus.Protocol.SendData;
|
|
|
|
|
|
using Mapster;
|
2025-04-21 23:47:11 +08:00
|
|
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
|
|
|
|
using Microsoft.Extensions.Logging;
|
|
|
|
|
|
using TouchSocket.Sockets;
|
|
|
|
|
|
using Volo.Abp.Domain.Repositories;
|
2024-09-30 17:10:43 +08:00
|
|
|
|
|
|
|
|
|
|
namespace JiShe.CollectBus.Protocol
|
2025-03-19 14:31:04 +08:00
|
|
|
|
{
|
2025-04-25 17:18:59 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// T6452007协议插件
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public class T6452007ProtocolPlugin : T37612012ProtocolPlugin
|
2024-09-30 17:10:43 +08:00
|
|
|
|
{
|
2025-04-25 12:01:15 +08:00
|
|
|
|
private readonly ILogger<T6452007ProtocolPlugin> _logger;
|
2025-04-25 17:18:59 +08:00
|
|
|
|
|
2025-04-25 12:01:15 +08:00
|
|
|
|
public readonly Dictionary<string, Telemetry6452007PacketBuilder.T6452007Delegate> T645ControlHandlers;
|
2025-04-24 17:48:20 +08:00
|
|
|
|
|
2024-12-19 16:07:07 +08:00
|
|
|
|
/// <summary>
|
2025-04-25 12:01:15 +08:00
|
|
|
|
/// Initializes a new instance of the <see cref="T6452007ProtocolPlugin"/> class.
|
2024-12-19 16:07:07 +08:00
|
|
|
|
/// </summary>
|
2025-02-24 13:47:12 +08:00
|
|
|
|
/// <param name="serviceProvider">The service provider.</param>
|
2025-04-25 17:18:59 +08:00
|
|
|
|
public T6452007ProtocolPlugin(IServiceProvider serviceProvider, ILogger<T6452007ProtocolPlugin> logger, ITcpService tcpService) : base(serviceProvider, logger, tcpService)
|
2024-09-30 17:53:14 +08:00
|
|
|
|
{
|
2025-04-22 23:45:08 +08:00
|
|
|
|
_logger = logger;
|
2025-04-25 12:01:15 +08:00
|
|
|
|
T645ControlHandlers = Telemetry6452007PacketBuilder.T645ControlHandlers;
|
2024-09-30 17:10:43 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-25 15:21:46 +08:00
|
|
|
|
public sealed override ProtocolInfo Info => new(nameof(T6452007ProtocolPlugin), "376.1/645-2007", "TCP", "376.1/645-2007协议", "DTS1980");
|
2024-10-21 13:30:53 +08:00
|
|
|
|
|
2025-04-21 23:47:11 +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
|
|
|
|
{
|
2025-04-25 14:23:06 +08:00
|
|
|
|
//TODO:645解析报文
|
|
|
|
|
|
//TB3761? tB3761 = Analysis3761(messageReceived);
|
|
|
|
|
|
//if (tB3761 != null)
|
2025-04-25 09:28:56 +08:00
|
|
|
|
//{
|
2025-04-25 14:23:06 +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)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// _logger.LogError($"解析AFN.链路接口检测报文失败,报文:{messageReceived},TB3761:{tB3761.Serialize()}");
|
|
|
|
|
|
// }
|
|
|
|
|
|
// else
|
|
|
|
|
|
// {
|
|
|
|
|
|
// if (tB3761.DT?.Fn == (int)FN.登录)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// // 登录回复
|
|
|
|
|
|
// if (tB3761.SEQ.CON == (int)CON.需要对该帧进行确认)
|
|
|
|
|
|
// await LoginAsync(client, messageReceived, tB3761.A.Code, tB3761.A.A3?.D1_D7, tB3761.SEQ?.PSEQ);
|
|
|
|
|
|
// }
|
|
|
|
|
|
// else if (tB3761.DT?.Fn == (int)FN.心跳)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// // 心跳回复
|
|
|
|
|
|
// //心跳帧有两种情况:
|
|
|
|
|
|
// //1. 集中器先有登录帧,再有心跳帧
|
|
|
|
|
|
// //2. 集中器没有登录帧,只有心跳帧
|
|
|
|
|
|
// await HeartbeatAsync(client, messageReceived, tB3761.A.Code, tB3761.A.A3?.D1_D7, tB3761.SEQ?.PSEQ);
|
|
|
|
|
|
// }
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
// await OnTcpNormalReceived(client, tB3761);
|
|
|
|
|
|
//}
|
|
|
|
|
|
//return (tB3761 as T)!;
|
|
|
|
|
|
return null;
|
2025-04-25 09:28:56 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-23 23:42:35 +08:00
|
|
|
|
/// <summary>
|
2025-04-24 17:48:20 +08:00
|
|
|
|
/// 组装报文
|
2025-04-23 23:42:35 +08:00
|
|
|
|
/// </summary>
|
2025-04-24 23:39:39 +08:00
|
|
|
|
/// <param name="request">报文构建参数</param>
|
2025-04-23 23:42:35 +08:00
|
|
|
|
/// <returns></returns>
|
2025-04-24 23:39:39 +08:00
|
|
|
|
public override async Task<ProtocolBuildResponse> BuildAsync(ProtocolBuildRequest request)
|
2025-04-23 23:42:35 +08:00
|
|
|
|
{
|
2025-04-24 23:39:39 +08:00
|
|
|
|
if (request == null)
|
|
|
|
|
|
{
|
2025-04-25 17:18:59 +08:00
|
|
|
|
_logger.LogError($"{nameof(ProtocolBuildResponse)} 报文构建失败,参数为空");
|
|
|
|
|
|
return new ProtocolBuildResponse();
|
2025-04-24 23:39:39 +08:00
|
|
|
|
}
|
2025-04-24 17:48:20 +08:00
|
|
|
|
var itemCodeArr = request.ItemCode.Split('_');
|
|
|
|
|
|
var aFNStr = itemCodeArr[0];
|
|
|
|
|
|
var aFN = (AFN)aFNStr.HexToDec();
|
|
|
|
|
|
var fn = int.Parse(itemCodeArr[1]);
|
|
|
|
|
|
Telemetry3761PacketResponse builderResponse = null;
|
|
|
|
|
|
|
|
|
|
|
|
List<string> dataUnit = new List<string>();
|
2025-04-27 09:40:31 +08:00
|
|
|
|
//数据转发场景 10H_F1
|
|
|
|
|
|
if (request.ItemCode == T37612012PacketItemCodeConst.AFN10HFN01H && request.SubProtocolRequest != null && string.IsNullOrWhiteSpace(request.SubProtocolRequest.ItemCode) == false)
|
2025-04-24 17:48:20 +08:00
|
|
|
|
{
|
2025-04-24 23:39:39 +08:00
|
|
|
|
var t645PacketHandlerName = $"C{request.SubProtocolRequest.ItemCode}_Send";
|
2025-04-25 12:01:15 +08:00
|
|
|
|
Telemetry6452007PacketResponse t645PacketResponse = null;
|
2025-04-24 17:48:20 +08:00
|
|
|
|
|
2025-04-24 23:39:39 +08:00
|
|
|
|
if (T645ControlHandlers != null && T645ControlHandlers.TryGetValue(t645PacketHandlerName
|
|
|
|
|
|
, out var t645PacketHandler))
|
2025-04-24 17:48:20 +08:00
|
|
|
|
{
|
2025-04-25 12:01:15 +08:00
|
|
|
|
t645PacketResponse = t645PacketHandler(new Telemetry6452007PacketRequest()
|
2025-04-24 17:48:20 +08:00
|
|
|
|
{
|
2025-04-24 23:39:39 +08:00
|
|
|
|
MeterAddress = request.SubProtocolRequest.MeterAddress,
|
|
|
|
|
|
Password = request.SubProtocolRequest.Password,
|
|
|
|
|
|
ItemCode = request.SubProtocolRequest.ItemCode,
|
2025-04-24 17:48:20 +08:00
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (t645PacketResponse != null)
|
|
|
|
|
|
{
|
|
|
|
|
|
dataUnit = t645PacketResponse.Data;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
string afnMethonCode = $"AFN{aFNStr}_Fn_Send";
|
2025-04-25 17:18:59 +08:00
|
|
|
|
if (base.T3761AFNHandlers != null && base.T3761AFNHandlers.TryGetValue(afnMethonCode
|
2025-04-24 17:48:20 +08:00
|
|
|
|
, out var handler))
|
|
|
|
|
|
{
|
|
|
|
|
|
builderResponse = handler(new Telemetry3761PacketRequest()
|
|
|
|
|
|
{
|
|
|
|
|
|
FocusAddress = request.FocusAddress,
|
|
|
|
|
|
Fn = fn,
|
|
|
|
|
|
Pn = request.Pn,
|
|
|
|
|
|
DataUnit = dataUnit,
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (builderResponse == null)
|
|
|
|
|
|
{
|
2025-04-24 23:39:39 +08:00
|
|
|
|
return new ProtocolBuildResponse();
|
2025-04-24 17:48:20 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-04-24 23:39:39 +08:00
|
|
|
|
var result = builderResponse.Adapt<ProtocolBuildResponse>();
|
2025-04-24 17:48:20 +08:00
|
|
|
|
result.IsSuccess = true;
|
2025-04-21 23:47:11 +08:00
|
|
|
|
|
2025-04-24 17:48:20 +08:00
|
|
|
|
return await Task.FromResult(result);
|
2025-04-21 23:47:11 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
|
}
|