servicepoint-tanks/TanksServer/ClientScreenServer.cs

113 lines
3.6 KiB
C#
Raw Normal View History

2024-04-07 17:17:11 +02:00
using System.Collections.Concurrent;
using System.Diagnostics;
2024-04-06 16:38:26 +02:00
using System.Net.WebSockets;
2024-04-06 20:32:54 +02:00
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging;
2024-04-07 11:19:14 +02:00
using TanksServer.Helpers;
using TanksServer.Services;
2024-04-06 16:38:26 +02:00
namespace TanksServer;
2024-04-06 20:32:54 +02:00
internal sealed class ClientScreenServer(
ILogger<ClientScreenServer> logger,
2024-04-06 21:15:26 +02:00
ILoggerFactory loggerFactory,
2024-04-07 13:02:49 +02:00
PixelDrawer drawer
2024-04-06 21:15:26 +02:00
) : IHostedLifecycleService, ITickStep
2024-04-06 16:38:26 +02:00
{
2024-04-07 17:17:11 +02:00
private readonly ConcurrentDictionary<ClientScreenServerConnection, byte> _connections = new();
private bool _closing;
2024-04-06 16:38:26 +02:00
2024-04-06 20:32:54 +02:00
public Task HandleClient(WebSocket socket)
2024-04-06 16:38:26 +02:00
{
2024-04-07 17:17:11 +02:00
if (_closing)
{
logger.LogWarning("ignoring request because connections are closing");
return Task.CompletedTask;
}
2024-04-06 20:32:54 +02:00
logger.LogDebug("HandleClient");
var connection =
2024-04-06 21:15:26 +02:00
new ClientScreenServerConnection(socket, loggerFactory.CreateLogger<ClientScreenServerConnection>(), this);
2024-04-07 17:17:11 +02:00
var added = _connections.TryAdd(connection, 0);
Debug.Assert(added);
2024-04-06 20:32:54 +02:00
return connection.Done;
2024-04-06 16:38:26 +02:00
}
2024-04-06 20:32:54 +02:00
public Task StoppingAsync(CancellationToken cancellationToken)
2024-04-06 16:38:26 +02:00
{
2024-04-06 20:32:54 +02:00
logger.LogInformation("closing connections");
2024-04-07 17:17:11 +02:00
_closing = true;
return Task.WhenAll(_connections.Keys.Select(c => c.CloseAsync()));
2024-04-06 16:38:26 +02:00
}
2024-04-06 21:15:26 +02:00
public Task TickAsync()
{
logger.LogTrace("Sending buffer to {} clients", _connections.Count);
2024-04-07 17:17:11 +02:00
return Task.WhenAll(_connections.Keys.Select(c => c.SendAsync(drawer.LastFrame)));
2024-04-06 21:15:26 +02:00
}
2024-04-06 20:32:54 +02:00
public Task StartAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartingAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StoppedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
2024-04-07 17:17:11 +02:00
private void Remove(ClientScreenServerConnection connection) => _connections.TryRemove(connection, out _);
private sealed class ClientScreenServerConnection: IDisposable
2024-04-06 21:15:26 +02:00
{
2024-04-07 17:17:11 +02:00
private readonly ByteChannelWebSocket _channel;
private readonly SemaphoreSlim _wantedFrames = new(1);
private readonly ClientScreenServer _server;
private readonly ILogger<ClientScreenServerConnection> _logger;
2024-04-06 16:38:26 +02:00
2024-04-07 17:17:11 +02:00
public ClientScreenServerConnection(WebSocket webSocket,
ILogger<ClientScreenServerConnection> logger,
ClientScreenServer server)
2024-04-06 21:15:26 +02:00
{
2024-04-07 17:17:11 +02:00
_server = server;
_logger = logger;
_channel = new(webSocket, logger, 0);
Done = ReceiveAsync();
2024-04-06 21:15:26 +02:00
}
2024-04-06 16:38:26 +02:00
2024-04-07 17:17:11 +02:00
public async Task SendAsync(DisplayPixelBuffer buf)
2024-04-06 21:15:26 +02:00
{
2024-04-07 17:17:11 +02:00
if (await _wantedFrames.WaitAsync(TimeSpan.Zero))
{
_logger.LogTrace("sending");
await _channel.Writer.WriteAsync(buf.Data);
}
else
{
_logger.LogTrace("client does not want a frame yet");
}
2024-04-06 21:15:26 +02:00
}
2024-04-06 20:32:54 +02:00
2024-04-07 17:17:11 +02:00
private async Task ReceiveAsync()
2024-04-06 21:15:26 +02:00
{
2024-04-07 17:17:11 +02:00
await foreach (var _ in _channel.Reader.ReadAllAsync())
{
_wantedFrames.Release();
}
_logger.LogTrace("done receiving");
_server.Remove(this);
}
public Task CloseAsync()
{
_logger.LogDebug("closing connection");
return _channel.CloseAsync();
}
public Task Done { get; }
public void Dispose()
{
_wantedFrames.Dispose();
Done.Dispose();
2024-04-06 21:15:26 +02:00
}
2024-04-06 16:38:26 +02:00
}
}