diff --git a/BackendTests/Controllers/GameControllerTests.cs b/BackendTests/Controllers/GameControllerTests.cs index 5325506..28ff4c2 100644 --- a/BackendTests/Controllers/GameControllerTests.cs +++ b/BackendTests/Controllers/GameControllerTests.cs @@ -6,7 +6,6 @@ using Microsoft.Extensions.Logging; using NSubstitute; using pacMan.Controllers; using pacMan.Game; -using pacMan.Interfaces; using pacMan.Services; using pacMan.Utils; @@ -16,16 +15,16 @@ public class GameControllerTests { private IActionService _actionService = null!; private GameController _controller = null!; - private IWebSocketService _webSocketService = null!; + private GameService _gameService = null!; [SetUp] public void Setup() { - _webSocketService = Substitute.For(Substitute.For>()); + _gameService = Substitute.For(Substitute.For>()); _actionService = Substitute.For( - Substitute.For>(), _webSocketService + Substitute.For>(), _gameService ); - _controller = new GameController(Substitute.For>(), _webSocketService, _actionService); + _controller = new GameController(Substitute.For>(), _gameService, _actionService); } [Test] @@ -42,18 +41,14 @@ public class GameControllerTests var result = runMethod!.Invoke(_controller, new object[] { wssResult, data }); if (result is ArraySegment resultSegment) - { Assert.Multiple(() => { Assert.That(resultSegment, Has.Count.EqualTo(data.Length)); Assert.That(Encoding.UTF8.GetString(resultSegment.ToArray()), Is.EqualTo(data.GetString(data.Length))); }); - // TODO not working, works like a normal method call - // _actionService.ReceivedWithAnyArgs().DoAction(default!); - } + // TODO not working, works like a normal method call + // _actionService.ReceivedWithAnyArgs().DoAction(default!); else - { Assert.Fail("Result is not an ArraySegment"); - } } } diff --git a/BackendTests/Services/ActionServiceTests.cs b/BackendTests/Services/ActionServiceTests.cs index ae3b42e..beaac86 100644 --- a/BackendTests/Services/ActionServiceTests.cs +++ b/BackendTests/Services/ActionServiceTests.cs @@ -4,7 +4,6 @@ using Microsoft.Extensions.Logging; using NSubstitute; using pacMan.Game; using pacMan.Game.Items; -using pacMan.Interfaces; using pacMan.Services; namespace BackendTests.Services; @@ -16,13 +15,13 @@ public class ActionServiceTests private readonly Player _whitePlayer = (Player)Players.Create("white"); private ActionMessage _blackMessage = null!; + private GameService _gameService = null!; private ActionMessage _redMessage = null!; private IActionService _service = null!; private Queue _spawns = null!; private ActionMessage _whiteMessage = null!; - private IWebSocketService _wssSub = null!; [SetUp] public void Setup() @@ -43,8 +42,8 @@ public class ActionServiceTests Action = GameAction.PlayerInfo, Data = JsonSerializer.Serialize(new { Player = _redPlayer, Spawns = CreateQueue() }) }; - _wssSub = Substitute.For(Substitute.For>()); - _service = new ActionService(Substitute.For>(), _wssSub); + _gameService = Substitute.For(Substitute.For>()); + _service = new ActionService(Substitute.For>(), _gameService); } private static Queue CreateQueue() => diff --git a/BackendTests/Services/WebSocketServiceTests.cs b/BackendTests/Services/WebSocketServiceTests.cs index 006a0d3..7844f49 100644 --- a/BackendTests/Services/WebSocketServiceTests.cs +++ b/BackendTests/Services/WebSocketServiceTests.cs @@ -3,7 +3,6 @@ using BackendTests.TestUtils; using Microsoft.Extensions.Logging; using NSubstitute; using pacMan.Game; -using pacMan.Interfaces; using pacMan.Services; using pacMan.Utils; @@ -14,7 +13,7 @@ public class WebSocketServiceTests private readonly DirectionalPosition _spawn3By3Up = new() { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up }; - private IWebSocketService _service = null!; + private GameService _service = null!; private Queue _spawns = null!; @@ -22,7 +21,7 @@ public class WebSocketServiceTests [SetUp] public void SetUp() { - _service = new WebSocketService(Substitute.For>()); + _service = new GameService(Substitute.For>()); _spawns = new Queue(new[] { _spawn3By3Up, @@ -40,7 +39,7 @@ public class WebSocketServiceTests var segment = "test".ToArraySegment(); using var webSocket = Substitute.For(); - _service.Send(webSocket, segment); + _service.Send(webSocket, segment).Wait(); webSocket.ReceivedWithAnyArgs().SendAsync(default, default, default, default); } diff --git a/pac-man-board-game/Controllers/GameController.cs b/pac-man-board-game/Controllers/GameController.cs index 5b6e160..a0b276c 100644 --- a/pac-man-board-game/Controllers/GameController.cs +++ b/pac-man-board-game/Controllers/GameController.cs @@ -1,7 +1,6 @@ using System.Net.WebSockets; using Microsoft.AspNetCore.Mvc; using pacMan.Game; -using pacMan.Interfaces; using pacMan.Services; using pacMan.Utils; @@ -13,8 +12,8 @@ public class GameController : GenericController // TODO reconnect using player i { private readonly IActionService _actionService; - public GameController(ILogger logger, IWebSocketService wsService, IActionService actionService) : - base(logger, wsService) => + public GameController(ILogger logger, GameService gameService, IActionService actionService) : + base(logger, gameService) => _actionService = actionService; [HttpGet("connect")] @@ -24,7 +23,7 @@ public class GameController : GenericController // TODO reconnect using player i public IEnumerable GetAllGames() { Logger.Log(LogLevel.Information, "Returning all games"); - return WsService.Games; + return GameService.Games; } diff --git a/pac-man-board-game/Controllers/GenericController.cs b/pac-man-board-game/Controllers/GenericController.cs index cba5bca..0ecfbb6 100644 --- a/pac-man-board-game/Controllers/GenericController.cs +++ b/pac-man-board-game/Controllers/GenericController.cs @@ -1,20 +1,20 @@ using System.Net.WebSockets; using Microsoft.AspNetCore.Mvc; -using pacMan.Interfaces; +using pacMan.Services; namespace pacMan.Controllers; -public abstract class GenericController : ControllerBase +public abstract class GenericController : ControllerBase // TODO only use WebSocketService in this class { private const int BufferSize = 1024 * 4; + protected readonly GameService GameService; protected readonly ILogger Logger; - protected readonly IWebSocketService WsService; private WebSocket? _webSocket; - protected GenericController(ILogger logger, IWebSocketService wsService) + protected GenericController(ILogger logger, GameService gameService) { Logger = logger; - WsService = wsService; + GameService = gameService; Logger.Log(LogLevel.Debug, "WebSocket Controller created"); } @@ -25,7 +25,7 @@ public abstract class GenericController : ControllerBase using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync(); Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id); _webSocket = webSocket; - WsService.Connections += WsServiceOnFire; + GameService.Connections += WsServiceOnFire; await Echo(); } else @@ -37,7 +37,7 @@ public abstract class GenericController : ControllerBase private async Task WsServiceOnFire(ArraySegment segment) { if (_webSocket == null) return; - await WsService.Send(_webSocket, segment); + await GameService.Send(_webSocket, segment); } @@ -50,16 +50,16 @@ public abstract class GenericController : ControllerBase do { var buffer = new byte[BufferSize]; - result = await WsService.Receive(_webSocket, buffer); + result = await GameService.Receive(_webSocket, buffer); if (result.CloseStatus.HasValue) break; var segment = Run(result, buffer); - WsService.SendToAll(segment); + GameService.SendToAll(segment); } while (true); - await WsService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription); + await GameService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription); } catch (WebSocketException e) { @@ -71,5 +71,5 @@ public abstract class GenericController : ControllerBase protected abstract ArraySegment Run(WebSocketReceiveResult result, byte[] data); - protected virtual void Disconnect() => WsService.Connections -= WsServiceOnFire; + protected virtual void Disconnect() => GameService.Connections -= WsServiceOnFire; } diff --git a/pac-man-board-game/Controllers/WsController.cs b/pac-man-board-game/Controllers/WsController.cs index de26825..9767421 100644 --- a/pac-man-board-game/Controllers/WsController.cs +++ b/pac-man-board-game/Controllers/WsController.cs @@ -1,6 +1,6 @@ using System.Net.WebSockets; using Microsoft.AspNetCore.Mvc; -using pacMan.Interfaces; +using pacMan.Services; namespace pacMan.Controllers; @@ -8,9 +8,7 @@ namespace pacMan.Controllers; [Route("api/[controller]")] public class WsController : GenericController { - public WsController(ILogger logger, IWebSocketService wsService) : base(logger, wsService) - { - } + public WsController(ILogger logger, GameService gameService) : base(logger, gameService) { } [HttpGet] public override async Task Accept() => await base.Accept(); @@ -20,4 +18,4 @@ public class WsController : GenericController var segment = new ArraySegment(data, 0, result.Count); return segment; } -} \ No newline at end of file +} diff --git a/pac-man-board-game/Interfaces/IWebSocketService.cs b/pac-man-board-game/Interfaces/IWebSocketService.cs index 4f789b4..52cf1fc 100644 --- a/pac-man-board-game/Interfaces/IWebSocketService.cs +++ b/pac-man-board-game/Interfaces/IWebSocketService.cs @@ -1,18 +1,10 @@ using System.Net.WebSockets; -using pacMan.Game; -using pacMan.Game.Items; -using pacMan.Services; namespace pacMan.Interfaces; public interface IWebSocketService { - SynchronizedCollection Games { get; } - int CountConnected { get; } - event Func, Task>? Connections; Task Send(WebSocket webSocket, ArraySegment segment); - void SendToAll(ArraySegment segment); Task Receive(WebSocket webSocket, byte[] buffer); Task Close(WebSocket webSocket, WebSocketCloseStatus closeStatus, string? closeStatusDescription); - GameGroup AddPlayer(IPlayer player, Queue spawns); } diff --git a/pac-man-board-game/Program.cs b/pac-man-board-game/Program.cs index 98f049f..826631a 100644 --- a/pac-man-board-game/Program.cs +++ b/pac-man-board-game/Program.cs @@ -6,7 +6,9 @@ var builder = WebApplication.CreateBuilder(args); // Add services to the container. builder.Services.AddControllersWithViews(); -builder.Services.AddSingleton() +builder.Services + .AddSingleton() + .AddSingleton() .AddTransient(); var app = builder.Build(); diff --git a/pac-man-board-game/Services/ActionService.cs b/pac-man-board-game/Services/ActionService.cs index fd210e4..bc66c3c 100644 --- a/pac-man-board-game/Services/ActionService.cs +++ b/pac-man-board-game/Services/ActionService.cs @@ -2,7 +2,6 @@ using System.Text.Json; using Microsoft.CSharp.RuntimeBinder; using pacMan.Game; using pacMan.Game.Items; -using pacMan.Interfaces; namespace pacMan.Services; @@ -21,14 +20,14 @@ public interface IActionService public class ActionService : IActionService { private readonly IDiceCup _diceCup; + private readonly GameService _gameService; private readonly ILogger _logger; - private readonly IWebSocketService _wsService; - public ActionService(ILogger logger, IWebSocketService wsService) + public ActionService(ILogger logger, GameService gameService) { _logger = logger; _diceCup = new DiceCup(); - _wsService = wsService; + _gameService = gameService; } public GameGroup? Group { get; set; } @@ -63,7 +62,7 @@ public class ActionService : IActionService PlayerInfoData data = JsonSerializer.Deserialize(message.Data); Player = data.Player; - Group = _wsService.AddPlayer(Player, data.Spawns); + Group = _gameService.AddPlayer(Player, data.Spawns); } catch (RuntimeBinderException e) { @@ -112,6 +111,6 @@ public struct PlayerInfoData public struct ReadyData { - public required bool AllReady { get; set; } + public required bool AllReady { get; init; } public required IEnumerable Players { get; set; } } diff --git a/pac-man-board-game/Services/GameService.cs b/pac-man-board-game/Services/GameService.cs new file mode 100644 index 0000000..6ecdd58 --- /dev/null +++ b/pac-man-board-game/Services/GameService.cs @@ -0,0 +1,35 @@ +using pacMan.Game; +using pacMan.Game.Items; + +namespace pacMan.Services; + +public class GameService : WebSocketService +{ + public GameService(ILogger logger) : base(logger) { } + + public SynchronizedCollection Games { get; } = new(); + + public event Func, Task>? Connections; // TODO remove and use GameGroup + + public void SendToAll(ArraySegment segment) + { + Connections?.Invoke(segment); + } + + public GameGroup AddPlayer(IPlayer player, Queue spawns) + { + var index = 0; + try + { + while (!Games[index].AddPlayer(player)) index++; + } + catch (ArgumentOutOfRangeException) + { + var game = new GameGroup(spawns); + game.AddPlayer(player); + Games.Add(game); + } + + return Games[index]; + } +} diff --git a/pac-man-board-game/Services/WebSocketService.cs b/pac-man-board-game/Services/WebSocketService.cs index e176719..68f6c66 100644 --- a/pac-man-board-game/Services/WebSocketService.cs +++ b/pac-man-board-game/Services/WebSocketService.cs @@ -1,6 +1,4 @@ using System.Net.WebSockets; -using pacMan.Game; -using pacMan.Game.Items; using pacMan.Interfaces; using pacMan.Utils; @@ -8,20 +6,14 @@ namespace pacMan.Services; public class WebSocketService : IWebSocketService { - private readonly ILogger _logger; + protected readonly ILogger Logger; public WebSocketService(ILogger logger) { - _logger = logger; + Logger = logger; logger.Log(LogLevel.Debug, "WebSocket Service created"); } - public SynchronizedCollection Games { get; } = new(); - - public int CountConnected => Connections?.GetInvocationList().Length ?? 0; - - public event Func, Task>? Connections; // TODO remove and use GameGroup - public async Task Send(WebSocket webSocket, ArraySegment segment) { await webSocket.SendAsync( @@ -30,18 +22,13 @@ public class WebSocketService : IWebSocketService true, CancellationToken.None); - _logger.Log(LogLevel.Debug, "Message sent to WebSocket"); - } - - public void SendToAll(ArraySegment segment) - { - Connections?.Invoke(segment); + Logger.Log(LogLevel.Debug, "Message sent to WebSocket"); } public async Task Receive(WebSocket webSocket, byte[] buffer) { var result = await webSocket.ReceiveAsync(new ArraySegment(buffer), CancellationToken.None); - _logger.Log(LogLevel.Debug, + Logger.Log(LogLevel.Debug, "Message \"{}\" received from WebSocket", buffer.GetString(result.Count)); return result; @@ -54,23 +41,6 @@ public class WebSocketService : IWebSocketService closeStatusDescription, CancellationToken.None); - _logger.Log(LogLevel.Information, "WebSocket connection closed"); - } - - public GameGroup AddPlayer(IPlayer player, Queue spawns) - { - var index = 0; - try - { - while (!Games[index].AddPlayer(player)) index++; - } - catch (ArgumentOutOfRangeException) - { - var game = new GameGroup(spawns); - game.AddPlayer(player); - Games.Add(game); - } - - return Games[index]; + Logger.Log(LogLevel.Information, "WebSocket connection closed"); } }