diff --git a/BackendTests/Services/GameServiceTests.cs b/BackendTests/Services/GameServiceTests.cs index 56438e5..f79ca0f 100644 --- a/BackendTests/Services/GameServiceTests.cs +++ b/BackendTests/Services/GameServiceTests.cs @@ -28,6 +28,32 @@ public class GameServiceTests }); } + #region CreateAndJoin(IPlayer player, Queue spawns) + + [Test] + public void CreateAndJoin_WhenEmpty() + { + var player = Players.Create("white"); + var group = _service.CreateAndJoin(player, _spawns); + + Assert.Multiple(() => + { + Assert.That(group.Players, Has.Count.EqualTo(1)); + Assert.That(group.NextPlayer, Is.EqualTo(player)); + Assert.That(_service.Games, Has.Count.EqualTo(1)); + }); + } + + [Test] + public void CreateAndJoin_SpawnsAreNotEqualToMaxPlayers() + { + var player = Players.Create("white"); + _spawns.Dequeue(); + Assert.Throws(() => _service.CreateAndJoin(player, _spawns)); + } + + #endregion + #region JoinbyId(Guid id) [Test] @@ -38,7 +64,7 @@ public class GameServiceTests Assert.Throws(() => _service.JoinById(Guid.NewGuid(), player)); } - + [Test] public void JoinById_WhenIdExists() { diff --git a/pac-man-board-game/Exceptions/GameNotPlayableException.cs b/pac-man-board-game/Exceptions/GameNotPlayableException.cs new file mode 100644 index 0000000..6f4a840 --- /dev/null +++ b/pac-man-board-game/Exceptions/GameNotPlayableException.cs @@ -0,0 +1,6 @@ +namespace pacMan.Exceptions; + +public class GameNotPlayableException : Exception +{ + public GameNotPlayableException(string message = "Game is not allowed to be played") : base(message) { } +} diff --git a/pac-man-board-game/Services/Game.cs b/pac-man-board-game/Services/Game.cs index 9dcd9cb..d04ab56 100644 --- a/pac-man-board-game/Services/Game.cs +++ b/pac-man-board-game/Services/Game.cs @@ -44,9 +44,15 @@ public class Game // TODO handle disconnects and reconnects public bool AddPlayer(IPlayer player) { if (Players.Count >= Rules.MaxPlayers || IsGameStarted) return false; + /* TODO remove above and uncomment below + if (Players.Count >= Rules.MaxPlayers) + throw new GameNotPlayableException("Game is full"); + if (IsGameStarted) + throw new GameNotPlayableException("Game has already started"); + */ player.State = State.WaitingForPlayers; - if (Players.Exists(p => p.Name == player.Name)) return true; + if (Players.Exists(p => p.Name == player.Name)) return true; // TODO change to false Players.Add(player); if (player.PacMan.SpawnPosition is null) SetSpawn(player); return true; diff --git a/pac-man-board-game/Services/GameService.cs b/pac-man-board-game/Services/GameService.cs index 84a4619..012f45c 100644 --- a/pac-man-board-game/Services/GameService.cs +++ b/pac-man-board-game/Services/GameService.cs @@ -4,13 +4,20 @@ using pacMan.GameStuff.Items; namespace pacMan.Services; +/// +/// The GameService class provides functionality for managing games in a WebSocket environment. It inherits from the WebSocketService class. +/// public class GameService : WebSocketService { public GameService(ILogger logger) : base(logger) { } + /// + /// A thread-safe collection (SynchronizedCollection) of "Game" objects. Utilized for managing multiple game instances simultaneously. + /// It represents all the current games being managed by GameService. + /// public SynchronizedCollection Games { get; } = new(); - public event Func, Task>? Connections; // TODO remove and use GameGroup + public event Func, Task>? Connections; // TODO remove and use Game public void SendToAll(ArraySegment segment) { @@ -34,12 +41,37 @@ public class GameService : WebSocketService return Games[index]; } + /// + /// This method tries to find a game with the specified id, add a player to it and return the updated game. + /// + /// The unique id of the game the player wants to join + /// The player instance that wants to join the game + /// Returns the updated Game object after adding the player. + /// Thrown if a game with the specified id cannot be found. public Game JoinById(Guid id, IPlayer player) { var game = Games.FirstOrDefault(g => g.Id == id) ?? throw new GameNotFoundException(); - var added = game.AddPlayer(player); + game.AddPlayer(player); - if (!added) throw new ArgumentException("Game is full"); + return game; + } + + + /// + /// Creates a new game and adds a player to it. + /// + /// The player instance that is going to join the new game. + /// A collection of spawn points arranged in a directional queue. + /// Returns the newly created Game object, with the player added to it. + /// Thrown if the number of spawns is not equal to the maximum number of players set by the Rules. + public Game CreateAndJoin(IPlayer player, Queue spawns) + { + if (spawns.Count != Rules.MaxPlayers) + throw new ArgumentException($"The number of spawns must be equal to {Rules.MaxPlayers}."); + + var game = new Game(spawns); + game.AddPlayer(player); + Games.Add(game); return game; } diff --git a/pac-man-board-game/Services/WebSocketService.cs b/pac-man-board-game/Services/WebSocketService.cs index 68f6c66..79fbad2 100644 --- a/pac-man-board-game/Services/WebSocketService.cs +++ b/pac-man-board-game/Services/WebSocketService.cs @@ -4,6 +4,7 @@ using pacMan.Utils; namespace pacMan.Services; + public class WebSocketService : IWebSocketService { protected readonly ILogger Logger;