Added Secure WebSockets

This commit is contained in:
Jeroen 2023-11-11 10:46:31 +01:00
parent 04fd2329f1
commit d6f0c7b5af
5 changed files with 333 additions and 9 deletions

View File

@ -15,10 +15,10 @@
<PackageProjectUrl>https://www.nuget.org/packages/EonaCat.Network/</PackageProjectUrl>
<PackageTags>EonaCat, Network, .NET Standard, EonaCatHelpers, Jeroen, Saey, Protocol, Quic, UDP, TCP, Web, Server</PackageTags>
<PackageReleaseNotes></PackageReleaseNotes>
<Description>EonaCat Networking library with Quic, TCP, UDP and a Webserver</Description>
<Version>1.0.9</Version>
<AssemblyVersion>1.0.0.9</AssemblyVersion>
<FileVersion>1.0.0.9</FileVersion>
<Description>EonaCat Networking library with Quic, TCP, UDP, WebSockets and a Webserver</Description>
<Version>1.1.0</Version>
<AssemblyVersion>1.1.0.0</AssemblyVersion>
<FileVersion>1.1.0.0</FileVersion>
<PackageIcon>icon.png</PackageIcon>
</PropertyGroup>
@ -52,9 +52,6 @@
<PackageReference Include="EonaCat.Json" Version="1.0.3" />
<PackageReference Include="EonaCat.LogSystem" Version="1.0.0" />
<PackageReference Include="EonaCat.Matchers" Version="1.0.0" />
</ItemGroup>
<ItemGroup>
<Folder Include="System\Sockets\" />
<PackageReference Include="System.Net.WebSockets" Version="4.3.0" />
</ItemGroup>
</Project>

View File

@ -13,5 +13,8 @@ namespace EonaCat.Network
public int Port => HasEndpoint ? ((IPEndPoint)EndPoint).Port : 0;
public Socket Socket { get; set; }
public bool IsIpv6 { get; set; }
public bool IsWebSocket { get; internal set; }
public string ClientId { get; internal set; }
public string ClientName { get; internal set; }
}
}

View File

@ -0,0 +1,133 @@
using System;
using System.Net.WebSockets;
using System.Security.Cryptography.X509Certificates;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
namespace EonaCat.Network
{
public class WebSocketSecureClient
{
private const int BUFFER_SIZE = 4096;
/// <summary>
/// OnConnect event
/// </summary>
public event Action<RemoteInfo> OnConnect;
/// <summary>
/// OnReceive event
/// </summary>
public event Action<RemoteInfo> OnReceive;
/// <summary>
/// OnDisconnect event
/// </summary>
public event Action<RemoteInfo> OnDisconnect;
/// <summary>
/// OnError event
/// </summary>
public event Action<Exception, string> OnError;
private ClientWebSocket _webSocket;
/// <summary>
/// The client name to be sent when connecting (optional).
/// </summary>
public string ClientName { get; set; }
/// <summary>
/// Create secure WebSocket client with certificate support
/// </summary>
/// <param name="uri"></param>
/// <param name="clientCertificate">The client certificate for the connection</param>
/// <param name="password">The password for the connection</param>
/// <returns></returns>
public Task ConnectAsync(string uri, X509Certificate2 clientCertificate = null, string password = null)
{
return CreateWebSocketClientAsync(uri, clientCertificate, password);
}
private async Task CreateWebSocketClientAsync(string uri, X509Certificate2 clientCertificate, string password)
{
_webSocket = new ClientWebSocket();
if (clientCertificate != null)
{
_webSocket.Options.ClientCertificates.Add(clientCertificate);
}
if (!string.IsNullOrEmpty(password))
{
var passwordBytes = Encoding.UTF8.GetBytes(password);
_webSocket.Options.SetRequestHeader("Password", Convert.ToBase64String(passwordBytes));
}
if (!string.IsNullOrEmpty(ClientName))
{
_webSocket.Options.SetRequestHeader("ClientName", ClientName);
}
try
{
Uri serverUri = new Uri(uri);
await _webSocket.ConnectAsync(serverUri, CancellationToken.None).ConfigureAwait(false);
OnConnect?.Invoke(new RemoteInfo { IsWebSocket = true });
_ = StartReceivingAsync();
}
catch (Exception ex)
{
OnError?.Invoke(ex, $"Exception: {ex.Message}");
Disconnect();
}
}
private async Task StartReceivingAsync()
{
byte[] buffer = new byte[BUFFER_SIZE];
while (_webSocket.State == WebSocketState.Open)
{
try
{
var result = await _webSocket.ReceiveAsync(new ArraySegment<byte>(buffer), CancellationToken.None).ConfigureAwait(false);
if (result.Count > 0)
{
byte[] data = new byte[result.Count];
Buffer.BlockCopy(buffer, 0, data, 0, result.Count);
OnReceive?.Invoke(new RemoteInfo { IsWebSocket = true, Data = data });
}
else
{
break;
}
}
catch (Exception ex)
{
OnError?.Invoke(ex, $"Exception: {ex.Message}");
break;
}
}
OnDisconnect?.Invoke(new RemoteInfo { IsWebSocket = true });
}
/// <summary>
/// Send data
/// </summary>
/// <param name="data"></param>
/// <returns></returns>
public Task SendAsync(byte[] data)
{
return _webSocket.SendAsync(new ArraySegment<byte>(data), WebSocketMessageType.Binary, true, CancellationToken.None);
}
/// <summary>
/// Disconnect
/// </summary>
public void Disconnect()
{
_webSocket?.CloseAsync(WebSocketCloseStatus.NormalClosure, "Disconnecting", CancellationToken.None);
}
}
}

View File

@ -0,0 +1,190 @@
using System;
using System.Collections.Generic;
using System.Net;
using System.Net.WebSockets;
using System.Security.Cryptography.X509Certificates;
using System.Text;
using System.Threading;
using System.Threading.Tasks;
namespace EonaCat.Network
{
public class WebSocketSecureServer
{
private const int BUFFER_SIZE = 4096;
/// <summary>
/// OnConnect event
/// </summary>
public event Action<RemoteInfo> OnConnect;
/// <summary>
/// OnReceive event
/// </summary>
public event Action<RemoteInfo> OnReceive;
/// <summary>
/// OnSend event
/// </summary>
public event Action<RemoteInfo> OnSend;
/// <summary>
/// OnDisconnect event
/// </summary>
public event Action<RemoteInfo> OnDisconnect;
/// <summary>
/// OnError event
/// </summary>
public event Action<Exception, string> OnError;
private readonly HttpListener _listener;
private CancellationTokenSource _cancellationTokenSource;
private Task _acceptTask;
private readonly X509Certificate2 _serverCertificate;
private readonly string _requiredPassword;
private bool _passwordProtectionEnabled;
private readonly Dictionary<string, WebSocket> _connectedClients = new Dictionary<string, WebSocket>();
private readonly Dictionary<string, string> _clientNames = new Dictionary<string, string>();
/// <summary>
/// Create secure WebSocket server with certificate support
/// </summary>
/// <param name="uriPrefixes">Array of URI prefixes to listen on, e.g., "https://localhost:8443/"</param>
/// <param name="serverCertificate">The server certificate for HTTPS</param>
/// <param name="requiredPassword">The password required from clients for the connection (optional)</param>
public WebSocketSecureServer(List<string> uriPrefixes, X509Certificate2 serverCertificate = null, string requiredPassword = null)
{
_listener = new HttpListener();
_serverCertificate = serverCertificate;
_requiredPassword = requiredPassword;
_passwordProtectionEnabled = !string.IsNullOrEmpty(requiredPassword);
foreach (var uriPrefix in uriPrefixes)
{
_listener.Prefixes.Add(uriPrefix);
}
}
/// <summary>
/// Start secure WebSocket server
/// </summary>
/// <param name="cancellationToken"></param>
/// <returns></returns>
public Task StartAsync(CancellationToken cancellationToken = default)
{
_listener.Start();
_cancellationTokenSource = CancellationTokenSource.CreateLinkedTokenSource(cancellationToken);
_acceptTask = AcceptConnectionsAsync(_cancellationTokenSource.Token);
return _acceptTask;
}
private async Task AcceptConnectionsAsync(CancellationToken cancellationToken)
{
while (!cancellationToken.IsCancellationRequested)
{
try
{
var context = await _listener.GetContextAsync().ConfigureAwait(false);
if (context.Request.IsWebSocketRequest)
{
if (_passwordProtectionEnabled)
{
var password = context.Request.Headers["Password"];
if (password != _requiredPassword)
{
context.Response.StatusCode = 401; // Unauthorized
context.Response.Close();
continue;
}
}
string clientName = null;
foreach (var key in context.Request.Headers.AllKeys)
{
if (key == "ClientName")
{
clientName = context.Request.Headers["ClientName"];
break;
}
}
var webSocketContext = await context.AcceptWebSocketAsync(subProtocol: null).ConfigureAwait(false);
_ = HandleWebSocketConnectionAsync(webSocketContext.WebSocket, clientName, cancellationToken);
}
else
{
context.Response.StatusCode = 400;
context.Response.Close();
}
}
catch (Exception ex)
{
OnError?.Invoke(ex, $"Exception: {ex.Message}");
}
}
}
private async Task HandleWebSocketConnectionAsync(WebSocket webSocket, string clientName, CancellationToken cancellationToken)
{
string clientId = Guid.NewGuid().ToString();
_connectedClients.Add(clientId, webSocket);
if (!string.IsNullOrEmpty(clientName))
{
_clientNames.TryAdd(clientId, clientName);
}
OnConnect?.Invoke(new RemoteInfo { IsWebSocket = true, ClientId = clientId, ClientName = clientName });
byte[] buffer = new byte[BUFFER_SIZE];
while (webSocket.State == WebSocketState.Open)
{
try
{
var result = await webSocket.ReceiveAsync(new ArraySegment<byte>(buffer), cancellationToken).ConfigureAwait(false);
if (result.Count > 0)
{
byte[] data = new byte[result.Count];
Buffer.BlockCopy(buffer, 0, data, 0, result.Count);
OnReceive?.Invoke(new RemoteInfo { IsWebSocket = true, ClientId = clientId, ClientName = clientName, Data = data });
}
else
{
break;
}
}
catch (Exception ex)
{
OnError?.Invoke(ex, $"Exception: {ex.Message}");
break;
}
}
OnDisconnect?.Invoke(new RemoteInfo { IsWebSocket = true, ClientId = clientId, ClientName = clientName });
_connectedClients.Remove(clientId);
_clientNames.Remove(clientId);
}
/// <summary>
/// Stop secure WebSocket server
/// </summary>
/// <returns></returns>
public async Task StopAsync()
{
_cancellationTokenSource.Cancel();
_listener.Stop();
if (_acceptTask != null)
{
try
{
await _acceptTask.ConfigureAwait(false);
}
catch (AggregateException) { }
}
}
}
}

View File

@ -2,4 +2,5 @@
------------
EonaCat Network Library
EonaCat Network Library