Compare commits

...

6 Commits

Author SHA1 Message Date
f6fd688f9f Merge branch 'main' of http://git.crash.work/cneicy/Shrink-IDC
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:06:34 +08:00
3a41f098fe 实现API功能并编写日志过滤方法
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:05:49 +08:00
2537d11f53 新增Config供过滤输出以及将来的其他功能使用
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:03:35 +08:00
7bacff5e24 使用责任链编写Logger
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:02:31 +08:00
f05dce81a0 更新proto
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:01:13 +08:00
b6539a3298 将Json操作工具化(但仍存在一些小bug导致不可用)
Signed-off-by: Eicy <im@crash.work>
2024-12-21 01:00:41 +08:00
19 changed files with 443 additions and 131 deletions

View File

@@ -1,19 +0,0 @@
using Debugger;
using Grpc.Core;
namespace Shrink.API;
public class BotServiceImpl : APIService.APIServiceBase
{
public override Task<DataResponse> SendData(DataRequest request, ServerCallContext context)
{
// 模拟向 QQ 机器人处理数据的逻辑
// 返回响应
return Task.FromResult(new DataResponse
{
Message = "114514",
Success = true
});
}
}

View File

@@ -1,31 +0,0 @@
syntax = "proto3";
option csharp_namespace = "Debugger";
package Debugger;
service APIService {
rpc SendData (DataRequest) returns (DataResponse);
}
message DataRequest {
enum msgType{
public = 0;
private = 1;
}
int32 uin = 1;
enum logLevel{
fatal = 0;
error = 1;
warn = 2;
info = 3;
verbose= 4;
debug = 5;
}
string text = 2;
}
message DataResponse {
string message = 1;
bool success = 2;
}

44
Config/Configuration.cs Normal file
View File

@@ -0,0 +1,44 @@
using Debugger;
using Newtonsoft.Json;
namespace Shrink.Config;
public struct Config
{
public LogLevel LogLevel;
public Config(LogLevel logLevel)
{
LogLevel = logLevel;
}
}
public class Configuration
{
private static readonly Lazy<Configuration> _instance = new(() => new Configuration());
public static Configuration Instance => _instance.Value;
private const string ConfigFilePath = "Config.json";
public Config Config;
private Configuration()
{
}
public async Task Init()
{
if (!File.Exists(ConfigFilePath))
{
await Save();
}
else
{
Config = JsonConvert.DeserializeObject<Config>(await File.ReadAllTextAsync(ConfigFilePath));
}
}
public async Task Save()
{
await File.WriteAllTextAsync(ConfigFilePath, JsonConvert.SerializeObject(Config));
}
}

View File

@@ -1,16 +0,0 @@
using Lagrange.Core.Event;
using Shrink.Login;
namespace Shrink.Event;
public class BotEventHandler
{
public BotEventHandler()
{
Console.WriteLine("Bot EventHandler");
}
private static readonly Lazy<BotEventHandler> _instance = new(() => new BotEventHandler());
public static BotEventHandler Instance => _instance.Value;
private EventInvoker _invoker = BotService.Instance.Client!.Invoker;
}

33
Logger/AbstractLogger.cs Normal file
View File

@@ -0,0 +1,33 @@
using Debugger;
using Shrink.Config;
namespace Shrink.Logger;
public abstract class AbstractLogger
{
private AbstractLogger? _nextLogger;
protected LogLevel LogLevel;
public void SetNextLogger(AbstractLogger? nextLogger)
{
_nextLogger = nextLogger;
}
public void LogMessage(string data, string eventName, LogLevel level, MsgType msgType, uint uin, string message)
{
//LogLevel = Info
//Config设置为Only Fatal
if (LogLevel == level)
{
//Info 3
//Fatal 0
if (LogLevel > Configuration.Instance.Config.LogLevel) return;
Log(data, eventName, msgType, uin, message);
LogManager.Instance.IsSent = true;
}
else
_nextLogger?.LogMessage(data, eventName, level, msgType, uin, message);
}
protected abstract void Log(string data, string eventName, MsgType msgType, uint uin, string message);
}

28
Logger/DebugLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class DebugLogger : AbstractLogger
{
public DebugLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Debug] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Debug] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

28
Logger/ErrorLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class ErrorLogger : AbstractLogger
{
public ErrorLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Error] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Error] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

28
Logger/FatalLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class FatalLogger : AbstractLogger
{
public FatalLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Fatal] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Fatal] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

28
Logger/InfoLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class InfoLogger : AbstractLogger
{
public InfoLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Info] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Info] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

28
Logger/LogManager.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
namespace Shrink.Logger;
public class LogManager
{
private static readonly Lazy<LogManager> _instance = new(() => new LogManager());
public static LogManager Instance => _instance.Value;
public AbstractLogger LogChain { get; }
public bool IsSent;
public LogManager()
{
AbstractLogger fatalLogger = new FatalLogger(LogLevel.Fatal);
AbstractLogger errorLogger = new ErrorLogger(LogLevel.Error);
AbstractLogger warnLogger = new WarnLogger(LogLevel.Warn);
AbstractLogger infoLogger = new InfoLogger(LogLevel.Info);
AbstractLogger verboseLogger = new VerboseLogger(LogLevel.Verbose);
AbstractLogger debugLogger = new DebugLogger(LogLevel.Debug);
fatalLogger.SetNextLogger(errorLogger);
errorLogger.SetNextLogger(warnLogger);
warnLogger.SetNextLogger(infoLogger);
infoLogger.SetNextLogger(verboseLogger);
verboseLogger.SetNextLogger(debugLogger);
LogChain = fatalLogger;
}
}

28
Logger/VerboseLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class VerboseLogger : AbstractLogger
{
public VerboseLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Verbose] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Verbose] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

28
Logger/WarnLogger.cs Normal file
View File

@@ -0,0 +1,28 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Message;
using Shrink.Service;
namespace Shrink.Logger;
public class WarnLogger : AbstractLogger
{
public WarnLogger(LogLevel level)
{
LogLevel = level;
}
protected override void Log(string data, string eventName, MsgType msgType, uint uin, string message)
{
if (msgType == MsgType.Public)
{
var chain = MessageBuilder.Group(uin).Text($"[{data}] [{eventName}] [Warn] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
else
{
var chain = MessageBuilder.Friend(uin).Text($"[{data}] [{eventName}] [Warn] {message}");
BotService.Instance.Client!.SendMessage(chain.Build());
}
}
}

View File

@@ -1,4 +1,5 @@
using BotService = Shrink.Login.BotService;
using Shrink.Config;
using Shrink.Service;
namespace Shrink;
@@ -6,6 +7,8 @@ public static class Program
{
public static async Task Main()
{
await Configuration.Instance.Init();
await BotService.Instance.Login();
await BotPassiveMsgHandler.Instance.Init();
}
}

32
Service/ApiService.cs Normal file
View File

@@ -0,0 +1,32 @@
using Debugger;
using Grpc.Core;
using Shrink.Logger;
namespace Shrink.Service;
public class ApiService : APIService.APIServiceBase
{
private static readonly Lazy<ApiService> _instance = new(() => new ApiService());
public static ApiService Instance => _instance.Value;
public override Task<DataResponse> SendData(DataRequest request, ServerCallContext context)
{
LogManager.Instance.LogChain.LogMessage(request.Data, request.Event, request.LogLevel, request.MsgType,
request.Uin, request.Text);
var response = new DataResponse();
if (LogManager.Instance.IsSent)
{
response.Message = $"已发送至: {request.Uin}。";
response.Success = true;
}
else
{
response.Message = "未发送,可能的原因是信息被设置为忽略或出现其他问题。";
response.Success = false;
}
LogManager.Instance.IsSent = false;
return Task.FromResult(response);
}
}

View File

@@ -0,0 +1,38 @@
using Debugger;
using Lagrange.Core.Common.Interface.Api;
using Lagrange.Core.Event;
using Lagrange.Core.Message;
using Shrink.Config;
namespace Shrink.Service;
public class BotPassiveMsgHandler
{
private static readonly Lazy<BotPassiveMsgHandler> _instance = new(() => new BotPassiveMsgHandler());
public static BotPassiveMsgHandler Instance => _instance.Value;
public EventInvoker Invoker { get; } = BotService.Instance.Client!.Invoker;
public Task Init()
{
Invoker.OnGroupMessageReceived += async (context, @event) =>
{
if (!@event.Chain.ToPreviewText().StartsWith("/switchlevel")) return;
var temp = @event.Chain.ToPreviewText()[("/switchlevel".Length + 1)..];
Configuration.Instance.Config = temp switch
{
"debug" => new Config.Config(LogLevel.Debug),
"info" => new Config.Config(LogLevel.Info),
"verbose" => new Config.Config(LogLevel.Verbose),
"warn" => new Config.Config(LogLevel.Warn),
"error" => new Config.Config(LogLevel.Error),
"fatal" => new Config.Config(LogLevel.Fatal),
_ => Configuration.Instance.Config
};
await Configuration.Instance.Save();
var chain = MessageBuilder.Group((uint)@event.Chain.GroupUin!)
.Text($"已将日志输出级别调整为{Configuration.Instance.Config.LogLevel}");
await context.SendMessage(chain.Build());
};
return Task.CompletedTask;
}
}

View File

@@ -1,16 +1,15 @@
using System.Text.Json;
using System.Text.Json.Serialization;
using Debugger;
using Grpc.Core;
using Lagrange.Core;
using Lagrange.Core.Common;
using Lagrange.Core.Common.Interface;
using Lagrange.Core.Common.Interface.Api;
using Shrink.API;
using Shrink.Logger;
using Shrink.Utility;
using Console = System.Console;
namespace Shrink.Login;
namespace Shrink.Service;
public class BotService
{
@@ -24,24 +23,24 @@ public class BotService
private static BotDeviceInfo GetDeviceInfo() =>
ReadOrCreateJsonFile(DeviceInfoFilePath, BotDeviceInfo.GenerateInfo);
JsonUtility.ReadOrCreateJsonFile(DeviceInfoFilePath, BotDeviceInfo.GenerateInfo);
// 登录方法
public async Task Login()
{
const int port = 50051;
var server = new Server
{
Services = { APIService.BindService(new BotServiceImpl()) },
Services = { APIService.BindService(new ApiService()) },
Ports = { new ServerPort("localhost", port, ServerCredentials.Insecure) }
};
var deviceInfo = File.Exists(DeviceInfoFilePath)
? ReadJsonFromFile<BotDeviceInfo>(DeviceInfoFilePath)
? JsonUtility.ReadJsonFromFile<BotDeviceInfo>(DeviceInfoFilePath)
: GetDeviceInfo();
var keyStore = File.Exists(KeystoreFilePath)
? ReadJsonFromFile<BotKeystore>(KeystoreFilePath)
? JsonUtility.ReadJsonFromFile<BotKeystore>(KeystoreFilePath)
: new BotKeystore();
Client = BotFactory.Create(new BotConfig
@@ -65,7 +64,9 @@ public class BotService
Console.WriteLine(@event.ToString());
_isOnline = true;
server.Start();
Console.WriteLine($"gRPC server listening on port {port}");
Console.WriteLine($"[{DateTime.Now}] [gRPC Server] [Info]: Listening on port: {port}");
/*LogManager.Instance.LogChain.LogMessage($"{DateTime.Now}", "LoginEvent", LogLevel.Info, MsgType.Public,
620902312, "Login");*/
};
Client.Invoker.OnBotOfflineEvent += async (_, @event) =>
{
@@ -92,45 +93,4 @@ public class BotService
await File.WriteAllTextAsync(KeystoreFilePath, JsonSerializer.Serialize(Client.UpdateKeystore()));
await File.WriteAllTextAsync(DeviceInfoFilePath, JsonSerializer.Serialize(Client.UpdateDeviceInfo()));
}
private static T ReadJsonFromFile<T>(string filePath)
{
try
{
var json = File.ReadAllText(filePath);
return JsonSerializer.Deserialize<T>(json,
new JsonSerializerOptions { ReferenceHandler = ReferenceHandler.Preserve })!;
}
catch (Exception ex)
{
Console.WriteLine($"读取文件出错: {filePath}: {ex.Message}");
throw;
}
}
private static T ReadOrCreateJsonFile<T>(string filePath, Func<T> createFunc)
{
if (File.Exists(filePath))
{
return ReadJsonFromFile<T>(filePath);
}
var newData = createFunc();
WriteJsonToFile(filePath, newData);
return newData;
}
private static void WriteJsonToFile<T>(string filePath, T data)
{
try
{
var json = JsonSerializer.Serialize(data);
File.WriteAllText(filePath, json);
}
catch (Exception ex)
{
Console.WriteLine($"写入文件出错: {filePath}: {ex.Message}");
throw;
}
}
}

View File

@@ -17,23 +17,12 @@
</PackageReference>
<PackageReference Include="Lagrange.Core" Version="0.3.1" />
<PackageReference Include="Mono.Cecil" Version="0.11.6" />
<PackageReference Include="Newtonsoft.Json" Version="13.0.3" />
<PackageReference Include="SSH.NET" Version="2024.2.0" />
</ItemGroup>
<ItemGroup>
<Compile Remove="Service\**" />
</ItemGroup>
<ItemGroup>
<Protobuf Include="API/service.proto" GrpcServices="Server" />
</ItemGroup>
<ItemGroup>
<EmbeddedResource Remove="Service\**" />
</ItemGroup>
<ItemGroup>
<None Remove="Service\**" />
<Protobuf Include="proto\apiService.proto" GrpcServices="Server" />
</ItemGroup>
</Project>

48
Utility/JsonUtility.cs Normal file
View File

@@ -0,0 +1,48 @@
using System.Text.Json;
using System.Text.Json.Serialization;
namespace Shrink.Utility;
public static class JsonUtility
{
public static void WriteJsonToFile<T>(string filePath, T data)
{
try
{
var json = JsonSerializer.Serialize(data);
File.WriteAllText(filePath, json);
}
catch (Exception ex)
{
System.Console.WriteLine($"写入文件出错: {filePath}: {ex.Message}");
throw;
}
}
public static T ReadJsonFromFile<T>(string filePath)
{
try
{
var json = File.ReadAllText(filePath);
return JsonSerializer.Deserialize<T>(json,
new JsonSerializerOptions { ReferenceHandler = ReferenceHandler.Preserve })!;
}
catch (Exception ex)
{
System.Console.WriteLine($"读取文件出错: {filePath}: {ex.Message}");
throw;
}
}
public static T ReadOrCreateJsonFile<T>(string filePath, Func<T> createFunc)
{
if (File.Exists(filePath))
{
return ReadJsonFromFile<T>(filePath);
}
var newData = createFunc();
WriteJsonToFile(filePath, newData);
return newData;
}
}

35
proto/apiService.proto Normal file
View File

@@ -0,0 +1,35 @@
syntax = "proto3";
option csharp_namespace = "Debugger";
package Debugger;
service APIService {
rpc SendData (DataRequest) returns (DataResponse);
}
enum MsgType{
public = 0;
private = 1;
}
enum LogLevel{
fatal = 0;
error = 1;
warn = 2;
info = 3;
verbose = 4;
debug = 5;
}
message DataRequest {
string data = 1;
string event = 2;
LogLevel logLevel = 3;
MsgType msgType = 4;
uint32 uin = 5;
string text = 6;
}
message DataResponse {
string message = 1;
bool success = 2;
}