Split up methods in controllers, added tests for GameService, refactored GameGroup name to Game and directory Game to GameStuff
This commit is contained in:
parent
fab3e3d13f
commit
c7bc473572
@ -5,7 +5,7 @@ using System.Text.Json;
|
|||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using NSubstitute;
|
using NSubstitute;
|
||||||
using pacMan.Controllers;
|
using pacMan.Controllers;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Services;
|
using pacMan.Services;
|
||||||
using pacMan.Utils;
|
using pacMan.Utils;
|
||||||
|
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace BackendTests.Game.Items;
|
namespace BackendTests.Game.Items;
|
||||||
|
|
||||||
@ -11,7 +11,7 @@ public class DiceCupTests
|
|||||||
var roll = diceCup.Roll;
|
var roll = diceCup.Roll;
|
||||||
Assert.That(roll, Has.Count.EqualTo(2));
|
Assert.That(roll, Has.Count.EqualTo(2));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void Roll_ReturnsNumbersInRange1To6()
|
public void Roll_ReturnsNumbersInRange1To6()
|
||||||
{
|
{
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace BackendTests.Game.Items;
|
namespace BackendTests.Game.Items;
|
||||||
|
|
||||||
|
@ -2,8 +2,8 @@ using System.Text.Json;
|
|||||||
using BackendTests.TestUtils;
|
using BackendTests.TestUtils;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using NSubstitute;
|
using NSubstitute;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
using pacMan.Services;
|
using pacMan.Services;
|
||||||
|
|
||||||
namespace BackendTests.Services;
|
namespace BackendTests.Services;
|
||||||
@ -171,7 +171,8 @@ public class ActionServiceTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void Ready_TwoReady()
|
public void Ready_TwoReady()
|
||||||
{
|
{
|
||||||
var group = new GameGroup(new Queue<DirectionalPosition>()) { Players = { _blackPlayer, _whitePlayer } };
|
var group = new pacMan.Services.Game(new Queue<DirectionalPosition>())
|
||||||
|
{ Players = { _blackPlayer, _whitePlayer } };
|
||||||
_service.Group = group;
|
_service.Group = group;
|
||||||
_service.Player = _blackPlayer;
|
_service.Player = _blackPlayer;
|
||||||
|
|
||||||
@ -194,7 +195,7 @@ public class ActionServiceTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void FindNextPlayer_NoPlayers()
|
public void FindNextPlayer_NoPlayers()
|
||||||
{
|
{
|
||||||
_service.Group = new GameGroup(new Queue<DirectionalPosition>());
|
_service.Group = new pacMan.Services.Game(new Queue<DirectionalPosition>());
|
||||||
Assert.Throws<InvalidOperationException>(() => _service.FindNextPlayer());
|
Assert.Throws<InvalidOperationException>(() => _service.FindNextPlayer());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -202,7 +203,7 @@ public class ActionServiceTests
|
|||||||
public void FindNextPlayer_OnePlayer()
|
public void FindNextPlayer_OnePlayer()
|
||||||
{
|
{
|
||||||
_service.Group =
|
_service.Group =
|
||||||
new GameGroup(new Queue<DirectionalPosition>(
|
new pacMan.Services.Game(new Queue<DirectionalPosition>(
|
||||||
new[] { new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up } }))
|
new[] { new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up } }))
|
||||||
{ Players = { _whitePlayer } };
|
{ Players = { _whitePlayer } };
|
||||||
|
|
||||||
@ -213,7 +214,7 @@ public class ActionServiceTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void FindNextPlayer_TwoPlayers()
|
public void FindNextPlayer_TwoPlayers()
|
||||||
{
|
{
|
||||||
_service.Group = new GameGroup(new Queue<DirectionalPosition>(
|
_service.Group = new pacMan.Services.Game(new Queue<DirectionalPosition>(
|
||||||
new[]
|
new[]
|
||||||
{
|
{
|
||||||
new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up },
|
new DirectionalPosition { At = new Position { X = 3, Y = 3 }, Direction = Direction.Up },
|
||||||
|
100
BackendTests/Services/GameServiceTests.cs
Normal file
100
BackendTests/Services/GameServiceTests.cs
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
using BackendTests.TestUtils;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using NSubstitute;
|
||||||
|
using pacMan.Exceptions;
|
||||||
|
using pacMan.GameStuff;
|
||||||
|
using pacMan.Services;
|
||||||
|
|
||||||
|
namespace BackendTests.Services;
|
||||||
|
|
||||||
|
public class GameServiceTests
|
||||||
|
{
|
||||||
|
private readonly DirectionalPosition _spawn3By3Up = new()
|
||||||
|
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
|
||||||
|
|
||||||
|
private GameService _service = null!;
|
||||||
|
private Queue<DirectionalPosition> _spawns = null!;
|
||||||
|
|
||||||
|
[SetUp]
|
||||||
|
public void SetUp()
|
||||||
|
{
|
||||||
|
_service = new GameService(Substitute.For<ILogger<GameService>>());
|
||||||
|
_spawns = new Queue<DirectionalPosition>(new[]
|
||||||
|
{
|
||||||
|
_spawn3By3Up,
|
||||||
|
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
|
||||||
|
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
|
||||||
|
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#region JoinbyId(Guid id)
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void JoinById_WhenIdNotExists()
|
||||||
|
{
|
||||||
|
var player = Players.Create("white");
|
||||||
|
_service.AddPlayer(player, _spawns);
|
||||||
|
|
||||||
|
Assert.Throws<GameNotFoundException>(() => _service.JoinById(Guid.NewGuid(), player));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void JoinById_WhenIdExists()
|
||||||
|
{
|
||||||
|
var player = Players.Create("white");
|
||||||
|
var group = _service.AddPlayer(player, _spawns);
|
||||||
|
|
||||||
|
var player2 = Players.Create("black");
|
||||||
|
var result = _service.JoinById(group.Id, player2);
|
||||||
|
|
||||||
|
Assert.Multiple(() =>
|
||||||
|
{
|
||||||
|
Assert.That(result, Is.EqualTo(group));
|
||||||
|
Assert.That(group.Players, Has.Count.EqualTo(2));
|
||||||
|
Assert.That(_service.Games, Has.Count.EqualTo(1));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region AddPlayer(IPlayer)
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void AddPlayer_ToEmptyGroup()
|
||||||
|
{
|
||||||
|
var player = Players.Create("white");
|
||||||
|
var group = _service.AddPlayer(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 AddPlayer_ToFullGroup()
|
||||||
|
{
|
||||||
|
for (var i = 0; i < 4; i++)
|
||||||
|
{
|
||||||
|
var player = Players.Create(i.ToString());
|
||||||
|
_service.AddPlayer(player, _spawns);
|
||||||
|
}
|
||||||
|
|
||||||
|
var player5 = Players.Create("white");
|
||||||
|
|
||||||
|
var group = _service.AddPlayer(player5, new Queue<DirectionalPosition>(new[] { _spawn3By3Up }));
|
||||||
|
|
||||||
|
Assert.Multiple(() =>
|
||||||
|
{
|
||||||
|
Assert.That(group.Players, Has.Count.EqualTo(1));
|
||||||
|
Assert.That(group.NextPlayer, Is.EqualTo(player5));
|
||||||
|
Assert.That(_service.Games, Has.Count.EqualTo(2));
|
||||||
|
Assert.That(_service.Games.First(), Has.Count.EqualTo(Rules.MaxPlayers));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#endregion
|
||||||
|
}
|
@ -1,13 +1,12 @@
|
|||||||
using BackendTests.TestUtils;
|
using BackendTests.TestUtils;
|
||||||
using pacMan.Exceptions;
|
using pacMan.Exceptions;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
using pacMan.Services;
|
|
||||||
using pacMan.Utils;
|
using pacMan.Utils;
|
||||||
|
|
||||||
namespace BackendTests.Services;
|
namespace BackendTests.Services;
|
||||||
|
|
||||||
public class GameGroupTests
|
public class GameTests
|
||||||
{
|
{
|
||||||
private readonly DirectionalPosition _spawn3By3Up = new()
|
private readonly DirectionalPosition _spawn3By3Up = new()
|
||||||
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
|
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
|
||||||
@ -22,7 +21,7 @@ public class GameGroupTests
|
|||||||
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Right };
|
{ At = new Position { X = 7, Y = 7 }, Direction = Direction.Right };
|
||||||
|
|
||||||
private IPlayer _bluePlayer = null!;
|
private IPlayer _bluePlayer = null!;
|
||||||
private GameGroup _gameGroup = null!;
|
private pacMan.Services.Game _game = null!;
|
||||||
private IPlayer _greenPlayer = null!;
|
private IPlayer _greenPlayer = null!;
|
||||||
private IPlayer _purplePlayer = null!;
|
private IPlayer _purplePlayer = null!;
|
||||||
private IPlayer _redPlayer = null!;
|
private IPlayer _redPlayer = null!;
|
||||||
@ -36,7 +35,7 @@ public class GameGroupTests
|
|||||||
_spawns = new Queue<DirectionalPosition>(
|
_spawns = new Queue<DirectionalPosition>(
|
||||||
new[] { _spawn3By3Up, _spawn7By7Left, _spawn7By7Down, _spawn7By7Right });
|
new[] { _spawn3By3Up, _spawn7By7Left, _spawn7By7Down, _spawn7By7Right });
|
||||||
|
|
||||||
_gameGroup = new GameGroup(_spawns);
|
_game = new pacMan.Services.Game(_spawns);
|
||||||
_redPlayer = Players.Create("red");
|
_redPlayer = Players.Create("red");
|
||||||
_bluePlayer = Players.Create("blue");
|
_bluePlayer = Players.Create("blue");
|
||||||
_yellowPlayer = Players.Create("yellow");
|
_yellowPlayer = Players.Create("yellow");
|
||||||
@ -46,10 +45,10 @@ public class GameGroupTests
|
|||||||
|
|
||||||
private void AddFullParty()
|
private void AddFullParty()
|
||||||
{
|
{
|
||||||
_gameGroup.AddPlayer(_bluePlayer);
|
_game.AddPlayer(_bluePlayer);
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
_gameGroup.AddPlayer(_yellowPlayer);
|
_game.AddPlayer(_yellowPlayer);
|
||||||
_gameGroup.AddPlayer(_greenPlayer);
|
_game.AddPlayer(_greenPlayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
#region NextPlayer()
|
#region NextPlayer()
|
||||||
@ -57,7 +56,7 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void NextPlayer_WhenEmpty()
|
public void NextPlayer_WhenEmpty()
|
||||||
{
|
{
|
||||||
Assert.Throws<InvalidOperationException>(() => _gameGroup.NextPlayer());
|
Assert.Throws<InvalidOperationException>(() => _game.NextPlayer());
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -67,7 +66,7 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void AddPlayer_WhenEmpty()
|
public void AddPlayer_WhenEmpty()
|
||||||
{
|
{
|
||||||
var added = _gameGroup.AddPlayer(_redPlayer);
|
var added = _game.AddPlayer(_redPlayer);
|
||||||
Assert.That(added, Is.True);
|
Assert.That(added, Is.True);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -76,16 +75,16 @@ public class GameGroupTests
|
|||||||
Assert.Multiple(() =>
|
Assert.Multiple(() =>
|
||||||
{
|
{
|
||||||
AddFullParty();
|
AddFullParty();
|
||||||
Assert.That(_gameGroup.Players.Count, Is.EqualTo(Rules.MaxPlayers));
|
Assert.That(_game.Players.Count, Is.EqualTo(Rules.MaxPlayers));
|
||||||
Assert.That(_gameGroup.AddPlayer(_purplePlayer), Is.False);
|
Assert.That(_game.AddPlayer(_purplePlayer), Is.False);
|
||||||
});
|
});
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void AddPlayer_WhenNameExists()
|
public void AddPlayer_WhenNameExists()
|
||||||
{
|
{
|
||||||
var redClone = Players.Clone(_redPlayer);
|
var redClone = Players.Clone(_redPlayer);
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
var added = _gameGroup.AddPlayer(redClone);
|
var added = _game.AddPlayer(redClone);
|
||||||
Assert.That(added, Is.True);
|
Assert.That(added, Is.True);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -93,14 +92,14 @@ public class GameGroupTests
|
|||||||
public void AddPlayer_WhenStateIsNotWaitingForPlayers()
|
public void AddPlayer_WhenStateIsNotWaitingForPlayers()
|
||||||
{
|
{
|
||||||
_redPlayer.State = State.InGame;
|
_redPlayer.State = State.InGame;
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
Assert.That(_redPlayer.State, Is.EqualTo(State.WaitingForPlayers));
|
Assert.That(_redPlayer.State, Is.EqualTo(State.WaitingForPlayers));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void AddPlayer_AddSpawnPosition()
|
public void AddPlayer_AddSpawnPosition()
|
||||||
{
|
{
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.Not.Null);
|
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.Not.Null);
|
||||||
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.EqualTo(_spawn3By3Up));
|
Assert.That(_redPlayer.PacMan.SpawnPosition, Is.EqualTo(_spawn3By3Up));
|
||||||
}
|
}
|
||||||
@ -108,16 +107,16 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void AddPlayer_WhenGameHasStarted()
|
public void AddPlayer_WhenGameHasStarted()
|
||||||
{
|
{
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
_gameGroup.AddPlayer(_bluePlayer);
|
_game.AddPlayer(_bluePlayer);
|
||||||
|
|
||||||
_gameGroup.SetReady(_redPlayer);
|
_game.SetReady(_redPlayer);
|
||||||
_gameGroup.SetReady(_bluePlayer);
|
_game.SetReady(_bluePlayer);
|
||||||
_gameGroup.SetAllInGame();
|
_game.SetAllInGame();
|
||||||
|
|
||||||
Assert.That(_gameGroup.AddPlayer(_greenPlayer), Is.False);
|
Assert.That(_game.AddPlayer(_greenPlayer), Is.False);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Sendtoall(ArraySegment<byte> segment)
|
#region Sendtoall(ArraySegment<byte> segment)
|
||||||
@ -125,7 +124,7 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void SendToAll_WhenConnectionsIsNull()
|
public void SendToAll_WhenConnectionsIsNull()
|
||||||
{
|
{
|
||||||
Assert.DoesNotThrow(() => _gameGroup.SendToAll(new { }.ToArraySegment()));
|
Assert.DoesNotThrow(() => _game.SendToAll(new { }.ToArraySegment()));
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
@ -134,10 +133,10 @@ public class GameGroupTests
|
|||||||
var counter = 0;
|
var counter = 0;
|
||||||
async Task Send(ArraySegment<byte> segment) => await Task.Run(() => counter++);
|
async Task Send(ArraySegment<byte> segment) => await Task.Run(() => counter++);
|
||||||
|
|
||||||
_gameGroup.Connections += Send;
|
_game.Connections += Send;
|
||||||
_gameGroup.Connections += Send;
|
_game.Connections += Send;
|
||||||
|
|
||||||
_gameGroup.SendToAll(new { }.ToArraySegment());
|
_game.SendToAll(new { }.ToArraySegment());
|
||||||
|
|
||||||
// TODO timeout after n amount of time
|
// TODO timeout after n amount of time
|
||||||
while (counter < 2) { }
|
while (counter < 2) { }
|
||||||
@ -152,10 +151,10 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void SetReady_ReturnsAllPlayers()
|
public void SetReady_ReturnsAllPlayers()
|
||||||
{
|
{
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
_gameGroup.AddPlayer(_bluePlayer);
|
_game.AddPlayer(_bluePlayer);
|
||||||
|
|
||||||
var players = _gameGroup.SetReady(_redPlayer).ToList();
|
var players = _game.SetReady(_redPlayer).ToList();
|
||||||
|
|
||||||
Assert.Multiple(() =>
|
Assert.Multiple(() =>
|
||||||
{
|
{
|
||||||
@ -168,11 +167,11 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void SetReady_SetsStateToReady()
|
public void SetReady_SetsStateToReady()
|
||||||
{
|
{
|
||||||
_gameGroup.AddPlayer(_redPlayer);
|
_game.AddPlayer(_redPlayer);
|
||||||
|
|
||||||
Assert.That(_redPlayer.State, Is.Not.EqualTo(State.Ready));
|
Assert.That(_redPlayer.State, Is.Not.EqualTo(State.Ready));
|
||||||
|
|
||||||
_gameGroup.SetReady(_redPlayer);
|
_game.SetReady(_redPlayer);
|
||||||
|
|
||||||
Assert.That(_redPlayer.State, Is.EqualTo(State.Ready));
|
Assert.That(_redPlayer.State, Is.EqualTo(State.Ready));
|
||||||
}
|
}
|
||||||
@ -180,7 +179,7 @@ public class GameGroupTests
|
|||||||
[Test]
|
[Test]
|
||||||
public void SetReady_WhenPlayerIsNotInPlayers()
|
public void SetReady_WhenPlayerIsNotInPlayers()
|
||||||
{
|
{
|
||||||
Assert.Throws<PlayerNotFoundException>(() => _gameGroup.SetReady(_redPlayer));
|
Assert.Throws<PlayerNotFoundException>(() => _game.SetReady(_redPlayer));
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -191,15 +190,15 @@ public class GameGroupTests
|
|||||||
public void SetAllInGame_SetsStateToInGame()
|
public void SetAllInGame_SetsStateToInGame()
|
||||||
{
|
{
|
||||||
AddFullParty();
|
AddFullParty();
|
||||||
_gameGroup.Players.ForEach(player => player.State = State.Ready);
|
_game.Players.ForEach(player => player.State = State.Ready);
|
||||||
Assert.That(_gameGroup.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.Ready));
|
Assert.That(_game.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.Ready));
|
||||||
|
|
||||||
var allInGame = _gameGroup.SetAllInGame();
|
var allInGame = _game.SetAllInGame();
|
||||||
|
|
||||||
Assert.Multiple(() =>
|
Assert.Multiple(() =>
|
||||||
{
|
{
|
||||||
Assert.That(allInGame, Is.True);
|
Assert.That(allInGame, Is.True);
|
||||||
Assert.That(_gameGroup.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.InGame));
|
Assert.That(_game.Players, Has.All.Property(nameof(IPlayer.State)).EqualTo(State.InGame));
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -207,15 +206,15 @@ public class GameGroupTests
|
|||||||
public void SetAllInGame_SetStateToInGame_WhenNotAllReady()
|
public void SetAllInGame_SetStateToInGame_WhenNotAllReady()
|
||||||
{
|
{
|
||||||
AddFullParty();
|
AddFullParty();
|
||||||
var allInGame = _gameGroup.SetAllInGame();
|
var allInGame = _game.SetAllInGame();
|
||||||
Assert.That(allInGame, Is.False);
|
Assert.That(allInGame, Is.False);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void SetAllInGame_WhenPlayersIsEmpty()
|
public void SetAllInGame_WhenPlayersIsEmpty()
|
||||||
{
|
{
|
||||||
_gameGroup.SetAllInGame();
|
_game.SetAllInGame();
|
||||||
Assert.That(_gameGroup.Players, Is.Empty);
|
Assert.That(_game.Players, Is.Empty);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
@ -226,15 +225,15 @@ public class GameGroupTests
|
|||||||
public void IsGameStarted_AllWaiting()
|
public void IsGameStarted_AllWaiting()
|
||||||
{
|
{
|
||||||
AddFullParty();
|
AddFullParty();
|
||||||
Assert.That(_gameGroup.IsGameStarted, Is.False);
|
Assert.That(_game.IsGameStarted, Is.False);
|
||||||
}
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void IsGameStarted_AllInGame()
|
public void IsGameStarted_AllInGame()
|
||||||
{
|
{
|
||||||
AddFullParty();
|
AddFullParty();
|
||||||
_gameGroup.Players.ForEach(player => player.State = State.InGame);
|
_game.Players.ForEach(player => player.State = State.InGame);
|
||||||
Assert.That(_gameGroup.IsGameStarted, Is.True);
|
Assert.That(_game.IsGameStarted, Is.True);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
@ -1,8 +1,7 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
using BackendTests.TestUtils;
|
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using NSubstitute;
|
using NSubstitute;
|
||||||
using pacMan.Game;
|
using pacMan.Interfaces;
|
||||||
using pacMan.Services;
|
using pacMan.Services;
|
||||||
using pacMan.Utils;
|
using pacMan.Utils;
|
||||||
|
|
||||||
@ -10,25 +9,12 @@ namespace BackendTests.Services;
|
|||||||
|
|
||||||
public class WebSocketServiceTests
|
public class WebSocketServiceTests
|
||||||
{
|
{
|
||||||
private readonly DirectionalPosition _spawn3By3Up = new()
|
private IWebSocketService _service = null!;
|
||||||
{ At = new Position { X = 3, Y = 3 }, Direction = Direction.Up };
|
|
||||||
|
|
||||||
private GameService _service = null!;
|
|
||||||
|
|
||||||
private Queue<DirectionalPosition> _spawns = null!;
|
|
||||||
|
|
||||||
|
|
||||||
[SetUp]
|
[SetUp]
|
||||||
public void SetUp()
|
public void SetUp()
|
||||||
{
|
{
|
||||||
_service = new GameService(Substitute.For<ILogger<GameService>>());
|
_service = new WebSocketService(Substitute.For<ILogger<WebSocketService>>());
|
||||||
_spawns = new Queue<DirectionalPosition>(new[]
|
|
||||||
{
|
|
||||||
_spawn3By3Up,
|
|
||||||
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
|
|
||||||
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down },
|
|
||||||
new DirectionalPosition { At = new Position { X = 7, Y = 7 }, Direction = Direction.Down }
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#region Send(Websocket, ArraySegment<byte>)
|
#region Send(Websocket, ArraySegment<byte>)
|
||||||
@ -140,44 +126,4 @@ public class WebSocketServiceTests
|
|||||||
}
|
}
|
||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region AddPlayer(IPlayer)
|
|
||||||
|
|
||||||
[Test]
|
|
||||||
public void AddPlayer_ToEmptyGroup()
|
|
||||||
{
|
|
||||||
var player = Players.Create("white");
|
|
||||||
var group = _service.AddPlayer(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 AddPlayer_ToFullGroup()
|
|
||||||
{
|
|
||||||
for (var i = 0; i < 4; i++)
|
|
||||||
{
|
|
||||||
var player = Players.Create(i.ToString());
|
|
||||||
_service.AddPlayer(player, _spawns);
|
|
||||||
}
|
|
||||||
|
|
||||||
var player5 = Players.Create("white");
|
|
||||||
|
|
||||||
var group = _service.AddPlayer(player5, new Queue<DirectionalPosition>(new[] { _spawn3By3Up }));
|
|
||||||
|
|
||||||
Assert.Multiple(() =>
|
|
||||||
{
|
|
||||||
Assert.That(group.Players, Has.Count.EqualTo(1));
|
|
||||||
Assert.That(group.NextPlayer, Is.EqualTo(player5));
|
|
||||||
Assert.That(_service.Games, Has.Count.EqualTo(2));
|
|
||||||
Assert.That(_service.Games.First(), Has.Count.EqualTo(Rules.MaxPlayers));
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace BackendTests.TestUtils;
|
namespace BackendTests.TestUtils;
|
||||||
|
|
||||||
|
@ -4,7 +4,7 @@ import {Button} from "../components/Button";
|
|||||||
|
|
||||||
const fetchAtom = atom(async () => {
|
const fetchAtom = atom(async () => {
|
||||||
const response = await fetch(import.meta.env.VITE_API_HTTP + "/allGames");
|
const response = await fetch(import.meta.env.VITE_API_HTTP + "/allGames");
|
||||||
return await response.json() as GameGroup[];
|
return await response.json() as Game[];
|
||||||
});
|
});
|
||||||
|
|
||||||
const LobbyPage: Component = () => (
|
const LobbyPage: Component = () => (
|
||||||
|
@ -35,7 +35,7 @@ type Path = {
|
|||||||
Direction: import("../game/direction").Direction
|
Direction: import("../game/direction").Direction
|
||||||
}
|
}
|
||||||
|
|
||||||
type GameGroup = {
|
type Game = {
|
||||||
readonly id: string,
|
readonly id: string,
|
||||||
readonly count: number,
|
readonly count: number,
|
||||||
readonly isGameStarted: boolean,
|
readonly isGameStarted: boolean,
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Services;
|
using pacMan.Services;
|
||||||
using pacMan.Utils;
|
using pacMan.Utils;
|
||||||
|
|
||||||
@ -11,19 +11,23 @@ namespace pacMan.Controllers;
|
|||||||
public class GameController : GenericController // TODO reconnect using player id
|
public class GameController : GenericController // TODO reconnect using player id
|
||||||
{
|
{
|
||||||
private readonly IActionService _actionService;
|
private readonly IActionService _actionService;
|
||||||
|
private readonly GameService _gameService;
|
||||||
|
|
||||||
public GameController(ILogger<GameController> logger, GameService gameService, IActionService actionService) :
|
public GameController(ILogger<GameController> logger, GameService gameService, IActionService actionService) :
|
||||||
base(logger, gameService) =>
|
base(logger, gameService)
|
||||||
|
{
|
||||||
|
_gameService = gameService;
|
||||||
_actionService = actionService;
|
_actionService = actionService;
|
||||||
|
}
|
||||||
|
|
||||||
[HttpGet("connect")]
|
[HttpGet("connect")]
|
||||||
public override async Task Accept() => await base.Accept();
|
public override async Task Accept() => await base.Accept();
|
||||||
|
|
||||||
[HttpGet("allGames")]
|
[HttpGet("allGames")]
|
||||||
public IEnumerable<GameGroup> GetAllGames()
|
public IEnumerable<Game> GetAllGames()
|
||||||
{
|
{
|
||||||
Logger.Log(LogLevel.Information, "Returning all games");
|
Logger.Log(LogLevel.Information, "Returning all games");
|
||||||
return GameService.Games;
|
return _gameService.Games;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -38,9 +42,23 @@ public class GameController : GenericController // TODO reconnect using player i
|
|||||||
return action.ToArraySegment();
|
return action.ToArraySegment();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected override void Send(ArraySegment<byte> segment) => _gameService.SendToAll(segment);
|
||||||
|
|
||||||
|
protected override Task Echo()
|
||||||
|
{
|
||||||
|
_gameService.Connections += WsServiceOnFire;
|
||||||
|
return base.Echo();
|
||||||
|
}
|
||||||
|
|
||||||
protected override void Disconnect()
|
protected override void Disconnect()
|
||||||
{
|
{
|
||||||
base.Disconnect();
|
_gameService.Connections -= WsServiceOnFire;
|
||||||
_actionService.Disconnect();
|
_actionService.Disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task WsServiceOnFire(ArraySegment<byte> segment)
|
||||||
|
{
|
||||||
|
if (WebSocket == null) return;
|
||||||
|
await GameService.Send(WebSocket, segment);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,17 +1,17 @@
|
|||||||
using System.Net.WebSockets;
|
using System.Net.WebSockets;
|
||||||
using Microsoft.AspNetCore.Mvc;
|
using Microsoft.AspNetCore.Mvc;
|
||||||
using pacMan.Services;
|
using pacMan.Interfaces;
|
||||||
|
|
||||||
namespace pacMan.Controllers;
|
namespace pacMan.Controllers;
|
||||||
|
|
||||||
public abstract class GenericController : ControllerBase // TODO only use WebSocketService in this class
|
public abstract class GenericController : ControllerBase
|
||||||
{
|
{
|
||||||
private const int BufferSize = 1024 * 4;
|
private const int BufferSize = 1024 * 4;
|
||||||
protected readonly GameService GameService;
|
protected readonly IWebSocketService GameService;
|
||||||
protected readonly ILogger<GenericController> Logger;
|
protected readonly ILogger<GenericController> Logger;
|
||||||
private WebSocket? _webSocket;
|
protected WebSocket? WebSocket;
|
||||||
|
|
||||||
protected GenericController(ILogger<GenericController> logger, GameService gameService)
|
protected GenericController(ILogger<GenericController> logger, IWebSocketService gameService)
|
||||||
{
|
{
|
||||||
Logger = logger;
|
Logger = logger;
|
||||||
GameService = gameService;
|
GameService = gameService;
|
||||||
@ -24,8 +24,7 @@ public abstract class GenericController : ControllerBase // TODO only use WebSoc
|
|||||||
{
|
{
|
||||||
using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
using var webSocket = await HttpContext.WebSockets.AcceptWebSocketAsync();
|
||||||
Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id);
|
Logger.Log(LogLevel.Information, "WebSocket connection established to {}", HttpContext.Connection.Id);
|
||||||
_webSocket = webSocket;
|
WebSocket = webSocket;
|
||||||
GameService.Connections += WsServiceOnFire;
|
|
||||||
await Echo();
|
await Echo();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -34,32 +33,25 @@ public abstract class GenericController : ControllerBase // TODO only use WebSoc
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task WsServiceOnFire(ArraySegment<byte> segment)
|
|
||||||
{
|
|
||||||
if (_webSocket == null) return;
|
|
||||||
await GameService.Send(_webSocket, segment);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
protected virtual async Task Echo()
|
protected virtual async Task Echo()
|
||||||
{
|
{
|
||||||
if (_webSocket == null) return;
|
if (WebSocket == null) return;
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
WebSocketReceiveResult? result;
|
WebSocketReceiveResult? result;
|
||||||
do
|
do
|
||||||
{
|
{
|
||||||
var buffer = new byte[BufferSize];
|
var buffer = new byte[BufferSize];
|
||||||
result = await GameService.Receive(_webSocket, buffer);
|
result = await GameService.Receive(WebSocket, buffer);
|
||||||
|
|
||||||
if (result.CloseStatus.HasValue) break;
|
if (result.CloseStatus.HasValue) break;
|
||||||
|
|
||||||
var segment = Run(result, buffer);
|
var segment = Run(result, buffer);
|
||||||
|
|
||||||
GameService.SendToAll(segment);
|
Send(segment);
|
||||||
} while (true);
|
} while (true);
|
||||||
|
|
||||||
await GameService.Close(_webSocket, result.CloseStatus.Value, result.CloseStatusDescription);
|
await GameService.Close(WebSocket, result.CloseStatus.Value, result.CloseStatusDescription);
|
||||||
}
|
}
|
||||||
catch (WebSocketException e)
|
catch (WebSocketException e)
|
||||||
{
|
{
|
||||||
@ -69,7 +61,13 @@ public abstract class GenericController : ControllerBase // TODO only use WebSoc
|
|||||||
Disconnect();
|
Disconnect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected virtual async void Send(ArraySegment<byte> segment)
|
||||||
|
{
|
||||||
|
if (WebSocket == null) return;
|
||||||
|
await GameService.Send(WebSocket, segment);
|
||||||
|
}
|
||||||
|
|
||||||
protected abstract ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data);
|
protected abstract ArraySegment<byte> Run(WebSocketReceiveResult result, byte[] data);
|
||||||
|
|
||||||
protected virtual void Disconnect() => GameService.Connections -= WsServiceOnFire;
|
protected virtual void Disconnect() { }
|
||||||
}
|
}
|
||||||
|
6
pac-man-board-game/Exceptions/GameNotFoundException.cs
Normal file
6
pac-man-board-game/Exceptions/GameNotFoundException.cs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
namespace pacMan.Exceptions;
|
||||||
|
|
||||||
|
public class GameNotFoundException : Exception
|
||||||
|
{
|
||||||
|
public GameNotFoundException(string message = "Game not found") : base(message) { }
|
||||||
|
}
|
@ -1,6 +1,6 @@
|
|||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
|
|
||||||
namespace pacMan.Game;
|
namespace pacMan.GameStuff;
|
||||||
|
|
||||||
public enum GameAction
|
public enum GameAction
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game;
|
namespace pacMan.GameStuff;
|
||||||
|
|
||||||
public class Character : IEquatable<Character>
|
public class Character : IEquatable<Character>
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game.Items;
|
namespace pacMan.GameStuff.Items;
|
||||||
|
|
||||||
public class Box : IEquatable<Box>
|
public class Box : IEquatable<Box>
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game.Items;
|
namespace pacMan.GameStuff.Items;
|
||||||
|
|
||||||
public interface IDice
|
public interface IDice
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game.Items;
|
namespace pacMan.GameStuff.Items;
|
||||||
|
|
||||||
public interface IDiceCup
|
public interface IDiceCup
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game.Items;
|
namespace pacMan.GameStuff.Items;
|
||||||
|
|
||||||
public interface IPellet
|
public interface IPellet
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game.Items;
|
namespace pacMan.GameStuff.Items;
|
||||||
|
|
||||||
public interface IPlayer
|
public interface IPlayer
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game;
|
namespace pacMan.GameStuff;
|
||||||
|
|
||||||
public class MovePath : IEquatable<MovePath>
|
public class MovePath : IEquatable<MovePath>
|
||||||
{
|
{
|
@ -1,4 +1,4 @@
|
|||||||
namespace pacMan.Game;
|
namespace pacMan.GameStuff;
|
||||||
|
|
||||||
public class Rules
|
public class Rules
|
||||||
{
|
{
|
||||||
@ -6,4 +6,4 @@ public class Rules
|
|||||||
public const int MaxPlayers = 4;
|
public const int MaxPlayers = 4;
|
||||||
public const int NumGhosts = 2;
|
public const int NumGhosts = 2;
|
||||||
public const int BoardSize = 10;
|
public const int BoardSize = 10;
|
||||||
}
|
}
|
@ -1,14 +1,14 @@
|
|||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using Microsoft.CSharp.RuntimeBinder;
|
using Microsoft.CSharp.RuntimeBinder;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace pacMan.Services;
|
namespace pacMan.Services;
|
||||||
|
|
||||||
public interface IActionService
|
public interface IActionService
|
||||||
{
|
{
|
||||||
IPlayer Player { set; }
|
IPlayer Player { set; }
|
||||||
GameGroup Group { set; }
|
Game Group { set; }
|
||||||
void DoAction(ActionMessage message);
|
void DoAction(ActionMessage message);
|
||||||
List<int> RollDice();
|
List<int> RollDice();
|
||||||
List<IPlayer> SetPlayerInfo(ActionMessage message);
|
List<IPlayer> SetPlayerInfo(ActionMessage message);
|
||||||
@ -30,7 +30,7 @@ public class ActionService : IActionService
|
|||||||
_gameService = gameService;
|
_gameService = gameService;
|
||||||
}
|
}
|
||||||
|
|
||||||
public GameGroup? Group { get; set; }
|
public Game? Group { get; set; }
|
||||||
|
|
||||||
public IPlayer? Player { get; set; }
|
public IPlayer? Player { get; set; }
|
||||||
|
|
||||||
|
@ -1,16 +1,16 @@
|
|||||||
using System.Text.Json.Serialization;
|
using System.Text.Json.Serialization;
|
||||||
using pacMan.Exceptions;
|
using pacMan.Exceptions;
|
||||||
using pacMan.Game;
|
using pacMan.GameStuff;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace pacMan.Services;
|
namespace pacMan.Services;
|
||||||
|
|
||||||
public class GameGroup // TODO handle disconnects and reconnects
|
public class Game // TODO handle disconnects and reconnects
|
||||||
{
|
{
|
||||||
private readonly Random _random = new();
|
private readonly Random _random = new();
|
||||||
private int _currentPlayerIndex;
|
private int _currentPlayerIndex;
|
||||||
|
|
||||||
public GameGroup(Queue<DirectionalPosition> spawns) => Spawns = spawns;
|
public Game(Queue<DirectionalPosition> spawns) => Spawns = spawns;
|
||||||
|
|
||||||
[JsonInclude] public Guid Id { get; } = Guid.NewGuid();
|
[JsonInclude] public Guid Id { get; } = Guid.NewGuid();
|
||||||
|
|
@ -1,5 +1,6 @@
|
|||||||
using pacMan.Game;
|
using pacMan.Exceptions;
|
||||||
using pacMan.Game.Items;
|
using pacMan.GameStuff;
|
||||||
|
using pacMan.GameStuff.Items;
|
||||||
|
|
||||||
namespace pacMan.Services;
|
namespace pacMan.Services;
|
||||||
|
|
||||||
@ -7,7 +8,7 @@ public class GameService : WebSocketService
|
|||||||
{
|
{
|
||||||
public GameService(ILogger<GameService> logger) : base(logger) { }
|
public GameService(ILogger<GameService> logger) : base(logger) { }
|
||||||
|
|
||||||
public SynchronizedCollection<GameGroup> Games { get; } = new();
|
public SynchronizedCollection<Game> Games { get; } = new();
|
||||||
|
|
||||||
public event Func<ArraySegment<byte>, Task>? Connections; // TODO remove and use GameGroup
|
public event Func<ArraySegment<byte>, Task>? Connections; // TODO remove and use GameGroup
|
||||||
|
|
||||||
@ -16,7 +17,7 @@ public class GameService : WebSocketService
|
|||||||
Connections?.Invoke(segment);
|
Connections?.Invoke(segment);
|
||||||
}
|
}
|
||||||
|
|
||||||
public GameGroup AddPlayer(IPlayer player, Queue<DirectionalPosition> spawns)
|
public Game AddPlayer(IPlayer player, Queue<DirectionalPosition> spawns)
|
||||||
{
|
{
|
||||||
var index = 0;
|
var index = 0;
|
||||||
try
|
try
|
||||||
@ -25,11 +26,21 @@ public class GameService : WebSocketService
|
|||||||
}
|
}
|
||||||
catch (ArgumentOutOfRangeException)
|
catch (ArgumentOutOfRangeException)
|
||||||
{
|
{
|
||||||
var game = new GameGroup(spawns);
|
var game = new Game(spawns);
|
||||||
game.AddPlayer(player);
|
game.AddPlayer(player);
|
||||||
Games.Add(game);
|
Games.Add(game);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Games[index];
|
return Games[index];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Game JoinById(Guid id, IPlayer player)
|
||||||
|
{
|
||||||
|
var game = Games.FirstOrDefault(g => g.Id == id) ?? throw new GameNotFoundException();
|
||||||
|
var added = game.AddPlayer(player);
|
||||||
|
|
||||||
|
if (!added) throw new ArgumentException("Game is full");
|
||||||
|
|
||||||
|
return game;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user