2025-07-27 23:34:01 +08:00
|
|
|
|
using Serein.Library.Api;
|
|
|
|
|
|
using Serein.Library.Utils;
|
2025-08-02 10:48:31 +08:00
|
|
|
|
using Serein.Proto.WebSocket.Handle;
|
2024-10-28 00:31:41 +08:00
|
|
|
|
using System;
|
2025-08-25 10:53:22 +08:00
|
|
|
|
using System.Diagnostics.CodeAnalysis;
|
2024-10-28 00:31:41 +08:00
|
|
|
|
using System.Threading.Tasks;
|
2025-08-02 12:09:36 +08:00
|
|
|
|
using static System.Runtime.InteropServices.JavaScript.JSType;
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
2025-08-02 10:48:31 +08:00
|
|
|
|
namespace Serein.Proto.WebSocket
|
2024-10-28 00:31:41 +08:00
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 消息处理上下文
|
|
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public class WebSocketHandleContext
|
2024-10-28 00:31:41 +08:00
|
|
|
|
{
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 构造函数,传入发送消息的异步方法
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="sendAsync"></param>
|
|
|
|
|
|
public WebSocketHandleContext(Func<string, Task> sendAsync)
|
2024-10-28 00:31:41 +08:00
|
|
|
|
{
|
2025-07-27 23:34:01 +08:00
|
|
|
|
_sendAsync = sendAsync;
|
2024-10-28 00:31:41 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// <summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// 重置上下文
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public void Reset()
|
2024-10-28 10:25:57 +08:00
|
|
|
|
{
|
2025-07-27 23:34:01 +08:00
|
|
|
|
MsgRequest = null;
|
2024-10-28 10:25:57 +08:00
|
|
|
|
MsgData = null;
|
2025-08-25 10:53:22 +08:00
|
|
|
|
ErrorMessage = null;
|
|
|
|
|
|
Model = null;
|
|
|
|
|
|
IsError = false;
|
2024-10-28 10:25:57 +08:00
|
|
|
|
}
|
2025-07-27 23:34:01 +08:00
|
|
|
|
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 标记是否已经处理,如果是,则提前退出
|
|
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public bool Handle { get; set; }
|
2025-08-02 10:48:31 +08:00
|
|
|
|
|
2025-08-02 12:09:36 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// WebSocket 模块配置
|
|
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public ModuleConfig Model { get; set; }
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-07-27 23:34:01 +08:00
|
|
|
|
/// 消息本体(IJsonToken)
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// </summary>
|
2025-07-30 21:15:07 +08:00
|
|
|
|
public IJsonToken? MsgRequest { get; set; }
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-08-02 12:09:36 +08:00
|
|
|
|
/// 此次消息的数据
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// </summary>
|
2025-08-02 12:09:36 +08:00
|
|
|
|
public IJsonToken? MsgData { get; set; }
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
2025-08-25 10:53:22 +08:00
|
|
|
|
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// <summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// 是否发生错误
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public bool IsError { get; set; }
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// 错误消息
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
public string ErrorMessage { get; set; }
|
|
|
|
|
|
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// <summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// 用于在同一个 Web Socket 中调用上下文中, 共享某个对象
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
private object? _wsTag;
|
|
|
|
|
|
private object _wsTagLockObj = new object();
|
2025-08-02 12:09:36 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-08-25 10:57:32 +08:00
|
|
|
|
/// 设置共享对象,不建议设置非托管对象
|
2025-08-02 12:09:36 +08:00
|
|
|
|
/// </summary>
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// <typeparam name="T"></typeparam>
|
|
|
|
|
|
/// <param name="tag"></param>
|
|
|
|
|
|
private void SetTag<T>(T tag)
|
|
|
|
|
|
{
|
|
|
|
|
|
lock (_wsTagLockObj)
|
|
|
|
|
|
{
|
|
|
|
|
|
_wsTag = tag;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-08-25 10:57:32 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取共享对象(将在同一个 Web Socket 调起的上下文中保持一致)
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="T"></typeparam>
|
|
|
|
|
|
/// <param name="tag"></param>
|
|
|
|
|
|
private object? GetTag()
|
|
|
|
|
|
{
|
|
|
|
|
|
TryGetTag(out object? tag);
|
|
|
|
|
|
return tag;
|
|
|
|
|
|
}
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取共享对象(将在同一个 Web Socket 调起的上下文中保持一致)
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="T"></typeparam>
|
|
|
|
|
|
/// <param name="tag"></param>
|
|
|
|
|
|
private T? GetTag<T>()
|
|
|
|
|
|
{
|
|
|
|
|
|
TryGetTag(out T? tag);
|
|
|
|
|
|
return tag;
|
|
|
|
|
|
}
|
2025-08-25 10:53:22 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取共享对象(将在同一个 Web Socket 调起的上下文中保持一致)
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <typeparam name="T"></typeparam>
|
|
|
|
|
|
/// <param name="tag"></param>
|
|
|
|
|
|
private bool TryGetTag<T>([NotNullWhen(true)] out T? tag)
|
|
|
|
|
|
{
|
|
|
|
|
|
lock (_wsTagLockObj)
|
|
|
|
|
|
{
|
|
|
|
|
|
if (_wsTag is T t)
|
|
|
|
|
|
{
|
|
|
|
|
|
tag = t;
|
|
|
|
|
|
return true;
|
|
|
|
|
|
}
|
|
|
|
|
|
tag = default;
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 异常外部感知使能
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public Action<Exception>? OnExceptionTracking { get; set; }
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 处理回复消息的函数
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public Func<WebSocketHandleContext, object, object> OnReplyMakeData { get; set; }
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取回复内容的回调
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public Action<string>? OnReply { get; set; }
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 发送消息
|
|
|
|
|
|
/// </summary>
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
2025-08-02 10:48:31 +08:00
|
|
|
|
private Func<string, Task> _sendAsync;
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
2025-08-25 10:53:22 +08:00
|
|
|
|
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 发送消息
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="msg"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public async Task SendAsync(string msg)
|
|
|
|
|
|
{
|
|
|
|
|
|
await _sendAsync.Invoke(msg);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// 触发异常追踪
|
2024-10-28 00:31:41 +08:00
|
|
|
|
/// </summary>
|
2025-08-02 10:48:31 +08:00
|
|
|
|
public void TriggerExceptionTracking(string exMessage)
|
2024-10-28 00:31:41 +08:00
|
|
|
|
{
|
2025-08-02 12:09:36 +08:00
|
|
|
|
var ex = new SereinWebSocketHandleException(exMessage);
|
|
|
|
|
|
TriggerExceptionTracking(ex);
|
2025-08-02 10:48:31 +08:00
|
|
|
|
}
|
2025-08-02 12:09:36 +08:00
|
|
|
|
|
2025-08-02 10:48:31 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 触发异常追踪
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public void TriggerExceptionTracking(Exception ex)
|
|
|
|
|
|
{
|
2025-08-02 12:09:36 +08:00
|
|
|
|
IsError = true;
|
|
|
|
|
|
var msgId = Model.MsgId;
|
|
|
|
|
|
var theme = Model.Theme;
|
|
|
|
|
|
ErrorMessage = $"请求[{msgId}]主题[{theme}]异常 :{ex.Message}";
|
|
|
|
|
|
OnExceptionTracking?.Invoke(ex);
|
|
|
|
|
|
var error = OnReplyMakeData?.Invoke(this, ex.Message); // 触发回复消息
|
|
|
|
|
|
_ = SendAsync(JsonHelper.Serialize(error)).ContinueWith((t) =>
|
2025-08-02 10:48:31 +08:00
|
|
|
|
{
|
2025-08-02 12:09:36 +08:00
|
|
|
|
if (t.IsFaulted)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine($"发送错误消息失败: {t.Exception?.Message}");
|
|
|
|
|
|
}
|
|
|
|
|
|
}); // 发送错误消息
|
2025-08-02 10:48:31 +08:00
|
|
|
|
}
|
2025-07-27 23:34:01 +08:00
|
|
|
|
|
|
|
|
|
|
|
2024-10-28 00:31:41 +08:00
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
}
|