MyTested.AspNetCore.Mvc is a unit testing library (currently in Alpha version) providing easy fluent interface to test the ASP.NET Core MVC framework. It is testing framework agnostic, so you can combine it with a test runner of your choice (e.g. xUnit, NUnit, etc.).
It is strongly advised to start with the tutorial (coming soon) in order to get familiar with MyTested.AspNetCore.Mvc. Additionally, you may see the documentation (coming soon) for full list of available features. MyTested.AspNetCore.Mvc is 100% covered by more than 1500 unit tests and should work correctly. Almost all items in the issues page are expected future features and enhancements.
You can install this library using NuGet into your test project (or reference it directly in your project.json
file). Currently MyTested.AspNetCore.Mvc works with ASP.NET Core MVC RC2.
Install-Package MyTested.AspNetCore.Mvc
After the downloading is complete, just add using MyTested.AspNetCore.Mvc;
to your source code and you are ready to test in the most elegant and developer friendly way.
using MyTested.AspNetCore.Mvc;
For other interesting packages check out:
- MyTested.WebApi - fluent testing framework for ASP.NET Web API 2
- MyTested.HttpServer - fluent testing framework for remote HTTP servers
- AspNet.Mvc.TypedRouting - typed routing and link generation for ASP.NET Core MVC
- ASP.NET MVC 5 Lambda Expression Helpers - typed expression based link generation for ASP.NET MVC 5
Make sure to check out the tutorial (coming soon) and the documentation (coming soon) for a preview of the available features.
You can also check out the provided samples for real-life ASP.NET Core MVC application testing.
First we need to create a TestStartup
class in the root of the test project in order to register the dependency injection services. The easiest way is to inherit from the web project's Startup
class and replace some of the services with mocked ones by using the provided extension methods.
namespace MyApp.Tests
{
using MyTested.AspNetCore.Mvc;
using Microsoft.Extensions.DependencyInjection;
public class TestStartup : Startup
{
public void ConfigureTestServices(IServiceCollection services)
{
base.ConfigureServices(services);
services.Replace<IRepository, MockedRepository>();
}
}
}
And then you can create a test case by using the fluent API the library provides. You are given a static MyMvc
class from which all assertions can be easily configured.
namespace MyApp.Tests.Controllers
{
using MyTested.AspNetCore.Mvc;
using MyApp.Controllers;
using Xunit;
public class HomeControllerShould
{
[Fact]
public void ReturnViewWhenCallingIndexAction()
{
MyMvc
.Controller<HomeController>()
.Calling(c => c.Index())
.ShouldReturn()
.View();
}
}
}
The example uses xUnit but you can use whatever testing framework you want. Basically, MyTested.AspNetCore.Mvc throws an unhandled exception with a friendly error message if the assertion does not pass and the test fails.
Here are some random examples of what the fluent testing API is capable of:
// tests a route for correct controller, action and resolved route values
MyMvc
.Routes()
.ShouldMap(request => request
.WithLocation("/My/Action/1")
.WithMethod(HttpMethod.Post)
.WithJsonBody(new
{
Integer = 1,
String = "Text"
}))
.To<MyController>(c => c.Action(1, new MyModel
{
Integer = 1,
String = "Text"
}));
// instantiates controller with the registered services
// and mocks authenticated user
// and tests for valid model state
// and tests for valid view bag entry
// and tests model from view result with specific assertions
MyMvc
.Controller<MvcController>()
.WithAuthenticatedUser(user => user.WithUsername("MyUserName"))
.Calling(c => c.SomeAction(requestModel))
.ShouldHave()
.ValidModelState()
.AndAlso()
.ShouldHave()
.ViewBag(viewBag => viewBag
.ContainingEntry("MyViewBagProperty", "MyViewBagValue"))
.AndAlso()
.ShouldReturn()
.View()
.WithModel<ResponseModel>()
.Passing(m =>
{
Assert.AreEqual(1, m.Id);
Assert.AreEqual("Some property value", m.SomeProperty);
});
// tests whether the action is declared with filters
MyMvc
.Controller<MvcController>()
.Calling(c => c.MyAction())
.ShouldHave()
.ActionAttributes(attributes => attributes
.RestrictingForHttpMethod(HttpMethod.Get)
.AllowingAnonymousRequests()
.ValidatingAntiForgeryToken()
.ChangingActionNameTo("AnotherAction"));
// tests whether model state error exists by using lambda expression
// and specific tests for the error messages
MyMvc
.Controller<MvcController>()
.Calling(c => c.MyAction(requestWithErrors))
.ShouldHave()
.ModelState(modelState => modelState.For<RequestModel>()
.ContainingNoErrorFor(m => m.NonRequiredProperty)
.AndAlso()
.ContainingErrorFor(m => m.RequiredProperty)
.ThatEquals("The RequiredProperty field is required."));
// tests whether the action throws
// with exception of certain type and with certain message
MyMvc
.Controller<MvcController>()
.Calling(c => c.ActionWithException())
.ShouldThrow()
.Exception()
.OfType<NullReferenceException>()
.AndAlso()
.WithMessage().ThatEquals("Test exception message");
Code by Ivaylo Kenov. Copyright 2015 Ivaylo Kenov (http://mytestedasp.net)
Currently MyTested.AspNetCore.Mvc is in alpha version and it is not advised to use it in production environments. The testing framework is fully tested and working correctly but the fluent APIs may change in the final production-ready build.
MyTested.AspNetCore.Mvc source code is available under GNU Affero General Public License/FOSS License Exception. The free version of the library allows up to 500 assertions (around 100 test cases) per test project. Additionally, full-featured licenses can be requested for free by individuals, startups and educational institutions. Commercial licensing with private support will also be available with the final release.
See the LICENSE for detailed information.
If you want an early-bird license key for free, send a contact message through http://mytestedasp.net.
If you have a feature request or bug report, leave an issue on the issues page or send a pull request. For general questions and comments, use the StackOverflow forum.