Testavior is a lightweight solution to help you develop Behavior Tests for ASP.NET Core.
Behavior Tests are a way of testing your application features applying different types of behaviors to cover a functional scenario.
It provides a simple and efficient approach to write automated tests for your ASP.NET Core application.
For more information about Behavior Testing with ASP.NET Core, please take a look here http://geeklearning.io/a-different-approach-to-test-your-asp-net-core-application
Testavior provides 2 libraries:
- Testavior.Configuration: Helps you configure your application to easily integrate behavior tests for your scenarios.
- Testavior: Provides a featured and highly configurable test environment for your behavior tests:
- Configured Test WebHost
- Configured authentication context
- Test authentication middleware
- Configurable test identity
- Identity claims helper
- Configured Entity Framework Core context using SQLite provider
- Serialization helper to handle URL encoded content
- Set of HTTP tools to handle CSRF protection (very useful to test edition scenarios)
- Assertion helper
On your ASP.NET Core project
- Install the GeekLearning.Testavior.Configuration nuget package
> dotnet add package GeekLearning.Testavior.Configuration
On your .NET Core Unit Test project
- Install the GeekLearning.Testavior nuget package
> dotnet add package GeekLearning.Testavior - Add your ASP.NET Core web project as a project reference
The Test environment provided by Testavior is based on a Startup Configuration Service that let you separate the Production environment configuration from the Test environment configuration.
This configuration service is represented by a contract IStartupConfigurationService which define 3 methods: Configure - ConfigureEnvironment - ConfigureService that have to be called within the Startup Routine to inject environment dependent configuration.
1 - In your ASP.NET Core project:
- Add a
StartupConfigurationServiceclass (change name if you wish) to your web project. - Implement the
IStartupConfigurationServiceinterface (optionally, inherit fromDefaultStartupConfigurationServiceto use the default empty implementation) - Implement the configuration specific to the Production environment and which must not be executed in the Test environment:
ConfigureServices: implement the configuration options that are specific to the Production environmentConfigure: implement the middleware configuration specific to the Production environmentConfigureEnvironment: implement what has to be executed before anything
Sample:
public class StartupConfigurationService : DefaultStartupConfigurationService
{
public override void ConfigureServices(IServiceCollection services, IConfigurationRoot configuration)
{
base.ConfigureServices(services, configuration);
var connection = "CONNECTION_STRING";
services.AddDbContext<[EF_DB_CONTEXT]>(options =>
options.UseSqlServer(connection));
}
}2 - In your Program class:
Inject your StartupConfigurationService by calling the ConfigureStartup method on your WebHostBuilder:
new WebHostBuilder()
...
.UseStartup<Startup>()
.ConfigureStartup<StartupConfigurationService>()3 - In your Startup class:
- Inject the
IStartupConfigurationServiceinterface into theStartupclass - Call the
ConfigureEnvironmentmethod at the end of theStartupconstructor - Call the
ConfigureServicesmethod at the end of the originalStartup.ConfigureServicesmethod - Call the
Configuremethod at the beginning of the originalStartup.Configuremethod
Sample:
public class Startup
{
private IStartupConfigurationService externalStartupConfiguration;
public Startup(IHostingEnvironment env, IStartupConfigurationService externalStartupConfiguration = null)
{
this.externalStartupConfiguration = externalStartupConfiguration;
this.externalStartupConfiguration.ConfigureEnvironment(env);
}
public void ConfigureServices(IServiceCollection services)
{
services.AddMvc()
// Pass configuration (IConfigurationRoot) to the configuration service if needed
this.externalStartupConfiguration.ConfigureServices(services, null);
}
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
this.externalStartupConfiguration.Configure(app, env, loggerFactory);
app.UseMvc();
}
}4 - In your test project file:
The Razor engine uses dependency files (.deps.json) to resolve some references at runtime. So in order to test the MVC part of a application, it is necessary to import these files. To do it, add the following section to your .csproj:
<Target Name="CopyDepsFiles" AfterTargets="Build" Condition="'$(TargetFramework)'!=''">
<ItemGroup>
<DepsFilePaths Include="$([System.IO.Path]::ChangeExtension('%(_ResolvedProjectReferencePaths.FullPath)', '.deps.json'))" />
</ItemGroup>
<Copy SourceFiles="%(DepsFilePaths.FullPath)" DestinationFolder="$(OutputPath)" Condition="Exists('%(DepsFilePaths.FullPath)')" />
</Target>5 - For xUnit users
If you intend to use xUnit, first follow the official documention, then add a xunit.runner.json file to your test project:
{
"shadowCopy": false
}and add the following section to your .csproj:
<ItemGroup>
<None Include="xunit.runner.json" CopyToOutputDirectory="PreserveNewest" />
</ItemGroup>A specific IStartupConfigurationService is required for the Test environment if you want to implement Test Specific configuration.
Testavior comes with a test specific IStartupConfigurationService implementation: TestStartupConfigurationService which provide a Test Environment full of useful features (see Features section).
Of course you can implement your own Startup configuration service (by using the onboard TestStartupConfigurationService or not).
To create a Test Environment, just instanciate the TestEnvironment class by passing it your ASP.NET Core application Startup, your IStartupConfigurationService implementation, the type of your EF Core ObjectContext and the relative path to your ASP.NET Core project (required to resolve MVC views).
var testEnvironment = new TestEnvironment<Startup, TestStartupConfigurationService<[EF_DB_CONTEXT]>>(
Path.Combine(System.AppContext.BaseDirectory, @"[PATH_TO_WEB_APP]"));Write your API test by just sending web requests using the Test Environment:
[TestMethod]
public void ScenarioShouldBeOk()
{
var testEnvironment = new TestEnvironment<Startup, TestStartupConfigurationService<[EF_DB_CONTEXT]>>(
Path.Combine(System.AppContext.BaseDirectory, @"[PATH_TO_WEB_APP]"));
var response = testEnvironment.Client.GetAsync("/api/data").Result;
response.EnsureSuccessStatusCode();
// Test result content
var result = JsonConvert.DeserializeObject<Data[]>(response.Content.ReadAsStringAsync().Result);
Assert.AreEqual("data", result.Data);
}Write a MVC test is almost as easy as testing an API except that you might want to test the Model returned by the server and not the View.
To do that, Testavior provides a ViewModel Repository that will intercept and store the view's models returned by the server.
You can access to the this repository using the ASP.NET Core dependency injection mechanism:
[TestMethod]
public void ScenarioShouldBeOk()
{
var testEnvironment = new TestEnvironment<Startup, TestStartupConfigurationService<[EF_DB_CONTEXT]>>(
Path.Combine(System.AppContext.BaseDirectory, @"[PATH_TO_WEB_APP]"));
testEnvironment.Client.GetAsync("/").Result.EnsureSuccessStatusCode();
var viewModel = testEnvironment
.ServiceProvider
.GetRequiredService<ViewModelRepository>()
.Get<[VIEWMODEL_TYPE]>();
Assert.AreEqual("data", viewModel.Data);
}And feel free to take a look at the Samples section ;)
Happy testing ! :)