xiugai daima

This commit is contained in:
cli 2024-10-29 16:28:14 +08:00
parent 07211eed8b
commit acd061b9ee
26 changed files with 910 additions and 1000 deletions

View File

@ -1,176 +0,0 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text.RegularExpressions;
namespace JiShe.CollectBus.Common
{
public class DataConvert
{
/// <summary>
/// 十进制转二进制
/// </summary>
/// <param name="decimalNumber">十进制数</param>
/// <returns></returns>
public static string DecToBin(int decimalNumber)
{
var binaryString = Convert.ToString(decimalNumber, 2);
return binaryString;
}
/// <summary>
/// 二进制转十六进制
/// </summary>
/// <param name="binaryString"></param>
/// <returns></returns>
public static string BinToHex(string binaryString)
{
var decimalNumber = Convert.ToInt32(binaryString, 2);// 将二进制字符串转换为整数
var hexString = Convert.ToString(decimalNumber, 16); //decimalNumber.ToString("X"); // 将整数转换为十六进制字符串
return hexString;
}
/// <summary>
/// 十进制转十六进制
/// </summary>
/// <param name="decimalNumber"></param>
/// <returns></returns>
public static string DecToHex(int decimalNumber)
{
var hexString = decimalNumber.ToString("X");
return hexString;
}
/// <summary>
/// 二进制转十进制
/// </summary>
/// <param name="binaryString"></param>
/// <returns></returns>
public static int BinToDec(string binaryString)
{
var decimalNumber = Convert.ToInt32(binaryString, 2);
return decimalNumber;
}
/// <summary>
/// 十六进制转十进制
/// </summary>
/// <param name="hexString"></param>
/// <returns></returns>
public static int HexToDec(string hexString)
{
var decimalNumber = Convert.ToInt32(hexString, 16);
return decimalNumber;
}
/// <summary>
/// 十六进制转二进制
/// </summary>
/// <param name="hexString"></param>
/// <returns></returns>
public static string HexToBin(string hexString)
{
var binaryValue = Convert.ToString(Convert.ToInt32(hexString, 16), 2);
return binaryValue;
}
/// <summary>
/// 字符串倒序
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static string StringReversed(string str)
{
var reversed = new string(str.Reverse().ToArray());
return reversed;
}
/// <summary>
/// 字符串分割成2个字符一组
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static List<string> StringToPairs(string str)
{
var pairs = str.Select((ch, index) => new { ch, index })
.GroupBy(x => x.index / 2)
.Select(g => string.Concat(g.Select(x => x.ch)))
.ToList();
return pairs;
}
/// <summary>
/// 格式化字符串
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static string StrAddSpan(string str)
{
if (str == "")
{
return "";
}
return Regex.Replace(str.Replace(" ", ""), @"(?<=[0-9A-Za-z]{2})[0-9A-Za-z]{2}", " $0").Trim();
}
/// <summary>
/// 格式化字符串且反转
/// </summary>
/// <param name="ste"></param>
/// <returns></returns>
public static string StrReverseOrder(string ste)
{
if (ste == "")
{
return "";
}
string[] strArr = ste.Split(new string[] { " " }, System.StringSplitOptions.RemoveEmptyEntries);
return string.Join(" ", strArr.Reverse());
}
/// <summary>
/// 数据值加33
/// </summary>
/// <param name="Str"></param>
/// <returns></returns>
public static string StrAddHex33(string str)
{
if (str == "")
{
return "";
}
string[] strArr = str.Split(new string[] { " " }, System.StringSplitOptions.RemoveEmptyEntries);
for (int i = 0; i < strArr.Length; i++)
{
strArr[i] = (Convert.ToInt32(strArr[i], 16) + Convert.ToInt32("33", 16)).ToString("X2");
if (strArr[i].Length > 2)
{
strArr[i] = strArr[i].Substring(strArr[i].Length - 2);
}
}
return string.Join(" ", strArr);
}
private static string AddHex33(string strGet)
{
string result;
if (string.IsNullOrEmpty(strGet))
{
result = "";
}
else
{
string[] source = StrAddSpan(strGet).Split(new char[]
{
' '
}, StringSplitOptions.RemoveEmptyEntries);
result = string.Join("", from s in source
select (Convert.ToInt32(s, 16) + Convert.ToInt32("33", 16)).ToString("X2"));
}
return result;
}
}
}

View File

@ -1,8 +1,4 @@
using System;
using System.Collections.Generic;
using System.Text;
namespace JiShe.CollectBus.Protocol.Contracts.Models
namespace JiShe.CollectBus.Common.Enums
{
/// <summary>
/// 规约标识 高位在前 D1+D0

View File

@ -0,0 +1,15 @@
using System;
using System.Collections.Generic;
using System.Text;
namespace JiShe.CollectBus.Common.Enums
{
public enum CommandChunkEnum
{
C = 6,
A = 7,
AFN = 12,
SEQ = 13,
FN = 17,
}
}

View File

@ -0,0 +1,78 @@
using JiShe.CollectBus.Common.Enums;
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Text.RegularExpressions;
using JiShe.CollectBus.Common.Models;
using System.Runtime.CompilerServices;
namespace JiShe.CollectBus.Common.Extensions
{
public static class HexStringExtensions
{
//起始字符
private const string startStr = "68";
//结束字符
private const string endStr = "16";
//头部字节长度
private const int hearderLen = 6;
//消息认证码字段长度
private const int pWLen = 16;
private const int tPLen = 6;
private const int FixedLength = 18;
public static object GetAnalyzeValue(this List<string> hexStringList, CommandChunkEnum chunk)
{
if (hexStringList.Count < hearderLen || hexStringList[0] != startStr || hexStringList[5] != startStr || hexStringList.Count < FixedLength)
{
return null;
}
switch (chunk)
{
case CommandChunkEnum.AFN:
var aFn = hexStringList[(int)CommandChunkEnum.AFN].HexToDec();//1字节
return aFn;
case CommandChunkEnum.FN:
//(DT2*8)+DT1=fn
var dt1Bin = hexStringList[(int)CommandChunkEnum.FN - 1].HexToBin();
var dt1 = dt1Bin != "0" ? dt1Bin.Length : 0;
var dt2 = hexStringList[(int)CommandChunkEnum.FN].HexToDec();
var fn = dt2 * 8 + dt1;
return fn;
case CommandChunkEnum.A:
var aHexList = hexStringList.Skip((int)CommandChunkEnum.A).Take(5).ToList();
var a1 = aHexList[1] + aHexList[0];
var a2 = aHexList[3] + aHexList[2];
var a2Dec = a2.HexToDec();
var a3 = aHexList[4];
var a = $"{a1}{a2Dec.ToString().PadLeft(5, '0')}";
var a3Bin = aHexList[4].HexToBin().PadLeft(8, '0');
var msa = a3Bin.Substring(0, 7).BinToDec();
return new Tuple<string,int>(a, msa);
case CommandChunkEnum.SEQ:
var seq = hexStringList[(int)CommandChunkEnum.SEQ].HexToBin().PadLeft(8, '0');
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);
return new Seq
{
CON = con,
FIRFIN = firfin,
PRSEQ = prseqBin.BinToDec(),
TpV = tpV
};
default:
throw new ArgumentOutOfRangeException(nameof(chunk), chunk, null);
}
}
}
}

View File

@ -0,0 +1,30 @@
using System;
namespace JiShe.CollectBus.Common.Extensions
{
public static class IntExtensions
{
/// <summary>
/// 十进制转二进制
/// </summary>
/// <param name="decimalNumber">十进制数</param>
/// <returns></returns>
public static string DecToBin(this int decimalNumber)
{
var binaryString = Convert.ToString(decimalNumber, 2);
return binaryString;
}
/// <summary>
/// 十进制转十六进制
/// </summary>
/// <param name="decimalNumber"></param>
/// <returns></returns>
public static string DecToHex(this int decimalNumber)
{
var hexString = decimalNumber.ToString("X");
return hexString;
}
}
}

View File

@ -47,38 +47,6 @@ namespace JiShe.CollectBus.Common.Extensions
return string.Copy(str);
}
/// <summary>Gets the phone number.</summary>
/// <param name="str">The string.</param>
/// <param name="pattern">The pattern.</param>
/// <returns>
/// <br />
/// </returns>
[Description("获取单个手机号")]
public static string GetPhoneNumber(this string str, string pattern = RegexConst.PhoneNumber)
{
var reg = new Regex(pattern);
var match = reg.Match(str);
if (match.Success)
{
return match.Value;
}
return null;
}
/// <summary>Gets the phone numbers.</summary>
/// <param name="str">The string.</param>
/// <param name="pattern">The pattern.</param>
/// <returns>
/// <br />
/// </returns>
[Description("获取全部手机号")]
public static List<string> GetPhoneNumbers(this string str, string pattern = RegexConst.PhoneNumber)
{
var reg = new Regex(pattern);
var matches = reg.Matches(str);
return (from Match item in matches select item.Value).ToList();
}
/// <summary>Regex Match the specified pattern.</summary>
/// <param name="str">The string.</param>
/// <param name="pattern">The pattern.</param>
@ -1106,6 +1074,147 @@ namespace JiShe.CollectBus.Common.Extensions
return string.Join(",", list) == intStr;
}
/// <summary>
/// 字符串倒序
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static string StringReversed(this string str)
{
var reversed = new string(str.Reverse().ToArray());
return reversed;
}
/// <summary>
/// 字符串分割成2个字符一组
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static List<string> StringToPairs(this string str)
{
var pairs = str.Select((ch, index) => new { ch, index })
.GroupBy(x => x.index / 2)
.Select(g => string.Concat(g.Select(x => x.ch)))
.ToList();
return pairs;
}
/// <summary>
/// 格式化字符串
/// </summary>
/// <param name="str"></param>
/// <returns></returns>
public static string StrAddSpan(this string str)
{
if (str == "")
{
return "";
}
return Regex.Replace(str.Replace(" ", ""), @"(?<=[0-9A-Za-z]{2})[0-9A-Za-z]{2}", " $0").Trim();
}
/// <summary>
/// 格式化字符串且反转
/// </summary>
/// <param name="ste"></param>
/// <returns></returns>
public static string StrReverseOrder(this string ste)
{
if (ste == "")
{
return "";
}
string[] strArr = ste.Split(new string[] { " " }, System.StringSplitOptions.RemoveEmptyEntries);
return string.Join(" ", strArr.Reverse());
}
/// <summary>
/// 二进制转十六进制
/// </summary>
/// <param name="binaryString"></param>
/// <returns></returns>
public static string BinToHex(this string binaryString)
{
var decimalNumber = Convert.ToInt32(binaryString, 2);// 将二进制字符串转换为整数
var hexString = Convert.ToString(decimalNumber, 16); //decimalNumber.ToString("X"); // 将整数转换为十六进制字符串
return hexString;
}
/// <summary>
/// 二进制转十进制
/// </summary>
/// <param name="binaryString"></param>
/// <returns></returns>
public static int BinToDec(this string binaryString)
{
var decimalNumber = Convert.ToInt32(binaryString, 2);
return decimalNumber;
}
/// <summary>
/// 十六进制转十进制
/// </summary>
/// <param name="hexString"></param>
/// <returns></returns>
public static int HexToDec(this string hexString)
{
var decimalNumber = Convert.ToInt32(hexString, 16);
return decimalNumber;
}
/// <summary>
/// 十六进制转二进制
/// </summary>
/// <param name="hexString"></param>
/// <returns></returns>
public static string HexToBin(this string hexString)
{
var binaryValue = Convert.ToString(Convert.ToInt32(hexString, 16), 2);
return binaryValue;
}
/// <summary>
/// 数据值加33
/// </summary>
public static string StrAddHex33(this string str)
{
if (str == "")
{
return "";
}
var strArr = str.Split(new string[] { " " }, System.StringSplitOptions.RemoveEmptyEntries);
for (int i = 0; i < strArr.Length; i++)
{
strArr[i] = (Convert.ToInt32(strArr[i], 16) + Convert.ToInt32("33", 16)).ToString("X2");
if (strArr[i].Length > 2)
{
strArr[i] = strArr[i].Substring(strArr[i].Length - 2);
}
}
return string.Join(" ", strArr);
}
private static string AddHex33(this string strGet)
{
string result;
if (string.IsNullOrEmpty(strGet))
{
result = "";
}
else
{
string[] source = StrAddSpan(strGet).Split(new char[]
{
' '
}, StringSplitOptions.RemoveEmptyEntries);
result = string.Join("", from s in source
select (Convert.ToInt32(s, 16) + Convert.ToInt32("33", 16)).ToString("X2"));
}
return result;
}
private static void CreateAlphaNumMask(StringBuilder buffer, string source, char mask, int length)
{
for (int i = 0; i < length; i++)

View File

@ -1,9 +1,7 @@
using JiShe.CollectBus.Common;
using System;
using System.Collections.Generic;
using System.Runtime.CompilerServices;
using System.Collections.Generic;
using JiShe.CollectBus.Common.Enums;
namespace JiShe.CollectBus.Protocol.Contracts.Models
namespace JiShe.CollectBus.Common.Models
{
//TODO
public class CommandReulstMsg

View File

@ -5,6 +5,7 @@ using JiShe.CollectBus.Common.Interfaces;
using TouchSocket.Core;
using Microsoft.Extensions.Hosting;
using System;
using Serilog;
// ReSharper disable once CheckNamespace
namespace Microsoft.Extensions.DependencyInjection
@ -22,7 +23,7 @@ namespace Microsoft.Extensions.DependencyInjection
{
if (typeof(IJiSheModule).IsAssignableFrom(type) && type is { IsClass: true, IsAbstract: false })
{
Log.Logger.Information($"正在加载模块{type.Name}...");
var instance = Activator.CreateInstance(type);
_ = (type.GetMethod("ConfigureServices")?.Invoke(instance, [services, hostContext]));
}
@ -90,6 +91,7 @@ namespace Microsoft.Extensions.DependencyInjection
{
var attr = type.GetCustomAttribute<ProtocolNameAttribute>();
if (attr == null) continue;
Log.Logger.Information($"正在加载插件{attr.Name}...");
var serviceDescriptor = new ServiceDescriptor(interfaceType, attr.Name, type, ServiceLifetime.Singleton);
services.Add(serviceDescriptor);
}
@ -102,6 +104,7 @@ namespace Microsoft.Extensions.DependencyInjection
{
var attr = type.GetCustomAttribute<ProtocolNameAttribute>();
if (attr == null) continue;
Log.Logger.Information($"正在加载插件{attr.Name}...");
var serviceDescriptor = new ServiceDescriptor(interfaceType, attr.Name, type, ServiceLifetime.Transient);
services.Add(serviceDescriptor);
}
@ -114,6 +117,7 @@ namespace Microsoft.Extensions.DependencyInjection
{
var attr = type.GetCustomAttribute<ProtocolNameAttribute>();
if (attr == null) continue;
Log.Logger.Information($"正在加载插件{attr.Name}...");
var serviceDescriptor = new ServiceDescriptor(interfaceType, attr.Name, type, ServiceLifetime.Scoped);
services.Add(serviceDescriptor);
}

View File

@ -30,6 +30,12 @@
<PackageReference Include="Microsoft.Extensions.Caching.StackExchangeRedis" Version="8.0.10" />
<PackageReference Include="Microsoft.Extensions.Hosting" Version="8.0.1" />
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="8.0.2" />
<PackageReference Include="Serilog" Version="4.1.0" />
<PackageReference Include="Serilog.AspNetCore" Version="8.0.3" />
<PackageReference Include="Serilog.Extensions.Logging" Version="8.0.0" />
<PackageReference Include="Serilog.Settings.Configuration" Version="8.0.4" />
<PackageReference Include="Serilog.Sinks.Console" Version="6.0.0" />
<PackageReference Include="Serilog.Sinks.File" Version="6.0.0" />
</ItemGroup>
<ItemGroup>

View File

@ -1,6 +1,7 @@
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Serilog;
namespace JiShe.CollectBus.Console
{
@ -9,7 +10,23 @@ namespace JiShe.CollectBus.Console
static void Main(string[] args)
{
CreateHostBuilder(args).Build().Run();
try
{
Log.Logger = new LoggerConfiguration()
.WriteTo.Console()
.CreateLogger();
CreateHostBuilder(args).Build().Run();
}
catch (Exception ex)
{
Log.Fatal(ex, "Application terminated unexpectedly");
}
finally
{
Log.CloseAndFlush();
}
}
public static IHostBuilder CreateHostBuilder(string[] args) =>
@ -29,9 +46,16 @@ namespace JiShe.CollectBus.Console
private static void ConfigureServices(IServiceCollection services, HostBuilderContext hostContext)
{
var configuration = hostContext.Configuration;
services.AddSerilog((context, lc) =>
{
lc.ReadFrom.Configuration(configuration)
.ReadFrom.Services(context);
});
services.ModuleRegister(hostContext);
services.ServiceRegister();
services.PluginServiceRegister();
services.AddTcp(configuration);
//services.AddUdp(configuration);
services.AddStackExchangeRedisCache(options =>

View File

@ -1,4 +1,31 @@
{
"Serilog": {
"Using": [ "Serilog.Sinks.Console", "Serilog.Sinks.File" ],
"MinimumLevel": "Information",
"Override": {
"Microsoft": "Warning",
"System": "Warning"
},
"WriteTo": [
{ "Name": "Console" },
{
"Name": "File",
"Args": {
"path": "Logs/log-.txt",
"rollingInterval": "Day"
//"rollOnFileSizeLimit": true,
//"formatter": "Serilog.Formatting.Compact.CompactJsonFormatter, Serilog.Formatting.Compact"
}
}
],
"Enrich": [ "FromLogContext", "WithMachineName", "WithThreadId" ],
"Properties": {
"Application": "CollectBus",
"Environment": "Development"
}
},
"ConnectionStrings": {
"Default": "Data Source=192.168.111.248;Port=3306;Database=JiSheCollectBus;uid=root;pwd=123456abcD;charset=utf8mb4;Allow User Variables=true;AllowLoadLocalInfile=true",
"ClickHouse": "host=localhost;port=8123;user=default;password=;database=default"

View File

@ -1,10 +1,11 @@
using JiShe.CollectBus.Core.Exceptions;
using Microsoft.Extensions.Logging;
using TouchSocket.Core;
using TouchSocket.Sockets;
namespace JiShe.CollectBus.Core.Plugins
{
public partial class TcpClosePlugin(ILog logger) : PluginBase
public partial class TcpClosePlugin(ILogger<TcpClosePlugin> logger) : PluginBase
{
[GeneratorPlugin(typeof(ITcpReceivedPlugin))]
public async Task OnTcpReceived(ITcpSessionClient client, ReceivedDataEventArgs e)
@ -15,7 +16,7 @@ namespace JiShe.CollectBus.Core.Plugins
}
catch (CloseException ex)
{
logger.Info("拦截到CloseException");
logger.LogInformation("拦截到CloseException");
client.Close(ex.Message);
}
catch (Exception exx)

View File

@ -1,17 +1,23 @@
using JiShe.CollectBus.Protocol.Contracts.Models;
using JiShe.CollectBus.Common.Enums;
using JiShe.CollectBus.Common.Extensions;
using JiShe.CollectBus.Protocol.Contracts.Models;
using TouchSocket.Core;
using TouchSocket.Sockets;
using JiShe.CollectBus.RabbitMQ.Senders;
using Microsoft.Extensions.Logging;
namespace JiShe.CollectBus.Core.Plugins
{
public partial class TcpServiceReceivedPlugin : PluginBase
{
private readonly INSender _nSender;
private readonly ILogger<TcpServiceReceivedPlugin> _logger;
public TcpServiceReceivedPlugin(INSender nSender)
public TcpServiceReceivedPlugin(INSender nSender, ILogger<TcpServiceReceivedPlugin> logger)
{
_nSender = nSender;
_logger = logger;
}
@ -32,29 +38,55 @@ namespace JiShe.CollectBus.Core.Plugins
//client.Logger.Info($"[TCP] 已从{client.GetIPPort()}接收到信息:{messageHexString}");
//await protocolPlugin.ReceivedAsync(client,e);
await _nSender.SendToReceivedAsync(new MessageReceivedEvent
var messageHexString = Convert.ToHexString(e.ByteBlock.Span);
var hexStringList = messageHexString.StringToPairs();
var aFn = (int?)hexStringList.GetAnalyzeValue(CommandChunkEnum.AFN);
var fn = (int?)hexStringList.GetAnalyzeValue(CommandChunkEnum.FN);
if (aFn.HasValue && fn.HasValue)
{
ClientIP = client.IP,
ClientId = client.Id,
MessageHexString = Convert.ToHexString(e.ByteBlock.Span),
Port = client.Port
});
if ((AFN)aFn == AFN.)
{
switch (fn)
{
case 1://登录
await _nSender.SendToReceivedLoginAsync(new MessageReceivedLoginEvent(client.Id,client.IP, client.Port, messageHexString,""));
break;
case 2://退出登录
await _nSender.SendToReceivedLoginAsync(new MessageReceivedLoginEvent(client.Id, client.IP, client.Port, messageHexString, ""));
break;
case 3://心跳
await _nSender.SendToReceivedHeartbeatAsync(new MessageReceivedHeartbeatEvent(client.Id, client.IP, client.Port, messageHexString, ""));
break;
}
}
else
{
await _nSender.SendToReceivedAsync(new MessageReceivedEvent(client.Id, client.IP, client.Port, messageHexString, ""));
}
}
else
{
_logger.LogError($"指令初步解析失败,指令内容:{messageHexString}");
}
await e.InvokeNext();
}
[GeneratorPlugin(typeof(ITcpConnectingPlugin))]
public async Task OnTcpConnecting(ITcpSessionClient client, ConnectingEventArgs e)
public async Task OnTcpConnecting(ITcpSessionClient client,ConnectingEventArgs e)
{
client.Logger.Info($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}正在连接中...");
_logger.LogInformation($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}正在连接中...");
await e.InvokeNext();
}
[GeneratorPlugin(typeof(ITcpConnectedPlugin))]
public async Task OnTcpConnected(ITcpSessionClient client, ConnectedEventArgs e)
public async Task OnTcpConnected(ITcpSessionClient client,ConnectedEventArgs e)
{
client.Logger.Info($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}已连接");
_logger.LogInformation($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}已连接");
await e.InvokeNext();
}
@ -62,7 +94,7 @@ namespace JiShe.CollectBus.Core.Plugins
[GeneratorPlugin(typeof(ITcpClosedPlugin))]
public async Task OnTcpClosed(ITcpSessionClient client, ClosedEventArgs e)
{
client.Logger.Info($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}已关闭连接");
_logger.LogInformation($"[TCP] ID:{client.Id} IP:{client.GetIPPort()}已关闭连接");
await e.InvokeNext();
}

View File

@ -1,4 +1,5 @@
using System;
using Microsoft.Extensions.Logging;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
@ -8,7 +9,7 @@ using TouchSocket.Sockets;
namespace JiShe.CollectBus.Core.Services
{
public partial class BusService : PluginBase
public partial class BusService(ILogger<BusService> logger) : PluginBase
{
[GeneratorPlugin(typeof(IServerStartedPlugin))]
public Task OnServerStarted(IServiceBase sender, ServiceStateEventArgs e)
@ -19,26 +20,26 @@ namespace JiShe.CollectBus.Core.Services
{
foreach (var item in service.Monitors)
{
ConsoleLogger.Default.Info($"TCP {item.Option.IpHost}");
logger.LogInformation($"TCP {item.Option.IpHost}");
}
break;
}
case UdpSession session:
ConsoleLogger.Default.Info($"UDP {session.Monitor.IPHost}");
logger.LogInformation($"UDP {session.Monitor.IPHost}");
break;
}
ConsoleLogger.Default.Info(e.ServerState == ServerState.Running
logger.LogInformation(e.ServerState == ServerState.Running
? $"服务器成功启动"
: $"服务器启动失败,状态:{e.ServerState},异常:{e.Exception}");
return e.InvokeNext();
}
[GeneratorPlugin(typeof(IServerStopedPlugin))]
public Task OnServerStoped(IServiceBase sender, ServiceStateEventArgs e)
public Task OnServerStoped(IServiceBase sender,ServiceStateEventArgs e)
{
Console.WriteLine("服务已停止");
logger.LogInformation("服务已停止");
return e.InvokeNext();
}
}

View File

@ -1,13 +1,343 @@
using System.Threading.Tasks;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using JiShe.CollectBus.Common;
using JiShe.CollectBus.Common.Enums;
using JiShe.CollectBus.Common.Extensions;
using JiShe.CollectBus.Common.Models;
using JiShe.CollectBus.Protocol.Contracts.Interfaces;
using JiShe.CollectBus.Protocol.Contracts.Models;
using Microsoft.Extensions.Logging;
namespace JiShe.CollectBus.Protocol.Contracts.Abstracts
{
public abstract class BaseProtocolPlugin : IProtocolPlugin
{
public readonly ILogger<BaseProtocolPlugin> _logger;
//起始字符
public const string stx = "68";
//结束字符
public const string end = "16";
//头部字节长度
public const int hearderLen = 6;
//消息认证码字段长度
public const int pWLen = 16;
public const int tPLen = 6;
protected BaseProtocolPlugin(ILogger<BaseProtocolPlugin> logger)
{
_logger = logger;
}
public abstract Task<ProtocolInfo> GetAsync();
public abstract Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent);
public abstract Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent, Action<byte[]>? sendAction = null);
/// <summary>
/// 登录帧解析
/// </summary>
/// <param name="messageReceivedEvent">报文</param>
/// <param name="sendAction">发送委托</param>
/// <returns></returns>
public virtual Task LoginAsync(MessageReceivedLoginEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
var hexStringList = messageReceivedEvent.MessageHexString.StringToPairs();
var aTuple = (Tuple<string, int>)hexStringList.GetAnalyzeValue(CommandChunkEnum.A);
var seq = (Seq)hexStringList.GetAnalyzeValue(CommandChunkEnum.SEQ);
var reqParam = new ReqParameter2
{
AFN = AFN.,
FunCode = (int)CFromStationFunCode.,
PRM = PRM.,
A = aTuple.Item1,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = seq.PRSEQ
},
MSA = aTuple.Item2,
Pn = 0,
Fn = 1
};
var bytes = GetCommandBytes(reqParam);
if (sendAction != null)
{
sendAction(bytes);
}
return Task.CompletedTask;
}
/// <summary>
/// 心跳帧解析
/// </summary>
/// <param name="messageReceivedEvent">报文</param>
/// <param name="sendAction">发送委托</param>
/// <returns></returns>
public virtual Task HeartbeatAsync(MessageReceivedHeartbeatEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
var hexStringList = messageReceivedEvent.MessageHexString.StringToPairs();
var aTuple = (Tuple<string,int>)hexStringList.GetAnalyzeValue(CommandChunkEnum.A);
var seq = (Seq)hexStringList.GetAnalyzeValue(CommandChunkEnum.SEQ);
if (seq.TpV == TpV.)
{
//解析
}
if (seq.CON == CON.)
{
var reqParam = new ReqParameter2()
{
AFN = AFN.,
FunCode = (int)CFromStationFunCode.,
PRM = PRM.,
A = aTuple.Item1,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = seq.PRSEQ,
},
MSA = aTuple.Item2,
Pn = 0,
Fn = 1
};
var bytes = GetCommandBytes(reqParam);
if (sendAction != null)
{
sendAction(bytes);
}
}
return Task.CompletedTask;
}
#region
public byte[] GetCommandBytes(ReqParameter reqParameter, List<string>? dataUnit = null)
{
var cmdStrList = new List<string>();
var userDatas = GetUserData(reqParameter, dataUnit);
var hearders = GetHeaders(userDatas.Count);
var cs = GetCS(userDatas);
cmdStrList.AddRange(hearders);
cmdStrList.AddRange(userDatas);
cmdStrList.Add(cs);
cmdStrList.Add(end);
_logger.LogInformation($"回复:{string.Join(" ", cmdStrList)}");
var bytes = cmdStrList.Select(x => Convert.ToByte(x, 16)).ToArray();
return bytes;
}
/// <summary>
/// 帧校验和
/// </summary>
/// <param name="userData">用户数据区</param>
/// <returns></returns>
public string GetCS(List<string> userData)
{
byte sum = 0;
foreach (var d in userData)
{
var b = Convert.ToByte(d, 16);
sum += b;
}
return sum.ToString("X2");
}
/// <summary>
/// 用户数据区
/// </summary>
/// <param name="reqParameter"></param>
/// <returns></returns>
public List<string> GetUserData(ReqParameter reqParameter, List<string>? dataUnit)
{
var c = GetC(reqParameter.FunCode, reqParameter.PRM);
var a = GetAList(reqParameter.A, reqParameter.MSA);
var linkUserData = GetLinkUserData(reqParameter.AFN, reqParameter.Seq,
((ReqParameter2)reqParameter).Pn, ((ReqParameter2)reqParameter).Fn, dataUnit);
var list = new List<string>() { c };
list.AddRange(a);
list.AddRange(linkUserData);
return list;
}
/// <summary>
/// 固定长度的报文头 起始字符+长度+长度+起始字符
/// </summary>
/// <param name="length"></param>
/// <returns></returns>
public List<string> GetHeaders(int length)
{
var headers = new List<string>();
headers.Add(stx);
var l = GetLength(length);
headers.AddRange(l);
headers.AddRange(l);
headers.Add(stx);
return headers;
}
/// <summary>
/// 长度 2字节 [用户数据区长度]
/// </summary>
/// <returns></returns>
public List<string> GetLength(int length1)
{
var binaryLen = length1.DecToBin();
var protocolIdentification = Enum.Format(typeof(ProtocolIdentification),
ProtocolIdentification.使, "d").PadLeft(2, '0');
var lenStr = $"{binaryLen}{protocolIdentification}";
var hexLen = lenStr.BinToHex();
hexLen = hexLen.PadLeft(4, '0');
var list = hexLen.StringToPairs();
list.Reverse();
return list;
}
/// <summary>
/// 控制域
/// </summary>
/// <param name="funCode">功能码</param>
/// <param name="fcb"></param>
/// <param name="fcv"></param>
/// <returns></returns>
public string GetC(int funCode, PRM pRM, int fcb = 0, FCV fcv = FCV.FCB位无效)
{
var cMasterStationFunCodeHex = funCode.DecToBin();
cMasterStationFunCodeHex = cMasterStationFunCodeHex.ToString().PadLeft(4, '0');
var strC = $"{(int)DIR.主站下行报文}{(int)pRM}{fcb}{(int)fcv}{cMasterStationFunCodeHex}";
var hexC = strC.BinToHex().PadLeft(2, '0');
return hexC;
}
/// <summary>
/// 地址域 3220 09872
/// </summary>
/// <param name="a1">行政区划码 BCD码 3220=2032</param>
/// <param name="a2">逻辑地址 BIN 09872=2690=>9026</param>
/// <param name="a3">主站地址 BIN 0~127</param>
/// <returns></returns>
public List<string> GetAList(string a, int mSA)
{
var list = new List<string>();
var a1 = a.Substring(0, 4);
var a1Pairs = a1.StringToPairs();
a1Pairs.Reverse();
list.AddRange(a1Pairs);
var a2 = Convert.ToInt32(a.Substring(4));
var decA2 = a2.DecToHex();
var a2Pairs = decA2.PadLeft(4, '0').StringToPairs();
a2Pairs.Reverse();
list.AddRange(a2Pairs);
//TODO:主站地址和组地址标志
var a3Bin = $"{mSA.DecToBin().PadLeft(7, '0')}0";
list.Add(a3Bin.BinToHex().PadLeft(2, '0'));
return list;
}
public List<string> GetLinkUserData(AFN aFN, Seq seq, int pn, int fn, List<string>? dataUnit)
{
var aFNValue = ((int)aFN).DecToHex().PadLeft(2, '0');
var sEQ = GetSEQ(seq.TpV, seq.FIRFIN, seq.CON, seq.PRSEQ);
var dA = GetDA(pn);
var dT = GetDT(fn);
var list = new List<string>() { aFNValue, sEQ };
list.AddRange(dA);
list.AddRange(dT);
if (dataUnit != null)
{
list.AddRange(dataUnit);
}
//list.AddRange(GetDataUnit(aFN,seq));
if (seq.TpV == TpV.)
list.AddRange(GetTp("00"));
return list;
}
/// <summary>
/// 帧序列域
/// </summary>
/// <param name="tpV"></param>
/// <param name="fIRFIN"></param>
/// <param name="cON"></param>
/// <returns></returns>
public string GetSEQ(TpV tpV, FIRFIN fIRFIN, CON cON, int pRSEQ)
{
var tpVValue = Enum.Format(typeof(TpV),
tpV, "d");
var fIRFINValue = Enum.Format(typeof(FIRFIN),
fIRFIN, "d");
var cONValue = (int)cON;
var sEQBin = $"{tpVValue}{fIRFINValue}{cONValue}{pRSEQ.DecToBin().PadLeft(4, '0')}";
var hexSEQ = sEQBin.BinToHex().PadLeft(2, '0');
return hexSEQ;
}
/// <summary>
/// 信息点标识
/// </summary>
/// <param name="pn">计量点</param>
/// <returns></returns>
public List<string> GetDA(int pn)
{
if (pn == 0)
return new List<string>() { "00", "00" };
var dA2 = (pn - 1) / 8 + 1;//信息点组从1开始 第几组
var dA1 = pn - (dA2 - 1) * 8;//pn % 8
var dA1Hex = "1".PadRight(dA1, '0').BinToHex();//对位信息 第几位 二进制有效位
var dA2Hex = dA2.DecToHex();
return new List<string>() { dA1Hex.PadLeft(2, '0'), dA2Hex.PadLeft(2, '0') };
}
/// <summary>
/// 数据单元标识
/// </summary>
/// <param name="fn"></param>
/// <returns></returns>
public List<string> GetDT(int fn)
{
var dT2 = (fn - 1) / 8;//从零开始 第几组
var dT1 = fn - dT2 * 8;
var dT1Hex = "1".PadRight(dT1, '0').BinToHex();//对位信息 第几位 二进制有效位
var dT2Hex = dT2.DecToHex();
return new List<string>() { dT1Hex.PadLeft(2, '0'), dT2Hex.PadLeft(2, '0') };
}
/// <summary>
/// 时间标签
/// </summary>
/// <param name="pFC">启动帧帧序号计数器PFC 1字节</param>
/// <param name="delayTime">允许发送传输延时时间 min 1字节</param>
/// <returns></returns>
public List<string> GetTp(string pFC = "00", int delayTime = 0)
{
var now = DateTime.Now; // 获取当前时间
var seconds = now.Second.ToString().PadLeft(2, '0'); // 获取当前秒数
var minutes = now.Minute.ToString().PadLeft(2, '0'); // 获取当前分钟数
var hours = now.Hour.ToString().PadLeft(2, '0'); // 获取当前小时数
var day = now.Day.ToString().PadLeft(2, '0'); // 获取当前日期的日数
return new List<string>() { pFC, seconds, minutes, hours, day, delayTime.ToString().PadLeft(2, '0') };
}
#endregion
}
}

View File

@ -1,4 +1,5 @@
using System.Threading.Tasks;
using System;
using System.Threading.Tasks;
using JiShe.CollectBus.Protocol.Contracts.Models;
using TouchSocket.Sockets;
@ -8,6 +9,10 @@ namespace JiShe.CollectBus.Protocol.Contracts.Interfaces
{
Task<ProtocolInfo> GetAsync();
Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent);
Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent, Action<byte[]>? sendAction = null);
Task LoginAsync(MessageReceivedLoginEvent messageReceivedEvent, Action<byte[]>? sendAction = null);
Task HeartbeatAsync(MessageReceivedHeartbeatEvent messageReceivedEvent, Action<byte[]>? sendAction = null);
}
}

View File

@ -2,6 +2,15 @@
{
public class MessageReceivedEvent
{
public MessageReceivedEvent(string clientId, string clientIp,int port, string messageHexString, string deviceNo)
{
ClientId = clientId;
ClientIP = clientIp;
Port = port;
MessageHexString = messageHexString;
DeviceNo = deviceNo;
}
/// <summary>
/// 客服端标识
/// </summary>
@ -27,4 +36,18 @@
/// </summary>
public string DeviceNo { get; set; }
}
public class MessageReceivedLoginEvent: MessageReceivedEvent
{
public MessageReceivedLoginEvent(string clientId, string clientIp, int port, string messageHexString, string deviceNo) : base(clientId, clientIp, port, messageHexString, deviceNo)
{
}
}
public class MessageReceivedHeartbeatEvent : MessageReceivedEvent
{
public MessageReceivedHeartbeatEvent(string clientId, string clientIp, int port, string messageHexString, string deviceNo) : base(clientId, clientIp, port, messageHexString, deviceNo)
{
}
}
}

View File

@ -3,6 +3,8 @@ using System.Collections.Generic;
using System.Net.Sockets;
using System.Text;
using JiShe.CollectBus.Common;
using JiShe.CollectBus.Common.Enums;
using JiShe.CollectBus.Common.Models;
namespace JiShe.CollectBus.Protocol.Contracts.Models
{

View File

@ -3,6 +3,7 @@ using JiShe.CollectBus.Protocol.Contracts.Abstracts;
using JiShe.CollectBus.Protocol.Contracts.Attributes;
using JiShe.CollectBus.Protocol.Contracts.Models;
using Microsoft.Extensions.Caching.Distributed;
using Microsoft.Extensions.Logging;
using TouchSocket.Sockets;
namespace JiShe.CollectBus.Protocol.Test
@ -10,12 +11,16 @@ namespace JiShe.CollectBus.Protocol.Test
[ProtocolName("TestProtocol")]
public class TestProtocolPlugin : BaseProtocolPlugin, ISingletonDependency
{
public TestProtocolPlugin(ILogger<BaseProtocolPlugin> logger) : base(logger)
{
}
public override Task<ProtocolInfo> GetAsync()
{
throw new NotImplementedException();
}
public override Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent)
public override Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
throw new NotImplementedException();
}

View File

@ -1,53 +1,29 @@
using JiShe.CollectBus.Common;
using JiShe.CollectBus.Common.Enums;
using JiShe.CollectBus.Common.Extensions;
using JiShe.CollectBus.Common.Extensions.DependencyInjections;
using JiShe.CollectBus.Common.Models;
using JiShe.CollectBus.Protocol.Contracts.Abstracts;
using JiShe.CollectBus.Protocol.Contracts.Attributes;
using JiShe.CollectBus.Protocol.Contracts.Models;
using JiShe.CollectBus.RabbitMQ.Senders;
using Microsoft.Extensions.Caching.Distributed;
using Microsoft.Extensions.Logging;
using TouchSocket.Sockets;
namespace JiShe.CollectBus.Protocol
{
[ProtocolName("StandardProtocol")]
public class StandardProtocolPlugin(INSender nSender) : BaseProtocolPlugin, ISingletonDependency
public class StandardProtocolPlugin(INSender nSender, ILogger<BaseProtocolPlugin> logger) : BaseProtocolPlugin(logger), ISingletonDependency
{
//起始字符
private const string stx = "68";
//结束字符
private const string end = "16";
//头部字节长度
private const int hearderLen = 6;
//消息认证码字段长度
private const int pWLen = 16;
private const int tPLen = 6;
static object locker = new object();
static List<int> MSA = new List<int>();
static Dictionary<string, List<int>> usingMSA = new Dictionary<string, List<int>>();
private string clientId = "";
static StandardProtocolPlugin()
{
for (int i = 1; i <= 127; i++)
{
MSA.Add(i);
}
}
public override async Task<ProtocolInfo> GetAsync()
{
var info = new ProtocolInfo("Standard", "376.1", "TCP", "376.1协议", "DTS1980");
return await Task.FromResult(info);
}
public override async Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent)
public override async Task AnalyzeAsync(MessageReceivedEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
clientId = messageReceivedEvent.ClientId;
var cmdResult = AnalysisCmd(messageReceivedEvent.MessageHexString);
if (cmdResult == null)
{
@ -57,29 +33,32 @@ namespace JiShe.CollectBus.Protocol
await Task.CompletedTask;
}
/// <summary>
/// Gets the msa.
/// </summary>
/// <param name="mark">The mark.</param>
/// <returns></returns>
public static int GetMSA(string mark)
public override async Task LoginAsync(MessageReceivedLoginEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
lock (locker)
async void SendAction(byte[] bytes)
{
if (!usingMSA.Keys.Contains(mark))
usingMSA.Add(mark, new List<int>());
int msa = MSA.Except(usingMSA[mark]).FirstOrDefault();
//if (msa == 1) msa = 2;//msa=1为自定义指令保留
usingMSA[mark].Add(msa);
if (msa == 127)
usingMSA[mark].RemoveAll(m => true);
return msa;
await nSender.SendToIssuedAsync(new MessageIssuedEvent { ClientId = messageReceivedEvent.ClientId, DeviceNo = messageReceivedEvent.DeviceNo, Message = bytes });
}
await base.LoginAsync(messageReceivedEvent, SendAction);
}
public override async Task HeartbeatAsync(MessageReceivedHeartbeatEvent messageReceivedEvent, Action<byte[]>? sendAction = null)
{
async void SendAction(byte[] bytes)
{
await nSender.SendToIssuedAsync(new MessageIssuedEvent { ClientId = messageReceivedEvent.ClientId, DeviceNo = messageReceivedEvent.DeviceNo, Message = bytes });
}
await base.HeartbeatAsync(messageReceivedEvent, SendAction);
}
#region
//68
@ -112,7 +91,7 @@ namespace JiShe.CollectBus.Protocol
public CommandReulst? AnalysisCmd(string cmd)
{
CommandReulst? commandReulst = null;
var hexStringList = DataConvert.StringToPairs(cmd);
var hexStringList = cmd.StringToPairs();
if (hexStringList.Count < hearderLen)
{
@ -125,8 +104,8 @@ namespace JiShe.CollectBus.Protocol
}
var lenHexStr = $"{hexStringList[2]}{hexStringList[1]}";
var lenBin = DataConvert.HexToBin(lenHexStr);
var len = DataConvert.BinToDec(lenBin.Remove(lenBin.Length - 2));
var lenBin = lenHexStr.HexToBin();
var len = lenBin.Remove(lenBin.Length - 2).BinToDec();
//验证长度
if (hexStringList.Count - 2 != hearderLen + len)
return commandReulst;
@ -137,14 +116,14 @@ namespace JiShe.CollectBus.Protocol
var aHexList = hexStringList.Skip(userDataIndex).Take(5).ToList();//地址域 5字节
var a = AnalysisA(aHexList);
var a3Bin = DataConvert.HexToBin(aHexList[4]).PadLeft(8, '0');
var mSA = DataConvert.BinToDec(a3Bin.Substring(0, 7));
var a3Bin = aHexList[4].HexToBin().PadLeft(8, '0');
var mSA = a3Bin.Substring(0, 7).BinToDec();
userDataIndex += 5;
var aFN = (AFN)DataConvert.HexToDec(hexStringList[userDataIndex]);//1字节
var aFN = (AFN)hexStringList[userDataIndex].HexToDec();//1字节
userDataIndex += 1;
var seq = DataConvert.HexToBin(hexStringList[userDataIndex]).PadLeft(8, '0');
var seq = hexStringList[userDataIndex].HexToBin().PadLeft(8, '0');
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));
@ -152,20 +131,19 @@ namespace JiShe.CollectBus.Protocol
userDataIndex += 1;
// (DA2 - 1) * 8 + DA1 = pn
var da1Bin = DataConvert.HexToBin(hexStringList[userDataIndex]);
var da1Bin = hexStringList[userDataIndex].HexToBin();
var da1 = da1Bin == "0" ? 0 : da1Bin.Length;
userDataIndex += 1;
var da2 = DataConvert.HexToDec(hexStringList[userDataIndex]);
userDataIndex += 1;
var da2 = hexStringList[userDataIndex].HexToDec();
var pn = da2 == 0 ? 0 : (da2 - 1) * 8 + da1;
userDataIndex += 1;
//(DT2*8)+DT1=fn
var dt1Bin = DataConvert.HexToBin(hexStringList[userDataIndex]);
var dt1Bin = hexStringList[userDataIndex].HexToBin();
var dt1 = dt1Bin != "0" ? dt1Bin.Length : 0;
userDataIndex += 1;
var dt2 = DataConvert.HexToDec(hexStringList[userDataIndex]);
userDataIndex += 1;
var dt2 = hexStringList[userDataIndex].HexToDec();
var fn = dt2 * 8 + dt1;
userDataIndex += 1;
//数据单元
var datas = hexStringList.Skip(userDataIndex).Take(len + hearderLen - userDataIndex).ToList();
@ -182,7 +160,7 @@ namespace JiShe.CollectBus.Protocol
TpV = tpV,
FIRFIN = fIRFIN,
CON = cON,
PRSEQ = DataConvert.BinToDec(prseqBin),
PRSEQ = prseqBin.BinToDec(),
},
CmdLength = len,
Pn = pn,
@ -202,7 +180,7 @@ namespace JiShe.CollectBus.Protocol
{
var a1 = aHexList[1] + aHexList[0];
var a2 = aHexList[3] + aHexList[2];
var a2Dec = DataConvert.HexToDec(a2);
var a2Dec = a2.HexToDec();
var a3 = aHexList[4];
var a = $"{a1}{a2Dec.ToString().PadLeft(5, '0')}";
return a;
@ -285,12 +263,6 @@ namespace JiShe.CollectBus.Protocol
Fn = 1
};
commandReulst.ReplyBytes = GetCommandBytes(reqParam);
nSender.SendToIssuedAsync(new MessageIssuedEvent
{
ClientId = clientId,
DeviceNo = "",
Message = commandReulst.ReplyBytes
});
}
else if (commandReulst.Fn == 2)//退出登录
{
@ -332,681 +304,6 @@ namespace JiShe.CollectBus.Protocol
commandReulst.ReplyBytes = GetCommandBytes(reqParam);
}
}
/// <summary>
/// 解析时间标签
/// </summary>
/// <param name="hexDatas"></param>
private void AnalysisTp(List<string> hexDatas)
{
var pFC = DataConvert.HexToDec(hexDatas[0]);//启动帧帧序号计数器
var seconds = Convert.ToInt32(hexDatas[1]); // 获取当前秒数
var minutes = Convert.ToInt32(hexDatas[2]); // 获取当前分钟数
var hours = Convert.ToInt32(hexDatas[3]); // 获取当前小时数
var day = Convert.ToInt32(hexDatas[4]); // 获取当前日期的日数
var delayTime = DataConvert.HexToDec(hexDatas[5]);//延迟时间 min
}
/// <summary>
/// 解析电表档案
/// </summary>
/// <param name="hexDatas"></param>
/// <returns></returns>
public List<MeterParameter> AnalysisAFN04F10DataUnit(List<string> hexDatas)
{
var meterList = new List<MeterParameter>();
var count = DataConvert.HexToDec($"{hexDatas[1]}{hexDatas[0]}");
//if (2 + count * 27 != hexDatas.Count - pWLen - tPLen - 2)
// return;
var index = 2;//数量
for (int i = 1; i <= count; i++)
{
var meterNumber = DataConvert.HexToDec($"{hexDatas[index + 1]}{hexDatas[index]}");
index += 2;
var pn = DataConvert.HexToDec($"{hexDatas[index + 1]}{hexDatas[index]}");
index += 2;
var baudRateAndPortBin = DataConvert.HexToBin(hexDatas[index]).PadLeft(8, '0');
var baudRate = DataConvert.BinToDec(baudRateAndPortBin.Substring(0, 3));
var port = DataConvert.BinToDec(baudRateAndPortBin.Substring(3, 5));
index += 1;
var protocolType = (CommunicationProtocolType)DataConvert.HexToDec(hexDatas[index]);
index += 1;
var addressHexList = hexDatas.Skip(index).Take(6).ToList();
addressHexList.Reverse();
var address = string.Join("", addressHexList);
index += 6;
var pwdHexList = hexDatas.Skip(index).Take(6).ToList();
pwdHexList.Reverse();
var password = string.Join("", pwdHexList.Take(3).ToList());
index += 6;
var rateNumberBin = DataConvert.HexToBin(hexDatas[index]).PadLeft(8, '0');
var rateNumber = DataConvert.BinToDec(rateNumberBin.Substring(4));
index += 1;
var intBitAndDecBitNumberBin = DataConvert.HexToBin(hexDatas[index]).PadLeft(8, '0');
var intBitNumber = DataConvert.BinToDec(intBitAndDecBitNumberBin.Substring(4, 2)) + 4;
var decBitNumber = DataConvert.BinToDec(intBitAndDecBitNumberBin.Substring(6, 2)) + 1;
index += 1;
// hexDatas.GetRange()
var collectorAddressHexList = hexDatas.Skip(index).Take(6).ToList();
collectorAddressHexList.Reverse();
var collectorAddress = string.Join("", collectorAddressHexList);
index += 6;
var userClassNumberBin = DataConvert.HexToBin(hexDatas[index]).PadLeft(8, '0');
var userClass = DataConvert.BinToDec(userClassNumberBin.Substring(0, 4));
var userSubClass = DataConvert.BinToDec(userClassNumberBin.Substring(4, 4));
index += 1;
meterList.Add(new MeterParameter()
{
Pn = pn,
BaudRate = baudRate,
Port = port,
ProtocolType = protocolType,
Address = address,
Password = password,
RateNumber = rateNumber,
IntegerBitNumber = intBitNumber,
DecimalBitNumber = decBitNumber,
CollectorAddress = collectorAddress,
UserCategoryNumber = userClass,
UserSubclassNumber = userSubClass,
});
}
return meterList;
}
/// <summary>
/// 解析实时数据F129
/// </summary>
/// <param name="hexDatas"></param>
private void AnalysisAFN0CF129DataUnit(List<string> hexDatas)
{
var minutes = Convert.ToInt32(hexDatas[0]); // 获取当前分钟数
var hours = Convert.ToInt32(hexDatas[1]); // 获取当前小时数
var day = Convert.ToInt32(hexDatas[2]); // 获取当前日期的日数
var month = Convert.ToInt32(hexDatas[3]); // 获取当前月份
var year = Convert.ToInt32(hexDatas[4]); // 获取当前日期的年份
var rateNumber = Convert.ToInt32(hexDatas[5]);
var kwhTotal = hexDatas.Skip(5).Take(5).ToList();
var kwhList = new List<decimal>();
var index = 11;
for (int i = 0; i < rateNumber; i++)
{
var kwhHexList = hexDatas.Skip(11).Take(5).ToList();
kwhHexList.Reverse();
var integerStr = $"{kwhHexList.Take(0)}{kwhHexList.Take(1)}{kwhHexList.Take(2)}";
var decimalValStr = $"{kwhHexList[3]}{kwhHexList[4]}";
var val = decimal.Parse($"{integerStr}{decimalValStr}");
kwhList.Add(val);
index += 5;
}
}
#endregion
#region
/// <summary>
/// 设置电表档案
/// </summary>
/// <param name="a">集中器地址</param>
/// <param name="meterParameters"></param>
public void GetSetAmmeterParameter(string a, List<MeterParameter> meterParameters)
{
var dataUnit = GetAFN04F10DataUnit(meterParameters);
var bytes = GetCommandBytes(new ReqParameter2()
{
AFN = AFN.,
FunCode = (int)CMasterStationFunCode.1,
A = a,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = 10,
},
MSA = GetMSA(a),
Pn = 0,
Fn = 10
}, dataUnit);
}
/// <summary>
/// 查询电表档案
/// </summary>
/// <param name="a"></param>
/// <param name="meterNumberList">对象序号</param>
public void GetAmmeterParameter(string a, List<int> meterNumberList)
{
var dataUnit = new List<string>();
var countHex = DataConvert.DecToHex(meterNumberList.Count()).PadLeft(4, '0');
var countHexPairs = DataConvert.StringToPairs(countHex);
countHexPairs.Reverse();
dataUnit.AddRange(countHexPairs);
foreach (var number in meterNumberList)
{
var numberHex = DataConvert.DecToHex(number).PadLeft(4, '0');
var numberHexPairs = DataConvert.StringToPairs(numberHex);
numberHexPairs.Reverse();
dataUnit.AddRange(numberHexPairs);
}
var bytes = GetCommandBytes(new ReqParameter2()
{
AFN = AFN.,
FunCode = (int)CMasterStationFunCode.2,
A = a,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = 0,
},
MSA = GetMSA(a),
Pn = 0,
Fn = 10
}, dataUnit);
}
/// <summary>
/// 电表抄读
/// </summary>
/// <param name="a"></param>
/// <param name="pn"></param>
public void GetAmmterReading(string a, int pn)
{
var bytes = GetCommandBytes(new ReqParameter2()
{
AFN = AFN.,
FunCode = (int)CMasterStationFunCode.2,
A = a,
Seq = new Seq()
{
TpV = TpV.,
FIRFIN = FIRFIN.,
CON = CON.,
PRSEQ = 2,
},
MSA = GetMSA(a),
Pn = pn,
Fn = 129
});
}
/// <summary>
/// 组装电表阀控
/// </summary>
/// <param name="address">电表地址</param>
/// <param name="specialnocode">特殊控制码</param>
/// <param name="password">密码</param>
/// <param name="state">是否为开阀</param>
/// <returns></returns>
public List<string> AmmeterValveControl(string address, string specialnocode, string password, bool state, string modelCode = "")
{
address = address.Trim().TrimStart('0');
if (address.Length < 12) address = address.PadLeft(12, '0');
string Code = string.Empty;
if (state)
{
if (string.IsNullOrEmpty(specialnocode))
Code = "1B";
else
Code = specialnocode == "1B" || specialnocode == "1C" ? specialnocode : "1C";
}
else
Code = "1A";//跳闸
if (specialnocode == "1W")
{
if (state)
Code = "1A";
else
Code = "1C";
}
var pwdLevel = "02";
if (modelCode == "HL_DTSU2625" || modelCode == "DDZY9866")
pwdLevel = "04";
else if (modelCode == "DDS2705")
pwdLevel = "03";
if (!string.IsNullOrWhiteSpace(password) && password.Contains("|"))
{
var sp = password.Split('|');
pwdLevel = sp[1];
password = sp[0];
}
string strDate = DataConvert.StrAddSpan(DateTime.Now.AddYears(3).ToString("000012ddMMyy"));
if (specialnocode == "1D" || modelCode == "SZBD_DDZY1225")
strDate = "FF FF FF FF FF FF";
string strP = DataConvert.StrReverseOrder(DataConvert.StrAddSpan(password));
string strSJY = " " + pwdLevel + " " + strP + " 01 00 00 00 " + Code + " 00 " + strDate;
string strLen = (strSJY.Replace(" ", "").Length / 2).ToString("X2");
string strReturn = "68 " + DataConvert.StrReverseOrder(DataConvert.StrAddSpan(address)) + " 68 1C " + strLen + " " + DataConvert.StrAddHex33(strSJY) + " ";
string strSum = strReturn.Split(new string[] { " " }, StringSplitOptions.RemoveEmptyEntries).Select(i => Convert.ToInt32(i, 16)).Sum().ToString("X");
strReturn += strSum.Substring(strSum.Length - 2) + " 16";
return strReturn.Split(' ').ToList();
}
/// <summary>
/// 帧命令组装
/// </summary>
/// <param name="reqParameter">请求参数</param>
/// <param name="dataUnit">数据单元</param>
/// <returns></returns>
public byte[] GetCommandBytes(ReqParameter reqParameter, List<string>? dataUnit = null)
{
var cmdStrList = new List<string>();
var userDatas = GetUserData(reqParameter, dataUnit);
var hearders = GetHeaders(userDatas.Count);
var cs = GetCS(userDatas);
cmdStrList.AddRange(hearders);
cmdStrList.AddRange(userDatas);
cmdStrList.Add(cs);
cmdStrList.Add(end);
Console.WriteLine($"回复:{string.Join(" ", cmdStrList)}");
var bytes = cmdStrList.Select(x => Convert.ToByte(x, 16)).ToArray();
return bytes;
}
/// <summary>
/// 固定长度的报文头 起始字符+长度+长度+起始字符
/// </summary>
/// <param name="length"></param>
/// <returns></returns>
private List<string> GetHeaders(int length)
{
var headers = new List<string>();
headers.Add(stx);
var l = GetLength(length);
headers.AddRange(l);
headers.AddRange(l);
headers.Add(stx);
return headers;
}
/// <summary>
/// 用户数据区
/// </summary>
/// <param name="reqParameter"></param>
/// <returns></returns>
public List<string> GetUserData(ReqParameter reqParameter, List<string>? dataUnit)
{
var c = GetC(reqParameter.FunCode, reqParameter.PRM);
var a = GetAList(reqParameter.A, reqParameter.MSA);
var linkUserData = GetLinkUserData(reqParameter.AFN, reqParameter.Seq,
((ReqParameter2)reqParameter).Pn, ((ReqParameter2)reqParameter).Fn, dataUnit);
var list = new List<string>() { c };
list.AddRange(a);
list.AddRange(linkUserData);
return list;
}
/// <summary>
/// 长度 2字节 [用户数据区长度]
/// </summary>
/// <returns></returns>
private List<string> GetLength(int length1)
{
var binaryLen = DataConvert.DecToBin(length1);
var protocolIdentification = Enum.Format(typeof(ProtocolIdentification),
ProtocolIdentification.使, "d").PadLeft(2, '0');
var lenStr = $"{binaryLen}{protocolIdentification}";
var hexLen = DataConvert.BinToHex(lenStr);
hexLen = hexLen.PadLeft(4, '0');
var list = DataConvert.StringToPairs(hexLen);
list.Reverse();
return list;
}
/// <summary>
/// 控制域
/// </summary>
/// <param name="funCode">功能码</param>
/// <param name="fcb"></param>
/// <param name="fcv"></param>
/// <returns></returns>
private string GetC(int funCode, PRM pRM, int fcb = 0, FCV fcv = FCV.FCB位无效)
{
var cMasterStationFunCodeHex = DataConvert.DecToBin(funCode);
cMasterStationFunCodeHex = cMasterStationFunCodeHex.ToString().PadLeft(4, '0');
var strC = $"{(int)DIR.主站下行报文}{(int)pRM}{fcb}{(int)fcv}{cMasterStationFunCodeHex}";
var hexC = DataConvert.BinToHex(strC).PadLeft(2, '0');
return hexC;
}
/// <summary>
/// 地址域 3220 09872
/// </summary>
/// <param name="a1">行政区划码 BCD码 3220=2032</param>
/// <param name="a2">逻辑地址 BIN 09872=2690=>9026</param>
/// <param name="a3">主站地址 BIN 0~127</param>
/// <returns></returns>
private List<string> GetAList(string a, int mSA)
{
var list = new List<string>();
var a1 = a.Substring(0, 4);
var a1Pairs = DataConvert.StringToPairs(a1);
a1Pairs.Reverse();
list.AddRange(a1Pairs);
var a2 = Convert.ToInt32(a.Substring(4));
var decA2 = DataConvert.DecToHex(a2);
var a2Pairs = DataConvert.StringToPairs(decA2.PadLeft(4, '0'));
a2Pairs.Reverse();
list.AddRange(a2Pairs);
//TODO:主站地址和组地址标志
var a3Bin = $"{DataConvert.DecToBin(mSA).PadLeft(7, '0')}0";
list.Add(DataConvert.BinToHex(a3Bin).PadLeft(2, '0'));
return list;
}
#region
private List<string> GetLinkUserData(AFN aFN, Seq seq, int pn, int fn, List<string>? dataUnit)
{
var aFNValue = DataConvert.DecToHex((int)aFN).PadLeft(2, '0');
var sEQ = GetSEQ(seq.TpV, seq.FIRFIN, seq.CON, seq.PRSEQ);
var dA = GetDA(pn);
var dT = GetDT(fn);
var list = new List<string>() { aFNValue, sEQ };
list.AddRange(dA);
list.AddRange(dT);
if (dataUnit != null)
{
list.AddRange(dataUnit);
}
//list.AddRange(GetDataUnit(aFN,seq));
if (seq.TpV == TpV.)
list.AddRange(GetTp("00"));
return list;
}
/// <summary>
/// 帧序列域
/// </summary>
/// <param name="tpV"></param>
/// <param name="fIRFIN"></param>
/// <param name="cON"></param>
/// <returns></returns>
private string GetSEQ(TpV tpV, FIRFIN fIRFIN, CON cON, int pRSEQ)
{
var tpVValue = Enum.Format(typeof(TpV),
tpV, "d");
var fIRFINValue = Enum.Format(typeof(FIRFIN),
fIRFIN, "d");
var cONValue = (int)cON;
var sEQBin = $"{tpVValue}{fIRFINValue}{cONValue}{DataConvert.DecToBin(pRSEQ).PadLeft(4, '0')}";
var hexSEQ = DataConvert.BinToHex(sEQBin).PadLeft(2, '0');
return hexSEQ;
}
/// <summary>
/// 信息点标识
/// </summary>
/// <param name="pn">计量点</param>
/// <returns></returns>
private List<string> GetDA(int pn)
{
if (pn == 0)
return new List<string>() { "00", "00" };
var dA2 = (pn - 1) / 8 + 1;//信息点组从1开始 第几组
var dA1 = pn - (dA2 - 1) * 8;//pn % 8
var dA1Hex = DataConvert.BinToHex("1".PadRight(dA1, '0'));//对位信息 第几位 二进制有效位
var dA2Hex = DataConvert.DecToHex(dA2);
return new List<string>() { dA1Hex.PadLeft(2, '0'), dA2Hex.PadLeft(2, '0') };
}
/// <summary>
/// 数据单元标识
/// </summary>
/// <param name="fn"></param>
/// <returns></returns>
private List<string> GetDT(int fn)
{
var dT2 = (fn - 1) / 8;//从零开始 第几组
var dT1 = fn - dT2 * 8;
var dT1Hex = DataConvert.BinToHex("1".PadRight(dT1, '0'));//对位信息 第几位 二进制有效位
var dT2Hex = DataConvert.DecToHex(dT2);
return new List<string>() { dT1Hex.PadLeft(2, '0'), dT2Hex.PadLeft(2, '0') };
}
private List<string> GetDataUnit(AFN aFN, Seq seq)
{
var datas = new List<string>();
switch (aFN)
{
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
case AFN.:
break;
default:
break;
}
if (seq.TpV == TpV.)
datas.AddRange(GetTp("00"));
return datas;
}
private void GetAFN00DataUnit(Seq seq)
{
//EC+Tp
}
/// <summary>
/// 终端电能表配置参数
/// </summary>
/// <param name="meterParameters"></param>
/// <returns></returns>
public List<string> GetAFN04F10DataUnit(List<MeterParameter> meterParameters)
{
var hexDatas = new List<string>();
var countHex = DataConvert.DecToHex(meterParameters.Count()).PadLeft(4, '0');
hexDatas.Add(countHex);
//TODO 优化代码:目标数据入参,返回类型为出参
for (int i = 0; i <= meterParameters.Count - 1; i++)
{
var meter = meterParameters[i];
var indexHex = DataConvert.DecToHex(i + 1).PadLeft(4, '0');
hexDatas.Add(indexHex);
var pnHex = DataConvert.DecToHex(meter.Pn).PadLeft(4, '0');
hexDatas.Add(pnHex);
var baudRateBin = DataConvert.DecToBin(meter.BaudRate).PadLeft(3, '0');
var portBin = DataConvert.DecToBin(meter.Port).PadLeft(5, '0');
var baudRateAndPortHex = DataConvert.BinToHex($"{baudRateBin}{portBin}").PadLeft(2, '0');
hexDatas.Add(baudRateAndPortHex);
var protocolTypeHex = DataConvert.DecToHex((int)meter.ProtocolType).PadLeft(2, '0');
hexDatas.Add(protocolTypeHex);
hexDatas.Add(meter.Address);
hexDatas.Add(meter.Password.PadLeft(12, '0'));
var rateNumberBin = $"0000{DataConvert.DecToBin(meter.RateNumber).PadLeft(4, '0')}";
var rateNumberHex = DataConvert.BinToHex(rateNumberBin).PadLeft(2, '0');
hexDatas.Add(rateNumberHex);
var intBitNumberBin = DataConvert.DecToBin(meter.IntegerBitNumber - 4).PadLeft(2, '0');
var decBitNumberBin = DataConvert.DecToBin(meter.DecimalBitNumber - 1).PadLeft(2, '0');
var intAndDecBitNumberBin = $"0000{intBitNumberBin}{decBitNumberBin}";
var intAndDecBitNumberHex = DataConvert.BinToHex(intAndDecBitNumberBin).PadLeft(2, '0');
hexDatas.Add(intAndDecBitNumberHex);
hexDatas.Add(meter.CollectorAddress.PadLeft(12, '0'));
var userCategoryNumberBin = DataConvert.DecToBin(meter.UserCategoryNumber).PadLeft(4, '0');
var userSubclassNumberBin = DataConvert.DecToBin(meter.UserSubclassNumber).PadLeft(4, '0');
var userNumberHex = DataConvert.BinToHex($"{userCategoryNumberBin}{userSubclassNumberBin}").PadLeft(2, '0');
hexDatas.Add(userNumberHex);
}
//高位在前,低位在后
var datas = new List<string>();
foreach (var hexData in hexDatas)
{
if (hexData.Length == 2)
datas.Add(hexData);
else
{
var lst = DataConvert.StringToPairs(hexData);
lst.Reverse();
datas.AddRange(lst);
}
}
datas.AddRange(GetPW());
return datas;
}
/// <summary>
/// 透明转发
/// </summary>
/// <param name="port">终端通信端口 1~31</param>
/// <param name="baudRate">0~7 对应300,600,1200,2400,4800,7200,9600,19200</param>
/// <param name="stopBit"></param>
/// <param name="parity"></param>
/// <param name="dataBit"></param>
/// <returns></returns>
public List<string> GetAFN1001DataUnit(int port, BaudRate baudRate, StopBit stopBit, Parity parity, DataBit dataBit,
int waitContentTimeout, int waitByteTimeout, List<string> datas)
{
var dataUnit = new List<string>();
var portHex = DataConvert.DecToHex(port).PadLeft(2, '0');
dataUnit.Add(portHex);
var baudRateBin = DataConvert.DecToBin((int)baudRate).PadLeft(3, '0');
var stopBitBin = DataConvert.DecToBin((int)stopBit);
var parityBin = parity != Parity.None ? $"1{DataConvert.DecToBin((int)parity)}" : $"0{DataConvert.DecToBin((int)parity)}";
var dataBitBin = DataConvert.DecToBin((int)dataBit).PadLeft(2, '0');
var controlHex = DataConvert.BinToHex($"{baudRateBin}{stopBitBin}{parityBin}{dataBitBin}").PadLeft(2, '0'); ;
dataUnit.Add(controlHex);
var waitContentTimeoutBin = $"1{DataConvert.DecToBin(waitContentTimeout).PadLeft(7, '0')}";
var waitContentTimeoutHex = DataConvert.BinToHex(waitContentTimeoutBin).PadLeft(2, '0');
var waitByteTimeoutHex = DataConvert.DecToHex(waitByteTimeout).PadLeft(2, '0');
dataUnit.Add(waitContentTimeoutHex);
dataUnit.Add(waitByteTimeoutHex);
var countHex = DataConvert.DecToHex(datas.Count).PadLeft(4, '0');
var countHexPairs = DataConvert.StringToPairs(countHex);
countHexPairs.Reverse();
dataUnit.AddRange(countHexPairs);
dataUnit.AddRange(datas);
return dataUnit;
}
//TODO AUX=消息认证码字段PW,16个字节+时间标签
private List<string> GetPW()
{
var str = "00";
var pWList = Enumerable.Repeat(str, pWLen).ToList();
return pWList;
}
/// <summary>
/// 时间标签
/// </summary>
/// <param name="pFC">启动帧帧序号计数器PFC 1字节</param>
/// <param name="delayTime">允许发送传输延时时间 min 1字节</param>
/// <returns></returns>
private List<string> GetTp(string pFC = "00", int delayTime = 0)
{
var now = DateTime.Now; // 获取当前时间
var seconds = now.Second.ToString().PadLeft(2, '0'); // 获取当前秒数
var minutes = now.Minute.ToString().PadLeft(2, '0'); // 获取当前分钟数
var hours = now.Hour.ToString().PadLeft(2, '0'); // 获取当前小时数
var day = now.Day.ToString().PadLeft(2, '0'); // 获取当前日期的日数
return new List<string>() { pFC, seconds, minutes, hours, day, delayTime.ToString().PadLeft(2, '0') };
}
#endregion
/// <summary>
/// 帧校验和
/// </summary>
/// <param name="userData">用户数据区</param>
/// <returns></returns>
private string GetCS(List<string> userData)
{
byte sum = 0;
foreach (var d in userData)
{
var b = Convert.ToByte(d, 16);
sum += b;
}
return sum.ToString("X2");
}
#endregion
}
}

View File

@ -1,7 +1,5 @@
using JiShe.CollectBus.Protocol.Contracts.Interfaces;
using JiShe.CollectBus.Protocol.Contracts.Models;
using JiShe.CollectBus.Protocol.Contracts.Models;
using MassTransit;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
using TouchSocket.Sockets;

View File

@ -0,0 +1,35 @@
using JiShe.CollectBus.Protocol.Contracts.Interfaces;
using JiShe.CollectBus.Protocol.Contracts.Models;
using MassTransit;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
namespace JiShe.CollectBus.RabbitMQ.Consumers
{
public class MessageReceivedHeartbeatConsumer : IConsumer<MessageReceivedHeartbeatEvent>
{
private readonly ILogger<MessageReceivedHeartbeatConsumer> _logger;
private readonly IServiceProvider _serviceProvider;
public MessageReceivedHeartbeatConsumer(ILogger<MessageReceivedHeartbeatConsumer> logger, IServiceProvider serviceProvider)
{
_logger = logger;
_serviceProvider = serviceProvider;
}
public async Task Consume(ConsumeContext<MessageReceivedHeartbeatEvent> context)
{
_logger.LogInformation("心跳消费队列开始处理");
var protocolPlugin = _serviceProvider.GetKeyedService<IProtocolPlugin>("StandardProtocol");
if (protocolPlugin == null)
{
_logger.LogError("【心跳消费队列开始处理】协议不存在!");
}
else
{
await protocolPlugin.HeartbeatAsync(context.Message);
_logger.LogInformation("心跳消费队列完成处理");
}
}
}
}

View File

@ -0,0 +1,35 @@
using JiShe.CollectBus.Protocol.Contracts.Interfaces;
using JiShe.CollectBus.Protocol.Contracts.Models;
using MassTransit;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Logging;
namespace JiShe.CollectBus.RabbitMQ.Consumers
{
public class MessageReceivedLoginConsumer : IConsumer<MessageReceivedLoginEvent>
{
private readonly ILogger<MessageReceivedLoginConsumer> _logger;
private readonly IServiceProvider _serviceProvider;
public MessageReceivedLoginConsumer(ILogger<MessageReceivedLoginConsumer> logger, IServiceProvider serviceProvider)
{
_logger = logger;
_serviceProvider = serviceProvider;
}
public async Task Consume(ConsumeContext<MessageReceivedLoginEvent> context)
{
_logger.LogInformation("登录消费队列开始处理");
var protocolPlugin = _serviceProvider.GetKeyedService<IProtocolPlugin>("StandardProtocol");
if (protocolPlugin == null)
{
_logger.LogError("【登录消费队列开始处理】协议不存在!");
}
else
{
await protocolPlugin.LoginAsync(context.Message);
_logger.LogInformation("登录消费队列完成处理");
}
}
}
}

View File

@ -1,10 +1,8 @@
using System;
using JiShe.CollectBus.Common.Interfaces;
using JiShe.CollectBus.Common.Interfaces;
using JiShe.CollectBus.RabbitMQ.Consumers;
using MassTransit;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using RabbitMQ.Client;
namespace JiShe.CollectBus.RabbitMQ
{
@ -17,6 +15,8 @@ namespace JiShe.CollectBus.RabbitMQ
{
x.AddConsumer<MessageReceivedConsumer>();
x.AddConsumer<MessageIssuedConsumer>();
x.AddConsumer<MessageReceivedLoginConsumer>();
x.AddConsumer<MessageReceivedHeartbeatConsumer>();
x.UsingRabbitMq((context, cfg) =>
{
cfg.Host(configuration["MQ:Host"], ushort.Parse(configuration["MQ:Port"] ?? string.Empty), configuration["MQ:VirtualHost"], h =>
@ -31,7 +31,18 @@ namespace JiShe.CollectBus.RabbitMQ
configurator.ConfigureConsumeTopology = false;
configurator.Consumer<MessageReceivedConsumer>(context);
});
// 登录
cfg.ReceiveEndpoint($"{configuration["MQ:Queue:Received"]}_Login" ?? string.Empty, configurator =>
{
configurator.ConfigureConsumeTopology = false;
configurator.Consumer<MessageReceivedLoginConsumer>(context);
});
// 心跳
cfg.ReceiveEndpoint($"{configuration["MQ:Queue:Received"]}_Heartbeat" ?? string.Empty, configurator =>
{
configurator.ConfigureConsumeTopology = false;
configurator.Consumer<MessageReceivedHeartbeatConsumer>(context);
});
// 消息下发队列
cfg.ReceiveEndpoint(configuration["MQ:Queue:Issued"] ?? string.Empty, configurator =>
{

View File

@ -12,6 +12,10 @@ namespace JiShe.CollectBus.RabbitMQ.Senders
Task SendToIssuedAsync(object message, CancellationToken cancellationToken = default);
Task SendToReceivedAsync<T>(T message, CancellationToken cancellationToken = default) where T : class;
Task SendToReceivedAsync(object message, CancellationToken cancellationToken = default);
Task SendToReceivedLoginAsync<T>(T message, CancellationToken cancellationToken = default) where T : class;
Task SendToReceivedLoginAsync(object message, CancellationToken cancellationToken = default);
Task SendToReceivedHeartbeatAsync<T>(T message, CancellationToken cancellationToken = default) where T : class;
Task SendToReceivedHeartbeatAsync(object message, CancellationToken cancellationToken = default);
Task SendAsync(string queueKey, object message, CancellationToken cancellationToken = default);
}

View File

@ -43,6 +43,26 @@ namespace JiShe.CollectBus.RabbitMQ.Senders
await SendAsync(_receivedKey, message, cancellationToken);
}
public async Task SendToReceivedLoginAsync<T>(T message, CancellationToken cancellationToken = default) where T : class
{
await SendAsync($"{_receivedKey}_Login", message, cancellationToken);
}
public async Task SendToReceivedLoginAsync(object message, CancellationToken cancellationToken = default)
{
await SendAsync($"{_receivedKey}_Login", message, cancellationToken);
}
public async Task SendToReceivedHeartbeatAsync<T>(T message, CancellationToken cancellationToken = default) where T : class
{
await SendAsync($"{_receivedKey}_Heartbeat", message, cancellationToken);
}
public async Task SendToReceivedHeartbeatAsync(object message, CancellationToken cancellationToken = default)
{
await SendAsync($"{_receivedKey}_Heartbeat", message, cancellationToken);
}
public async Task SendAsync(string queueKey, object message, CancellationToken cancellationToken = default)
{