2022-12-20 12:20:52 -08:00
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Diagnostics;
|
|
|
|
|
using System.Net;
|
|
|
|
|
using System.Net.WebSockets;
|
|
|
|
|
using NLog;
|
|
|
|
|
using Ragon.Common;
|
|
|
|
|
using Ragon.Core.Server;
|
|
|
|
|
using Ragon.Server;
|
|
|
|
|
using Ragon.Server.NativeWebSockets;
|
|
|
|
|
|
|
|
|
|
namespace Ragon.Core;
|
|
|
|
|
|
|
|
|
|
public class NativeWebSocketServer : INetworkServer
|
|
|
|
|
{
|
2022-12-25 03:13:01 -08:00
|
|
|
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
|
|
|
private INetworkListener _networkListener;
|
|
|
|
|
private Stack<ushort> _sequencer;
|
|
|
|
|
private Executor _executor;
|
|
|
|
|
private HttpListener _httpListener;
|
|
|
|
|
private WebSocketConnection[] _connections;
|
|
|
|
|
private List<WebSocketConnection> _activeConnections;
|
|
|
|
|
private CancellationTokenSource _cancellationTokenSource;
|
|
|
|
|
|
|
|
|
|
public NativeWebSocketServer(Executor executor)
|
|
|
|
|
{
|
|
|
|
|
_sequencer = new Stack<ushort>();
|
|
|
|
|
_connections = Array.Empty<WebSocketConnection>();
|
|
|
|
|
_activeConnections = new List<WebSocketConnection>();
|
|
|
|
|
_executor = executor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async void StartAccept(CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
while (!cancellationToken.IsCancellationRequested)
|
2022-12-20 12:20:52 -08:00
|
|
|
{
|
2022-12-25 03:13:01 -08:00
|
|
|
var context = await _httpListener.GetContextAsync();
|
|
|
|
|
if (!context.Request.IsWebSocketRequest) continue;
|
2022-12-20 12:20:52 -08:00
|
|
|
|
2022-12-25 03:13:01 -08:00
|
|
|
var webSocketContext = await context.AcceptWebSocketAsync(null);
|
|
|
|
|
var webSocket = webSocketContext.WebSocket;
|
2022-12-20 12:20:52 -08:00
|
|
|
|
2022-12-25 03:13:01 -08:00
|
|
|
var peerId = _sequencer.Pop();
|
|
|
|
|
var connection = new WebSocketConnection(webSocket, peerId);
|
|
|
|
|
|
|
|
|
|
_connections[peerId] = connection;
|
|
|
|
|
StartListen(connection, cancellationToken);
|
2022-12-20 12:20:52 -08:00
|
|
|
}
|
2022-12-25 03:13:01 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async void StartListen(WebSocketConnection connection, CancellationToken cancellationToken)
|
|
|
|
|
{
|
|
|
|
|
_activeConnections.Add(connection);
|
|
|
|
|
_networkListener.OnConnected(connection);
|
|
|
|
|
|
|
|
|
|
var webSocket = connection.Socket;
|
|
|
|
|
var bytes = new byte[2048];
|
|
|
|
|
var buffer = new Memory<byte>(bytes);
|
|
|
|
|
while (
|
|
|
|
|
webSocket.State == WebSocketState.Open ||
|
|
|
|
|
!cancellationToken.IsCancellationRequested)
|
2022-12-20 12:20:52 -08:00
|
|
|
{
|
2022-12-25 03:13:01 -08:00
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var result = await webSocket.ReceiveAsync(buffer, cancellationToken);
|
|
|
|
|
var dataRaw = buffer.Slice(0, result.Count);
|
2022-12-20 12:20:52 -08:00
|
|
|
|
2022-12-25 03:13:01 -08:00
|
|
|
_networkListener.OnData(connection, dataRaw.ToArray());
|
|
|
|
|
}
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
{
|
|
|
|
|
break;
|
|
|
|
|
}
|
2022-12-20 12:20:52 -08:00
|
|
|
}
|
|
|
|
|
|
2022-12-25 03:13:01 -08:00
|
|
|
_sequencer.Push(connection.Id);
|
|
|
|
|
_activeConnections.Remove(connection);
|
|
|
|
|
_networkListener.OnDisconnected(connection);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Poll()
|
|
|
|
|
{
|
|
|
|
|
Flush();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public async void Flush()
|
|
|
|
|
{
|
|
|
|
|
foreach (var conn in _activeConnections)
|
|
|
|
|
await conn.Flush();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Start(
|
|
|
|
|
INetworkListener listener,
|
|
|
|
|
NetworkConfiguration configuration
|
|
|
|
|
)
|
|
|
|
|
{
|
|
|
|
|
_networkListener = listener;
|
|
|
|
|
_cancellationTokenSource = new CancellationTokenSource();
|
|
|
|
|
|
|
|
|
|
var limit = (ushort)configuration.LimitConnections;
|
|
|
|
|
for (ushort i = limit; i != 0; i--)
|
|
|
|
|
_sequencer.Push(i);
|
|
|
|
|
|
|
|
|
|
_sequencer.Push(0);
|
|
|
|
|
|
|
|
|
|
_connections = new WebSocketConnection[configuration.LimitConnections];
|
|
|
|
|
|
|
|
|
|
_httpListener = new HttpListener();
|
|
|
|
|
_httpListener.Prefixes.Add($"http://127.0.0.1:{configuration.Port}/");
|
|
|
|
|
_httpListener.Start();
|
|
|
|
|
|
|
|
|
|
_executor.Run(() => StartAccept(_cancellationTokenSource.Token));
|
|
|
|
|
|
|
|
|
|
var protocolDecoded = RagonVersion.Parse(configuration.Protocol);
|
|
|
|
|
_logger.Info($"Listen at http://*:{configuration.Port}/");
|
|
|
|
|
_logger.Info($"Protocol: {protocolDecoded}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Stop()
|
|
|
|
|
{
|
|
|
|
|
_cancellationTokenSource.Cancel();
|
|
|
|
|
_httpListener.Stop();
|
|
|
|
|
}
|
2022-12-20 12:20:52 -08:00
|
|
|
}
|