Entity Framework Core 2 Unit Testing in .NET Core 2
Full source code available here.
Unit testing Entity Framework used to be quite a chore, but over the past few years it has become significantly easier.
In this post I’m going to show you how to use the InMemory
database with named instances.
The ProductsController
is fairly standard, it takes a DbContext
in the constructor which is instantiated by dependency injection (see the Startup.cs
in the source code if you don’t know how to do this).
1public ProductsController(ProductsContext context)
2{
3 _productsContext = context;
4}
The action methods use the context to query the database and return responses to the callers. From the response status code the caller knows if there is also content (products) in the response.
1[HttpGet]
2public async Task<IActionResult> Get()
3{
4 var products = await _productsContext.Products.ToListAsync();
5
6 if (!products.Any())
7 {
8 return NotFound();
9 }
10 return Ok(products);
11}
Testing
In the unit test project, (see here if you need help creating this), I’ve added a ProductsControllerTests
class.
In my Get
test I created an instance of a DbContextOptions
by using the DbContextOptionsBuilder
. On the builder I specify that I want an InMemoryDatabase
and give it a unique name for this test - this is important, if the name is not unique across all tests or if it is left blank you will end up with a database shared by more than one test.
In the Arrange
portion of the test I also instantiate the ProductsContext
; then instantiate a few products, add them to the context; don’t forget to save the changes; finally create an instance of the of ProductsController
.
Here is the full Arrange
portion of the test.
1// Arrange
2var options = new DbContextOptionsBuilder<ProductsContext>()
3.UseInMemoryDatabase(databaseName: "Get test")
4.Options;
5
6ProductsContext productsContext = new ProductsContext(options);
7Product product01 = new Product { Name = "first", ProductId = 1, Sku = "abc" };
8Product product02 = new Product { Name = "second", ProductId = 2, Sku = "def" };
9
10productsContext.Products.Add(product01);
11productsContext.Products.Add(product02);
12await productsContext.SaveChangesAsync();
13ProductsController controller = new ProductsController(productsContext);
Next I Act
, just calling the controller’s Get
method. The controller already has the context and the get method will query it just like a real database.
1// Act
2var actionResult = await controller.Get();
Then I Assert.
1// Assert
2var okObjectResult = actionResult as OkObjectResult;
3
4List<Product> products = okObjectResult.Value as List<Product>;
5
6Assert.Equal(2, products.Count());
Note the as OkObjectResult
type, I know it’s that type because that is what I return from the ProductsController
.
I assign the value of the okObjectResult
to a List
.Assert
that there are two entries in the list.
Testing NotFound - 404
In the ProductsController
I also have a Get
that takes a productId
as an int
and returns a product.
You should also test the scenario that no entries in the database match the search. In this case I am returning a NotFound
, you may choose to return something else like an OK
with nothing in it. In the case of a list, some people like to return an empty list; there is much debate on this topic and I have no interest in adding my two cents.
To test the NotFound
response, the Act
and Assert
change a little. I call the GET
method with an id that I know will return nothing.
1// Act
2var actionResult = await controller.Get(999);
3
4// Assert
5var notFoundResult = actionResult as NotFoundResult;
6
7Assert.Equal(404, notFoundResult.StatusCode);
Full source code available here.