T type logging
This commit is contained in:
@@ -0,0 +1,174 @@
|
||||
using System.Net.WebSockets;
|
||||
using System.Text;
|
||||
using Core.Annotations;
|
||||
using Core.Helpers;
|
||||
using Core.Models.Eft.Ws;
|
||||
using Core.Models.Spt.Config;
|
||||
using Core.Models.Utils;
|
||||
using Core.Servers.Ws.Message;
|
||||
using Core.Services;
|
||||
using Core.Utils;
|
||||
|
||||
namespace Core.Servers.Ws;
|
||||
|
||||
[Injectable(InjectionType.Singleton)]
|
||||
public class SptWebSocketConnectionHandler : IWebSocketConnectionHandler
|
||||
{
|
||||
protected Dictionary<string, WebSocket> _sockets = new();
|
||||
protected Dictionary<string, Timer> _socketAliveTimers = new();
|
||||
protected Dictionary<string, CancellationTokenSource> _receiveTasks = new();
|
||||
protected ISptLogger<SptWebSocketConnectionHandler> _logger;
|
||||
protected LocalisationService _localisationService;
|
||||
protected JsonUtil _jsonUtil;
|
||||
protected ProfileHelper _profileHelper;
|
||||
protected HttpConfig _httpConfig;
|
||||
protected IEnumerable<ISptWebSocketMessageHandler> _messageHandlers;
|
||||
|
||||
protected WsPing _defaultNotification = new();
|
||||
|
||||
public SptWebSocketConnectionHandler(
|
||||
ISptLogger<SptWebSocketConnectionHandler> logger,
|
||||
LocalisationService localisationService,
|
||||
JsonUtil jsonUtil,
|
||||
ProfileHelper profileHelper,
|
||||
ConfigServer configServer,
|
||||
IEnumerable<ISptWebSocketMessageHandler> messageHandlers
|
||||
)
|
||||
{
|
||||
_logger = logger;
|
||||
_localisationService = localisationService;
|
||||
_jsonUtil = jsonUtil;
|
||||
_profileHelper = profileHelper;
|
||||
_messageHandlers = messageHandlers;
|
||||
_httpConfig = configServer.GetConfig<HttpConfig>();
|
||||
}
|
||||
|
||||
public string GetHookUrl() => "/notifierServer/getwebsocket/";
|
||||
public string GetSocketId() => "SPT WebSocket Handler";
|
||||
|
||||
public Task OnConnection(WebSocket ws, HttpContext context)
|
||||
{
|
||||
var splitUrl = context.Request.Path.Value.Substring(0, context.Request.Path.Value.IndexOf("?")).Split("/");
|
||||
var sessionID = splitUrl.First();
|
||||
var playerProfile = _profileHelper.GetFullProfile(sessionID);
|
||||
var playerInfoText = $"{playerProfile.ProfileInfo.Username} ({sessionID})";
|
||||
|
||||
_logger.Info(_localisationService.GetText("websocket-player_connected", playerInfoText));
|
||||
|
||||
_sockets.Add(sessionID, ws);
|
||||
|
||||
_socketAliveTimers.Add(
|
||||
sessionID,
|
||||
new Timer(
|
||||
async (_) =>
|
||||
{
|
||||
_logger.Debug(_localisationService.GetText("websocket-pinging_player", sessionID));
|
||||
|
||||
if (ws.State == WebSocketState.Open)
|
||||
{
|
||||
await ws.SendAsync(
|
||||
Encoding.UTF8.GetBytes(_jsonUtil.Serialize(_defaultNotification)),
|
||||
WebSocketMessageType.Text,
|
||||
true,
|
||||
CancellationToken.None
|
||||
);
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.Debug(_localisationService.GetText("websocket-socket_lost_deleting_handle"));
|
||||
var timer = _socketAliveTimers[sessionID];
|
||||
timer.Change(Timeout.Infinite, Timeout.Infinite);
|
||||
_socketAliveTimers.Remove(sessionID);
|
||||
_sockets.Remove(sessionID);
|
||||
var receiveTask = _receiveTasks[sessionID];
|
||||
await receiveTask.CancelAsync();
|
||||
}
|
||||
},
|
||||
null,
|
||||
TimeSpan.Zero,
|
||||
TimeSpan.FromMilliseconds(_httpConfig.WebSocketPingDelayMs)
|
||||
)
|
||||
);
|
||||
|
||||
_receiveTasks.Add(sessionID, new());
|
||||
var cancelToken = _receiveTasks[sessionID].Token;
|
||||
Task.Factory.StartNew((_) => ReceiveTask(sessionID, ws, cancelToken), null, cancelToken);
|
||||
|
||||
return Task.CompletedTask;
|
||||
}
|
||||
|
||||
public async Task SendMessageAsync(string sessionID, WsNotificationEvent output)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (IsWebSocketConnected(sessionID))
|
||||
{
|
||||
var ws = GetSessionWebSocket(sessionID);
|
||||
|
||||
await ws.SendAsync(
|
||||
Encoding.UTF8.GetBytes(_jsonUtil.Serialize(output)),
|
||||
WebSocketMessageType.Text,
|
||||
true,
|
||||
CancellationToken.None
|
||||
);
|
||||
_logger.Debug(_localisationService.GetText("websocket-message_sent"));
|
||||
}
|
||||
else
|
||||
{
|
||||
_logger.Debug(_localisationService.GetText("websocket-not_ready_message_not_sent", sessionID));
|
||||
}
|
||||
}
|
||||
catch (Exception err)
|
||||
{
|
||||
_logger.Error(_localisationService.GetText("websocket-message_send_failed_with_error", err));
|
||||
}
|
||||
}
|
||||
|
||||
private void ReceiveTask(string sessionID, WebSocket ws, CancellationToken cancelToken)
|
||||
{
|
||||
List<byte> readBytes = new();
|
||||
while (ws.State == WebSocketState.Open)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (cancelToken.IsCancellationRequested)
|
||||
break;
|
||||
var isEndOfMessage = false;
|
||||
while (!isEndOfMessage)
|
||||
{
|
||||
var buffer = new ArraySegment<byte>(new byte[1024 * 4]);
|
||||
var readTask = ws.ReceiveAsync(buffer, cancelToken);
|
||||
readTask.Wait(cancelToken);
|
||||
readBytes.AddRange(buffer);
|
||||
isEndOfMessage = readTask.Result.EndOfMessage;
|
||||
}
|
||||
foreach (var sptWebSocketMessageHandler in _messageHandlers)
|
||||
{
|
||||
sptWebSocketMessageHandler.OnSptMessage(sessionID, ws, readBytes.ToArray()).Wait();
|
||||
}
|
||||
}
|
||||
catch (OperationCanceledException e)
|
||||
{
|
||||
_logger.Info("WebSocket disconnecting, receive task finalized...");
|
||||
}
|
||||
catch (Exception e)
|
||||
{
|
||||
_logger.Error("Error reading data from WebSocket", e);
|
||||
}
|
||||
finally
|
||||
{
|
||||
readBytes.Clear();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public bool IsWebSocketConnected(string sessionID)
|
||||
{
|
||||
return _sockets.TryGetValue(sessionID, out var socket) && socket.State == WebSocketState.Open;
|
||||
}
|
||||
|
||||
public WebSocket GetSessionWebSocket(string sessionID)
|
||||
{
|
||||
return _sockets[sessionID];
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user