prepare to send different data per client

This commit is contained in:
Vinzenz Schroeter 2024-04-15 20:34:23 +02:00
parent fcd84d2c83
commit fbaad86555
15 changed files with 245 additions and 128 deletions

View file

@ -1,13 +1,16 @@
using DisplayCommands;
using TanksServer.GameLogic; using TanksServer.GameLogic;
namespace TanksServer.Graphics; namespace TanksServer.Graphics;
internal sealed class DrawBulletsStep(BulletManager bullets) : IDrawStep internal sealed class DrawBulletsStep(BulletManager bullets) : IDrawStep
{ {
public void Draw(PixelGrid buffer) public void Draw(GamePixelGrid pixels)
{ {
foreach (var position in bullets.GetAll().Select(b => b.Position.ToPixelPosition())) foreach (var bullet in bullets.GetAll())
buffer[(ushort)position.X, (ushort)position.Y] = true; {
var position = bullet.Position.ToPixelPosition();
pixels[position.X, position.Y].EntityType = GamePixelEntityType.Bullet;
pixels[position.X, position.Y].BelongsTo = bullet.Owner;
}
} }
} }

View file

@ -1,11 +1,10 @@
using DisplayCommands;
using TanksServer.GameLogic; using TanksServer.GameLogic;
namespace TanksServer.Graphics; namespace TanksServer.Graphics;
internal sealed class DrawMapStep(MapService map) : IDrawStep internal sealed class DrawMapStep(MapService map) : IDrawStep
{ {
public void Draw(PixelGrid buffer) public void Draw(GamePixelGrid pixels)
{ {
for (ushort y = 0; y < MapService.PixelsPerColumn; y++) for (ushort y = 0; y < MapService.PixelsPerColumn; y++)
for (ushort x = 0; x < MapService.PixelsPerRow; x++) for (ushort x = 0; x < MapService.PixelsPerRow; x++)
@ -13,7 +12,8 @@ internal sealed class DrawMapStep(MapService map) : IDrawStep
var pixel = new PixelPosition(x, y); var pixel = new PixelPosition(x, y);
if (!map.Current.IsWall(pixel)) if (!map.Current.IsWall(pixel))
continue; continue;
buffer[x, y] = true;
pixels[x, y].EntityType = GamePixelEntityType.Wall;
} }
} }
} }

View file

@ -27,7 +27,7 @@ internal sealed class DrawTanksStep : IDrawStep
_tankSpriteWidth = tankImage.Width; _tankSpriteWidth = tankImage.Width;
} }
public void Draw(PixelGrid buffer) public void Draw(GamePixelGrid pixels)
{ {
foreach (var tank in _tanks) foreach (var tank in _tanks)
{ {
@ -40,7 +40,8 @@ internal sealed class DrawTanksStep : IDrawStep
continue; continue;
var (x, y) = tankPosition.GetPixelRelative(dx, dy); var (x, y) = tankPosition.GetPixelRelative(dx, dy);
buffer[(ushort)x, (ushort)y] = true; pixels[x, y].EntityType = GamePixelEntityType.Tank;
pixels[x, y].BelongsTo = tank.Owner;
} }
} }
} }

View file

@ -0,0 +1,21 @@
namespace TanksServer.Graphics;
internal sealed class GamePixel
{
public Player? BelongsTo { get; set; }
public GamePixelEntityType? EntityType { get; set; }
public void Clear()
{
BelongsTo = null;
EntityType = null;
}
}
internal enum GamePixelEntityType : byte
{
Wall = 0x0,
Tank = 0x1,
Bullet = 0x2
}

View file

@ -0,0 +1,47 @@
using System.Collections;
using System.Diagnostics;
namespace TanksServer.Graphics;
internal sealed class GamePixelGrid : IEnumerable<GamePixel>
{
public int Width { get; }
public int Height { get; }
private readonly GamePixel[,] _pixels;
public GamePixelGrid(int width, int height)
{
Width = width;
Height = height;
_pixels = new GamePixel[height, width];
for (var row = 0; row < height; row++)
for (var column = 0; column < width; column++)
_pixels[row, column] = new GamePixel();
}
public GamePixel this[int x, int y]
{
get
{
Debug.Assert(y * Width + x < _pixels.Length);
return _pixels[y, x];
}
}
public void Clear()
{
foreach (var pixel in _pixels)
pixel.Clear();
}
IEnumerator IEnumerable.GetEnumerator() => GetEnumerator();
public IEnumerator<GamePixel> GetEnumerator()
{
for (var row = 0; row < Height; row++)
for (var column = 0; column < Width; column++)
yield return _pixels[row, column];
}
}

View file

@ -5,18 +5,30 @@ namespace TanksServer.Graphics;
internal sealed class GeneratePixelsTickStep( internal sealed class GeneratePixelsTickStep(
IEnumerable<IDrawStep> drawSteps, IEnumerable<IDrawStep> drawSteps,
LastFinishedFrameProvider lastFrameProvider IEnumerable<IFrameConsumer> consumers
) : ITickStep ) : ITickStep
{ {
private readonly List<IDrawStep> _drawSteps = drawSteps.ToList(); private readonly List<IDrawStep> _drawSteps = drawSteps.ToList();
private readonly PixelGrid _drawGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn); private readonly List<IFrameConsumer> _consumers = consumers.ToList();
public Task TickAsync() private readonly PixelGrid _observerPixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
private readonly GamePixelGrid _gamePixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
public async Task TickAsync()
{ {
_drawGrid.Clear(); _gamePixelGrid.Clear();
foreach (var step in _drawSteps) foreach (var step in _drawSteps)
step.Draw(_drawGrid); step.Draw(_gamePixelGrid);
lastFrameProvider.LastFrame = _drawGrid;
return Task.CompletedTask; _observerPixelGrid.Clear();
for (var y = 0; y < MapService.PixelsPerColumn; y++)
for (var x = 0; x < MapService.PixelsPerRow; x++)
{
if (_gamePixelGrid[x, y].EntityType.HasValue)
_observerPixelGrid[(ushort)x, (ushort)y] = true;
}
foreach (var consumer in _consumers)
await consumer.OnFrameDoneAsync(_gamePixelGrid, _observerPixelGrid);
} }
} }

View file

@ -4,5 +4,5 @@ namespace TanksServer.Graphics;
internal interface IDrawStep internal interface IDrawStep
{ {
void Draw(PixelGrid buffer); void Draw(GamePixelGrid pixels);
} }

View file

@ -0,0 +1,8 @@
using DisplayCommands;
namespace TanksServer.Graphics;
internal interface IFrameConsumer
{
Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels);
}

View file

@ -1,14 +0,0 @@
using DisplayCommands;
namespace TanksServer.Graphics;
internal sealed class LastFinishedFrameProvider
{
private PixelGrid? _lastFrame;
public PixelGrid LastFrame
{
get => _lastFrame ?? throw new InvalidOperationException("first frame not yet drawn");
set => _lastFrame = value;
}
}

View file

@ -7,8 +7,8 @@ internal sealed class ByteChannelWebSocket(WebSocket socket, ILogger logger, int
{ {
private readonly byte[] _buffer = new byte[messageSize]; private readonly byte[] _buffer = new byte[messageSize];
public ValueTask SendAsync(ReadOnlyMemory<byte> data) => public ValueTask SendAsync(ReadOnlyMemory<byte> data, bool endOfMessage = true) =>
socket.SendAsync(data, WebSocketMessageType.Binary, true, CancellationToken.None); socket.SendAsync(data, WebSocketMessageType.Binary, endOfMessage, CancellationToken.None);
public async IAsyncEnumerable<Memory<byte>> ReadAllAsync() public async IAsyncEnumerable<Memory<byte>> ReadAllAsync()
{ {

View file

@ -1,15 +1,15 @@
using System.Diagnostics; using System.Diagnostics;
using System.Net.WebSockets; using System.Net.WebSockets;
using System.Threading.Channels;
using DisplayCommands; using DisplayCommands;
using Microsoft.Extensions.Hosting; using Microsoft.Extensions.Hosting;
using TanksServer.Graphics;
namespace TanksServer.Interactivity; namespace TanksServer.Interactivity;
internal sealed class ClientScreenServer( internal sealed class ClientScreenServer(
ILogger<ClientScreenServer> logger, ILogger<ClientScreenServer> logger,
ILoggerFactory loggerFactory ILoggerFactory loggerFactory
) : IHostedLifecycleService ) : IHostedLifecycleService, IFrameConsumer
{ {
private readonly ConcurrentDictionary<ClientScreenServerConnection, byte> _connections = new(); private readonly ConcurrentDictionary<ClientScreenServerConnection, byte> _connections = new();
private bool _closing; private bool _closing;
@ -37,76 +37,21 @@ internal sealed class ClientScreenServer(
return connection.Done; return connection.Done;
} }
private void Remove(ClientScreenServerConnection connection) public void Remove(ClientScreenServerConnection connection) => _connections.TryRemove(connection, out _);
{
_connections.TryRemove(connection, out _);
}
public IEnumerable<ClientScreenServerConnection> GetConnections() => _connections.Keys; public IEnumerable<ClientScreenServerConnection> GetConnections() => _connections.Keys;
public Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels)
{
var tasks = _connections.Keys
.Select(c => c.SendAsync(observerPixels, gamePixelGrid));
return Task.WhenAll(tasks);
}
public Task StartAsync(CancellationToken cancellationToken) => Task.CompletedTask; public Task StartAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask; public Task StopAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartedAsync(CancellationToken cancellationToken) => Task.CompletedTask; public Task StartedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartingAsync(CancellationToken cancellationToken) => Task.CompletedTask; public Task StartingAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StoppedAsync(CancellationToken cancellationToken) => Task.CompletedTask; public Task StoppedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
internal sealed class ClientScreenServerConnection : IDisposable
{
private readonly ByteChannelWebSocket _channel;
private readonly ILogger<ClientScreenServerConnection> _logger;
private readonly ClientScreenServer _server;
private readonly SemaphoreSlim _wantedFrames = new(1);
public ClientScreenServerConnection(WebSocket webSocket,
ILogger<ClientScreenServerConnection> logger,
ClientScreenServer server)
{
_server = server;
_logger = logger;
_channel = new ByteChannelWebSocket(webSocket, logger, 0);
Done = ReceiveAsync();
}
public Task Done { get; }
public void Dispose()
{
_wantedFrames.Dispose();
Done.Dispose();
}
public async Task SendAsync(PixelGrid pixels)
{
if (!await _wantedFrames.WaitAsync(TimeSpan.Zero))
{
_logger.LogTrace("client does not want a frame yet");
return;
}
_logger.LogTrace("sending");
try
{
await _channel.SendAsync(pixels.Data);
}
catch (WebSocketException ex)
{
_logger.LogWarning(ex, "send failed");
}
}
private async Task ReceiveAsync()
{
await foreach (var _ in _channel.ReadAllAsync())
_wantedFrames.Release();
_logger.LogTrace("done receiving");
_server.Remove(this);
}
public Task CloseAsync()
{
_logger.LogDebug("closing connection");
return _channel.CloseAsync();
}
}
} }

View file

@ -0,0 +1,115 @@
using System.Diagnostics;
using System.Net.WebSockets;
using DisplayCommands;
using TanksServer.GameLogic;
using TanksServer.Graphics;
namespace TanksServer.Interactivity;
internal sealed class ClientScreenServerConnection : IDisposable
{
private readonly ByteChannelWebSocket _channel;
private readonly ILogger<ClientScreenServerConnection> _logger;
private readonly ClientScreenServer _server;
private readonly SemaphoreSlim _wantedFrames = new(1);
private readonly Guid? _playerGuid = null;
private readonly PlayerScreenData? _playerScreenData = null;
public ClientScreenServerConnection(
WebSocket webSocket,
ILogger<ClientScreenServerConnection> logger,
ClientScreenServer server,
Guid? playerGuid = null
)
{
_server = server;
_logger = logger;
_playerGuid = playerGuid;
if (playerGuid.HasValue)
_playerScreenData = new PlayerScreenData();
_channel = new ByteChannelWebSocket(webSocket, logger, 0);
Done = ReceiveAsync();
}
public Task Done { get; }
public void Dispose()
{
_wantedFrames.Dispose();
Done.Dispose();
}
public async Task SendAsync(PixelGrid pixels, GamePixelGrid gamePixelGrid)
{
if (!await _wantedFrames.WaitAsync(TimeSpan.Zero))
{
_logger.LogTrace("client does not want a frame yet");
return;
}
if (_playerScreenData != null)
RefreshPlayerSpecificData(gamePixelGrid);
_logger.LogTrace("sending");
try
{
await _channel.SendAsync(pixels.Data, _playerScreenData == null);
if (_playerScreenData != null)
await _channel.SendAsync(_playerScreenData.GetPacket());
}
catch (WebSocketException ex)
{
_logger.LogWarning(ex, "send failed");
}
}
private void RefreshPlayerSpecificData(GamePixelGrid gamePixelGrid)
{
Debug.Assert(_playerScreenData != null);
_playerScreenData.Clear();
foreach (var gamePixel in gamePixelGrid)
{
if (!gamePixel.EntityType.HasValue)
continue;
_playerScreenData.Add(gamePixel.EntityType.Value, gamePixel.BelongsTo?.Id == _playerGuid);
}
}
private async Task ReceiveAsync()
{
await foreach (var _ in _channel.ReadAllAsync())
_wantedFrames.Release();
_logger.LogTrace("done receiving");
_server.Remove(this);
}
public Task CloseAsync()
{
_logger.LogDebug("closing connection");
return _channel.CloseAsync();
}
}
internal sealed class PlayerScreenData
{
private Memory<byte> _data = new byte[MapService.PixelsPerRow * MapService.PixelsPerColumn];
public int Count { get; private set; } = 0;
public void Clear() => Count = 0;
public ReadOnlyMemory<byte> GetPacket() => _data[..Count];
public void Add(GamePixelEntityType entityKind, bool isCurrentPlayer)
{
var result = (byte)(isCurrentPlayer ? 0x1b : 0x0b);
var kind = (byte)entityKind;
Debug.Assert(kind < 3);
result += (byte)(kind << 2);
_data.Span[Count] = result;
Count++;
}
}

View file

@ -1,18 +0,0 @@
using TanksServer.GameLogic;
using TanksServer.Graphics;
namespace TanksServer.Interactivity;
internal sealed class SendToClientScreen(
ClientScreenServer clientScreenServer,
LastFinishedFrameProvider lastFinishedFrameProvider
) : ITickStep
{
public Task TickAsync()
{
var tasks = clientScreenServer
.GetConnections()
.Select(c => c.SendAsync(lastFinishedFrameProvider.LastFrame));
return Task.WhenAll(tasks);
}
}

View file

@ -6,14 +6,13 @@ using TanksServer.Graphics;
namespace TanksServer.Interactivity; namespace TanksServer.Interactivity;
internal sealed class SendToServicePointDisplay : ITickStep internal sealed class SendToServicePointDisplay : IFrameConsumer
{ {
private const int ScoresWidth = 12; private const int ScoresWidth = 12;
private const int ScoresHeight = 20; private const int ScoresHeight = 20;
private const int ScoresPlayerRows = ScoresHeight - 5; private const int ScoresPlayerRows = ScoresHeight - 5;
private readonly IDisplayConnection _displayConnection; private readonly IDisplayConnection _displayConnection;
private readonly LastFinishedFrameProvider _lastFinishedFrameProvider;
private readonly ILogger<SendToServicePointDisplay> _logger; private readonly ILogger<SendToServicePointDisplay> _logger;
private readonly PlayerServer _players; private readonly PlayerServer _players;
private readonly Cp437Grid _scoresBuffer; private readonly Cp437Grid _scoresBuffer;
@ -22,13 +21,11 @@ internal sealed class SendToServicePointDisplay : ITickStep
private DateTime _nextFailLog = DateTime.Now; private DateTime _nextFailLog = DateTime.Now;
public SendToServicePointDisplay( public SendToServicePointDisplay(
LastFinishedFrameProvider lastFinishedFrameProvider,
PlayerServer players, PlayerServer players,
ILogger<SendToServicePointDisplay> logger, ILogger<SendToServicePointDisplay> logger,
IDisplayConnection displayConnection IDisplayConnection displayConnection
) )
{ {
_lastFinishedFrameProvider = lastFinishedFrameProvider;
_players = players; _players = players;
_logger = logger; _logger = logger;
_displayConnection = displayConnection; _displayConnection = displayConnection;
@ -45,17 +42,16 @@ internal sealed class SendToServicePointDisplay : ITickStep
}; };
} }
public async Task TickAsync() public async Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels)
{ {
RefreshScores(); RefreshScores();
try try
{ {
await _displayConnection.SendCp437DataAsync(MapService.TilesPerRow, 0, _scoresBuffer); await _displayConnection.SendCp437DataAsync(MapService.TilesPerRow, 0, _scoresBuffer);
var currentFrame = _lastFinishedFrameProvider.LastFrame; if (_lastSentFrame == observerPixels)
if (_lastSentFrame == currentFrame)
return; return;
_lastSentFrame = currentFrame; _lastSentFrame = observerPixels;
await _displayConnection.SendBitmapLinearWindowAsync(0, 0, _lastSentFrame); await _displayConnection.SendBitmapLinearWindowAsync(0, 0, _lastSentFrame);
} }
catch (SocketException ex) catch (SocketException ex)

View file

@ -123,7 +123,6 @@ public static class Program
builder.Services.AddSingleton<ControlsServer>(); builder.Services.AddSingleton<ControlsServer>();
builder.Services.AddSingleton<PlayerServer>(); builder.Services.AddSingleton<PlayerServer>();
builder.Services.AddSingleton<ClientScreenServer>(); builder.Services.AddSingleton<ClientScreenServer>();
builder.Services.AddSingleton<LastFinishedFrameProvider>();
builder.Services.AddSingleton<SpawnQueue>(); builder.Services.AddSingleton<SpawnQueue>();
builder.Services.AddHostedService<GameTickWorker>(); builder.Services.AddHostedService<GameTickWorker>();
@ -138,13 +137,15 @@ public static class Program
builder.Services.AddSingleton<ITickStep, ShootFromTanks>(); builder.Services.AddSingleton<ITickStep, ShootFromTanks>();
builder.Services.AddSingleton<ITickStep, SpawnNewTanks>(); builder.Services.AddSingleton<ITickStep, SpawnNewTanks>();
builder.Services.AddSingleton<ITickStep, GeneratePixelsTickStep>(); builder.Services.AddSingleton<ITickStep, GeneratePixelsTickStep>();
builder.Services.AddSingleton<ITickStep, SendToServicePointDisplay>();
builder.Services.AddSingleton<ITickStep, SendToClientScreen>();
builder.Services.AddSingleton<IDrawStep, DrawMapStep>(); builder.Services.AddSingleton<IDrawStep, DrawMapStep>();
builder.Services.AddSingleton<IDrawStep, DrawTanksStep>(); builder.Services.AddSingleton<IDrawStep, DrawTanksStep>();
builder.Services.AddSingleton<IDrawStep, DrawBulletsStep>(); builder.Services.AddSingleton<IDrawStep, DrawBulletsStep>();
builder.Services.AddSingleton<IFrameConsumer, SendToServicePointDisplay>();
builder.Services.AddSingleton<IFrameConsumer, ClientScreenServer>(sp =>
sp.GetRequiredService<ClientScreenServer>());
builder.Services.Configure<TanksConfiguration>( builder.Services.Configure<TanksConfiguration>(
builder.Configuration.GetSection("Tanks")); builder.Configuration.GetSection("Tanks"));
builder.Services.Configure<PlayersConfiguration>( builder.Services.Configure<PlayersConfiguration>(