From 6f23cf161de19c4e220ea683361d4bedec4acfb7 Mon Sep 17 00:00:00 2001
From: Nano Taboada <87288+nanotaboada@users.noreply.github.com>
Date: Wed, 14 Feb 2024 00:32:54 -0300
Subject: [PATCH] feat: add unit tests for Retrieve methods of Service layer
---
...net.AspNetCore.Samples.WebApi.Tests.csproj | 5 +-
.../PlayerServiceTests.cs | 100 ++++++-
.../Controllers/PlayersController.cs | 8 +-
.../Data/PlayerContextInitializer.cs | 4 +-
.../Data/PlayerDataBuilder.cs | 246 ++++++++++++++++--
Dotnet.AspNetCore.Samples.WebApi/Program.cs | 8 +-
.../Services/IPlayerService.cs | 2 +-
.../Services/PlayerService.cs | 2 +-
8 files changed, 327 insertions(+), 48 deletions(-)
diff --git a/Dotnet.AspNetCore.Samples.WebApi.Tests/Dotnet.AspNetCore.Samples.WebApi.Tests.csproj b/Dotnet.AspNetCore.Samples.WebApi.Tests/Dotnet.AspNetCore.Samples.WebApi.Tests.csproj
index 0f3b4a6..4ebd07b 100644
--- a/Dotnet.AspNetCore.Samples.WebApi.Tests/Dotnet.AspNetCore.Samples.WebApi.Tests.csproj
+++ b/Dotnet.AspNetCore.Samples.WebApi.Tests/Dotnet.AspNetCore.Samples.WebApi.Tests.csproj
@@ -12,7 +12,6 @@
-
runtime; build; native; contentfiles; analyzers; buildtransitive
@@ -24,8 +23,8 @@
-
-
+
+
diff --git a/Dotnet.AspNetCore.Samples.WebApi.Tests/PlayerServiceTests.cs b/Dotnet.AspNetCore.Samples.WebApi.Tests/PlayerServiceTests.cs
index f71afd8..f2e05fd 100644
--- a/Dotnet.AspNetCore.Samples.WebApi.Tests/PlayerServiceTests.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi.Tests/PlayerServiceTests.cs
@@ -1,28 +1,94 @@
+using System.Data.Common;
using System.Diagnostics;
using Dotnet.AspNetCore.Samples.WebApi.Data;
using Dotnet.AspNetCore.Samples.WebApi.Models;
using Dotnet.AspNetCore.Samples.WebApi.Services;
using FluentAssertions;
+using Microsoft.Data.Sqlite;
+using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Caching.Memory;
using Microsoft.Extensions.Logging;
using Moq;
-using Moq.EntityFrameworkCore;
namespace Dotnet.AspNetCore.Samples.WebApi.Tests;
-public class PlayerServiceTests
+public class PlayerServiceTests : IDisposable
{
+ private readonly DbConnection dbConnection;
+ private readonly DbContextOptions dbContextOptions;
+
+ public PlayerServiceTests()
+ {
+ dbConnection = new SqliteConnection("Filename=:memory:");
+ dbConnection.Open();
+
+ dbContextOptions = new DbContextOptionsBuilder()
+ .UseSqlite(dbConnection)
+ .Options;
+
+ using var context = new PlayerContext(dbContextOptions);
+
+ if (context.Database.EnsureCreated())
+ {
+ using var dbCommand = context.Database.GetDbConnection().CreateCommand();
+
+ dbCommand.CommandText = """
+ CREATE TABLE IF NOT EXISTS "players"
+ (
+ "id" INTEGER,
+ "firstName" TEXT NOT NULL,
+ "middleName" TEXT,
+ "lastName" TEXT NOT NULL,
+ "dateOfBirth" TEXT,
+ "squadNumber" INTEGER NOT NULL,
+ "position" TEXT NOT NULL,
+ "abbrPosition" TEXT,
+ "team" TEXT,
+ "league" TEXT,
+ "starting11" BOOLEAN,
+ PRIMARY KEY("id")
+ );
+ """;
+
+ dbCommand.ExecuteNonQuery();
+ }
+
+ context.AddRange(PlayerDataBuilder.SeedWithDeserializedJson());
+ context.SaveChanges();
+ }
+
+ PlayerContext CreatePlayerContext() => new PlayerContext(dbContextOptions);
+ public void Dispose() => dbConnection.Dispose();
+
[Fact]
[Trait("Category", "Retrieve")]
- public async Task GivenRetrieve_WhenInvokedTwice_ThenSecondExecutionTimeShouldBeLessThanFirst()
+ public async Task GivenRetrieveAsync_WhenInvoked_ThenShouldReturnAllPlayers()
{
// Arrange
- var players = PlayerDataBuilder.SeedWithStarting11().ToList();
- var context = new Mock();
- context.Setup(context => context.Players).ReturnsDbSet(players);
+ var players = PlayerDataBuilder.SeedWithDeserializedJson();
+ var context = CreatePlayerContext();
var logger = new Mock>();
var memoryCache = new MemoryCache(new MemoryCacheOptions());
- var service = new PlayerService(context.Object, logger.Object, memoryCache);
+
+ var service = new PlayerService(context, logger.Object, memoryCache);
+
+ // Act
+ var result = await service.RetrieveAsync();
+
+ // Assert
+ result.Should().BeEquivalentTo(players);
+ }
+
+ [Fact]
+ [Trait("Category", "Retrieve")]
+ public async Task GivenRetrieveAsync_WhenInvokedTwice_ThenSecondExecutionTimeShouldBeLessThanFirst()
+ {
+ // Arrange
+ var context = CreatePlayerContext();
+ var logger = new Mock>();
+ var memoryCache = new MemoryCache(new MemoryCacheOptions());
+
+ var service = new PlayerService(context, logger.Object, memoryCache);
// Act
var first = await ExecutionTimeAsync(() => service.RetrieveAsync());
@@ -41,4 +107,24 @@ private async Task ExecutionTimeAsync(Func awaitable)
return stopwatch.ElapsedMilliseconds;
}
+
+ [Fact]
+ [Trait("Category", "Retrieve")]
+ public async Task GivenRetrieveByIdAsync_WhenInvokedWithPlayerId_ThenShouldReturnThePlayer()
+ {
+ // Arrange
+ var player = PlayerDataBuilder.SeedOneById(10);
+ var context = CreatePlayerContext();
+ var logger = new Mock>();
+ var memoryCache = new MemoryCache(new MemoryCacheOptions());
+
+ var service = new PlayerService(context, logger.Object, memoryCache);
+
+ // Act
+ var result = await service.RetrieveByIdAsync(10);
+
+ // Assert
+ result.Should().BeOfType();
+ result.Should().BeEquivalentTo(player);
+ }
}
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Controllers/PlayersController.cs b/Dotnet.AspNetCore.Samples.WebApi/Controllers/PlayersController.cs
index ec738a9..7d80a19 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Controllers/PlayersController.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Controllers/PlayersController.cs
@@ -38,7 +38,7 @@ HTTP POST
[HttpPost]
public async Task> PostPlayer(Player player)
{
- if (await _playerService.RetrieveById(player.Id) != null)
+ if (await _playerService.RetrieveByIdAsync(player.Id) != null)
{
return Conflict();
}
@@ -74,7 +74,7 @@ public async Task>> GetPlayers()
[HttpGet("{id}")]
public async Task> GetPlayer(long id)
{
- var player = await _playerService.RetrieveById(id);
+ var player = await _playerService.RetrieveByIdAsync(id);
if (player != null)
{
@@ -100,7 +100,7 @@ public async Task PutPlayer(long id, Player player)
{
return BadRequest();
}
- else if (await _playerService.RetrieveById(id) == null)
+ else if (await _playerService.RetrieveByIdAsync(id) == null)
{
return NotFound();
}
@@ -121,7 +121,7 @@ HTTP DELETE
[HttpDelete("{id}")]
public async Task DeletePlayer(long id)
{
- if (await _playerService.RetrieveById(id) == null)
+ if (await _playerService.RetrieveByIdAsync(id) == null)
{
return NotFound();
}
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerContextInitializer.cs b/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerContextInitializer.cs
index e493168..8422195 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerContextInitializer.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerContextInitializer.cs
@@ -1,5 +1,6 @@
using Dotnet.AspNetCore.Samples.WebApi.Data;
using Dotnet.AspNetCore.Samples.WebApi.Models;
+using Microsoft.EntityFrameworkCore;
namespace Dotnet.AspNetCore.Samples.WebApi;
@@ -10,9 +11,10 @@ public static void Seed(IApplicationBuilder applicationBuilder)
using (var scope = applicationBuilder.ApplicationServices.CreateScope())
{
var context = scope.ServiceProvider.GetService();
-
+
if (context != null)
{
+ // https://learn.microsoft.com/en-us/ef/core/managing-schemas/ensure-created
context.Database.EnsureCreated();
if (!context.Players.Any())
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerDataBuilder.cs b/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerDataBuilder.cs
index f70ab4b..e698167 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerDataBuilder.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Data/PlayerDataBuilder.cs
@@ -5,7 +5,203 @@ namespace Dotnet.AspNetCore.Samples.WebApi.Data;
public static class PlayerDataBuilder
{
- public static Player[] SeedWithStarting11()
+ public static Player? SeedOneById(int id)
+ {
+ return SeedWithStarting11()
+ .SingleOrDefault(player => player.Id == id);
+ }
+
+ public static List SeedWithStarting11()
+ {
+ var players = new List();
+
+ players.Add(
+ new Player
+ {
+ Id = 1,
+ FirstName = "Damián",
+ MiddleName = "Emiliano",
+ LastName = "Martínez",
+ DateOfBirth = new DateTime(1992, 9, 1, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 23,
+ Position = "Goalkeeper",
+ AbbrPosition = "GK",
+ Team = "Aston Villa FC",
+ League = "Premier League",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 2,
+ FirstName = "Nahuel",
+ LastName = "Molina",
+ DateOfBirth = new DateTime(1998, 4, 5, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 26,
+ Position = "Right-Back",
+ AbbrPosition = "RB",
+ Team = "Altético Madrid",
+ League = "La Liga",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 3,
+ FirstName = "Cristian",
+ MiddleName = "Gabriel",
+ LastName = "Romero",
+ DateOfBirth = new DateTime(1998, 4, 26, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 13,
+ Position = "Centre-Back",
+ AbbrPosition = "CB",
+ Team = "Tottenham Hotspur",
+ League = "Premier League",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 4,
+ FirstName = "Nicolás",
+ MiddleName = "Hernán Gonzalo",
+ LastName = "Otamendi",
+ DateOfBirth = new DateTime(1988, 2, 11, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 19,
+ Position = "Centre-Back",
+ AbbrPosition = "CB",
+ Team = "SL Benfica",
+ League = "Liga Portugal",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 5,
+ FirstName = "Nicolás",
+ MiddleName = "Alejandro",
+ LastName = "Tagliafico",
+ DateOfBirth = new DateTime(1992, 8, 30, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 3,
+ Position = "Left-Back",
+ AbbrPosition = "LB",
+ Team = "Olympique Lyon",
+ League = "Ligue 1",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 6,
+ FirstName = "Ángel",
+ MiddleName = "Fabián",
+ LastName = "Di María",
+ DateOfBirth = new DateTime(1988, 2, 13, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 11,
+ Position = "Right Winger",
+ AbbrPosition = "RW",
+ Team = "SL Benfica",
+ League = "Liga Portugal",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 7,
+ FirstName = "Rodrigo",
+ MiddleName = "Javier",
+ LastName = "de Paul",
+ DateOfBirth = new DateTime(1994, 5, 23, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 7,
+ Position = "Central Midfield",
+ AbbrPosition = "CM",
+ Team = "Altético Madrid",
+ League = "La Liga",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 8,
+ FirstName = "Enzo",
+ MiddleName = "Jeremías",
+ LastName = "Fernández",
+ DateOfBirth = new DateTime(2001, 1, 16, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 24,
+ Position = "Central Midfield",
+ AbbrPosition = "CM",
+ Team = "Chelsea FC",
+ League = "Premier League",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 9,
+ FirstName = "Alexis",
+ LastName = "Mac Allister",
+ DateOfBirth = new DateTime(1998, 12, 23, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 20,
+ Position = "Central Midfield",
+ AbbrPosition = "CM",
+ Team = "Liverpool FC",
+ League = "Premier League",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 10,
+ FirstName = "Lionel",
+ MiddleName = "Andrés",
+ LastName = "Messi",
+ DateOfBirth = new DateTime(1987, 6, 23, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 10,
+ Position = "Right Winger",
+ AbbrPosition = "RW",
+ Team = "Inter Miami CF",
+ League = "Major League Soccer",
+ Starting11 = true,
+ }
+ );
+
+ players.Add(
+ new Player
+ {
+ Id = 11,
+ FirstName = "Julián",
+ LastName = "Álvarez",
+ DateOfBirth = new DateTime(2000, 1, 30, 0, 0, 0, DateTimeKind.Utc),
+ SquadNumber = 9,
+ Position = "Centre-Forward",
+ AbbrPosition = "CF",
+ Team = "Manchester City",
+ League = "Premier League",
+ Starting11 = true,
+ }
+ );
+
+ return players;
+ }
+ public static List SeedWithDeserializedJson()
{
var players = new List();
@@ -15,143 +211,143 @@ public static Player[] SeedWithStarting11()
"firstName": "Damián",
"middleName": "Emiliano",
"lastName": "Martínez",
- "dateOfBirth": "1992-09-01T21:00:00-00:00",
+ "dateOfBirth": "1992-09-01T00:00:00.000Z",
"squadNumber": 23,
"position": "Goalkeeper",
"abbrPosition": "GK",
"team": "Aston Villa FC",
"league": "Premier League",
- "starting11": false
+ "starting11": true
},
{
"id": 2,
"firstName": "Nahuel",
"middleName": null,
"lastName": "Molina",
- "dateOfBirth": "1998-04-05T21:00:00-00:00",
+ "dateOfBirth": "1998-04-05T00:00:00.000Z",
"squadNumber": 26,
"position": "Right-Back",
"abbrPosition": "RB",
"team": "Atlético Madrid",
"league": "La Liga",
- "starting11": false
+ "starting11": true
},
{
"id": 3,
"firstName": "Cristian",
"middleName": "Gabriel",
"lastName": "Romero",
- "dateOfBirth": "1998-04-26T21:00:00-00:00",
+ "dateOfBirth": "1998-04-26T00:00:00.000Z",
"squadNumber": 13,
"position": "Centre-Back",
"abbrPosition": "CB",
"team": "Tottenham Hotspur",
"league": "Premier League",
- "starting11": false
+ "starting11": true
},
{
"id": 4,
"firstName": "Nicolás",
"middleName": "Hernán Gonzalo",
"lastName": "Otamendi",
- "dateOfBirth": "1988-02-11T21:00:00-00:00",
+ "dateOfBirth": "1988-02-11T00:00:00.000Z",
"squadNumber": 19,
"position": "Centre-Back",
"abbrPosition": "CB",
"team": "SL Benfica",
"league": "Liga Portugal",
- "starting11": false
+ "starting11": true
},
{
"id": 5,
"firstName": "Nicolás",
"middleName": "Alejandro",
"lastName": "Tagliafico",
- "dateOfBirth": "1992-08-30T21:00:00-00:00",
+ "dateOfBirth": "1992-08-30T00:00:00.000Z",
"squadNumber": 3,
"position": "Left-Back",
"abbrPosition": "LB",
"team": "Olympique Lyon",
"league": "Ligue 1",
- "starting11": false
+ "starting11": true
},
{
"id": 6,
"firstName": "Ángel",
"middleName": "Fabián",
"lastName": "Di María",
- "dateOfBirth": "1988-02-13T21:00:00-00:00",
+ "dateOfBirth": "1988-02-13T00:00:00.000Z",
"squadNumber": 11,
"position": "Right Winger",
- "abbrPosition": "LW",
+ "abbrPosition": "RW",
"team": "SL Benfica",
"league": "Liga Portugal",
- "starting11": false
+ "starting11": true
},
{
"id": 7,
"firstName": "Rodrigo",
"middleName": "Javier",
"lastName": "de Paul",
- "dateOfBirth": "1994-05-23T21:00:00-00:00",
+ "dateOfBirth": "1994-05-23T00:00:00.000Z",
"squadNumber": 7,
"position": "Central Midfield",
"abbrPosition": "CM",
"team": "Atlético Madrid",
"league": "La Liga",
- "starting11": false
+ "starting11": true
},
{
"id": 8,
"firstName": "Enzo",
"middleName": "Jeremías",
"lastName": "Fernández",
- "dateOfBirth": "2001-01-16T21:00:00-00:00",
+ "dateOfBirth": "2001-01-16T00:00:00.000Z",
"squadNumber": 24,
"position": "Central Midfield",
"abbrPosition": "CM",
"team": "Chelsea FC",
"league": "Premier League",
- "starting11": false
+ "starting11": true
},
{
"id": 9,
"firstName": "Alexis",
"middleName": null,
"lastName": "Mac Allister",
- "dateOfBirth": "1998-12-23T21:00:00-00:00",
+ "dateOfBirth": "1998-12-23T00:00:00.000Z",
"squadNumber": 20,
"position": "Central Midfield",
"abbrPosition": "CM",
"team": "Liverpool FC",
"league": "Premier League",
- "starting11": false
+ "starting11": true
},
{
"id": 10,
"firstName": "Lionel",
"middleName": "Andrés",
"lastName": "Messi",
- "dateOfBirth": "1987-06-23T21:00:00-00:00",
+ "dateOfBirth": "1987-06-23T00:00:00.000Z",
"squadNumber": 10,
"position": "Right Winger",
"abbrPosition": "RW",
"team": "Inter Miami CF",
"league": "Major League Soccer",
- "starting11": false
+ "starting11": true
},
{
"id": 11,
"firstName": "Julián",
"middleName": null,
"lastName": "Álvarez",
- "dateOfBirth": "2000-01-30T21:00:00-00:00",
+ "dateOfBirth": "2000-01-30T00:00:00.000Z",
"squadNumber": 9,
"position": "Centre-Forward",
"abbrPosition": "CF",
"team": "Manchester City",
"league": "Premier League",
- "starting11": false
+ "starting11": true
}]
""";
@@ -163,6 +359,6 @@ public static Player[] SeedWithStarting11()
players.AddRange(starting11);
}
- return players.ToArray();
+ return players;
}
}
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Program.cs b/Dotnet.AspNetCore.Samples.WebApi/Program.cs
index 5e95af7..e4cd65c 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Program.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Program.cs
@@ -1,6 +1,7 @@
using Dotnet.AspNetCore.Samples.WebApi;
using Dotnet.AspNetCore.Samples.WebApi.Models;
using Dotnet.AspNetCore.Samples.WebApi.Services;
+using Microsoft.Data.Sqlite;
using Microsoft.EntityFrameworkCore;
var builder = WebApplication.CreateBuilder(args);
@@ -13,13 +14,8 @@
builder.Services.AddControllers();
-// var localApplicationData = Environment.SpecialFolder.LocalApplicationData;
-// var folderPath = Environment.GetFolderPath(localApplicationData);
-var projectDataFolder = "Data";
-var path = Path.Join(projectDataFolder, "players-sqlite3.db");
-
builder.Services.AddDbContext(options =>
- options.UseSqlite($"Data Source={path}")
+ options.UseSqlite(@"Data Source=Data/players-sqlite3.db")
);
// Learn more about configuring Swagger/OpenAPI at https://aka.ms/aspnetcore/swashbuckle
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Services/IPlayerService.cs b/Dotnet.AspNetCore.Samples.WebApi/Services/IPlayerService.cs
index 96d32dd..3774c55 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Services/IPlayerService.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Services/IPlayerService.cs
@@ -9,7 +9,7 @@ public interface IPlayerService
public Task> RetrieveAsync();
- public ValueTask RetrieveById(long id);
+ public ValueTask RetrieveByIdAsync(long id);
public Task UpdateAsync(Player player);
diff --git a/Dotnet.AspNetCore.Samples.WebApi/Services/PlayerService.cs b/Dotnet.AspNetCore.Samples.WebApi/Services/PlayerService.cs
index 12e82b2..deaf7b1 100644
--- a/Dotnet.AspNetCore.Samples.WebApi/Services/PlayerService.cs
+++ b/Dotnet.AspNetCore.Samples.WebApi/Services/PlayerService.cs
@@ -57,7 +57,7 @@ public async Task> RetrieveAsync()
}
}
- public ValueTask RetrieveById(long id)
+ public ValueTask RetrieveByIdAsync(long id)
{
return _playerContext.Players.FindAsync(id);
}