using System.Net.WebSockets; using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Logging; using TanksServer.Helpers; using TanksServer.Models; namespace TanksServer.Servers; internal sealed class ControlsServer(ILogger logger, ILoggerFactory loggerFactory) : IHostedLifecycleService { private readonly List _connections = new(); public Task HandleClient(WebSocket ws, Player player) { logger.LogDebug("control client connected {}", player.Id); var clientLogger = loggerFactory.CreateLogger(); var sock = new ControlsServerConnection(ws, clientLogger, this, player); _connections.Add(sock); return sock.Done; } public Task StoppingAsync(CancellationToken cancellationToken) { return Task.WhenAll(_connections.Select(c => c.CloseAsync())); } 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; private void Remove(ControlsServerConnection connection) { _connections.Remove(connection); } private sealed class ControlsServerConnection { private readonly ByteChannelWebSocket _binaryWebSocket; private readonly ILogger _logger; private readonly ControlsServer _server; private readonly Player _player; public ControlsServerConnection(WebSocket socket, ILogger logger, ControlsServer server, Player player) { _logger = logger; _server = server; _player = player; _binaryWebSocket = new(socket, logger, 2); Done = ReceiveAsync(); } public Task Done { get; } private enum MessageType : byte { Enable = 0x01, Disable = 0x02, } private enum InputType : byte { Forward = 0x01, Backward = 0x02, Left = 0x03, Right = 0x04, Shoot = 0x05 } private async Task ReceiveAsync() { await foreach (var buffer in _binaryWebSocket.Reader.ReadAllAsync()) { var type = (MessageType)buffer[0]; var control = (InputType)buffer[1]; _logger.LogTrace("player input {} {} {}", _player.Id, type, control); var isEnable = type switch { MessageType.Enable => true, MessageType.Disable => false, _ => throw new ArgumentException("invalid message type") }; switch (control) { case InputType.Forward: _player.Controls.Forward = isEnable; break; case InputType.Backward: _player.Controls.Backward = isEnable; break; case InputType.Left: _player.Controls.TurnLeft = isEnable; break; case InputType.Right: _player.Controls.TurnRight = isEnable; break; case InputType.Shoot: _player.Controls.Shoot = isEnable; break; default: throw new ArgumentException("invalid control type"); } } _server.Remove(this); } public Task CloseAsync() => _binaryWebSocket.CloseAsync(); } }