Aspnetcore: Integration testing is hard to set up and may break on shared framework servicing

Created on 17 May 2018  路  23Comments  路  Source: dotnet/aspnetcore

Sort of a follow-up to @poke's https://github.com/dotnet/sdk/issues/2253 but orthogonal.

TL;DR: Unit tests should have an option to run on the asp.net core shared framework using the same servicing strategy that the main subject-under-test application uses, which means referencing the shared framework using the latest known version of the web SDK and applying NuGet and assembly resolution accordingly.

Story:
As a developer, I want to add a new test project with a reference to my asp.net core project so that I can write integration tests.

Naive steps:

  • dotnet new razor -o ../RazorPagesProject
  • dotnet new xunit
  • dotnet add reference ../RazorPagesProject
  • dotnet add package Microsoft.AspNetCore.Mvc.Testing -v 2.1.0-rc1-final
  • add C# code
    ```c#
public class SampleIntegrationTests : IClassFixture<WebApplicationFactory<Startup>>
{
    public HttpClient Client { get; }

    public SampleIntegrationTests(WebApplicationFactory<Startup> fixture)
    {
        Client = fixture.CreateDefaultClient();
    }

    [Fact]
    public async Task ItShallWork()
    {
        var response = await Client.GetAsync("/");
        Assert.Equal(HttpStatusCode.OK, response.StatusCode);
    }
}
 * dotnet test

Expected: Test passes

Actual:

Starting test execution, please wait...
[xUnit.net 00:00:00.5209160] Discovering: SampleTests
[xUnit.net 00:00:00.5793000] Discovered: SampleTests
[xUnit.net 00:00:00.5854810] Starting: SampleTests
[xUnit.net 00:00:00.7291840] SampleTests.SampleIntegrationTests.ItShallWork [FAIL]
[xUnit.net 00:00:00.7305700] System.IO.FileNotFoundException : Could not load file or assembly 'Microsoft.AspNetCore, Version=2.1.0.0, Culture=neutral, PublicKeyToken=adb9793829ddae60'. The system cannot find the file specified.
[xUnit.net 00:00:00.7306390]
[xUnit.net 00:00:00.7461290] Finished: SampleTests
Failed SampleTests.SampleIntegrationTests.ItShallWork
Error Message:
System.IO.FileNotFoundException : Could not load file or assembly 'Microsoft.AspNetCore, Version=2.1.0.0, Culture=neutral, PublicKeyToken=adb9793829ddae60'. Thesystem cannot find the file specified.

Workaround:
Add a Package reference to the test project (!):

```xml
<ItemGroup>
    <PackageReference Include="Microsoft.AspNetCore.All" Version="2.1.0-rc1-final" />
</ItemGroup>

What I really want:

  • It should just work 鈩笍
  • Integration tests shall run on the asp.net core shared framework
  • Integration tests shall use the same patch level of asp.net core components that dotnet run or dotnet publish + dotnet myapp.dll would use

What could be done here? a web-test-SDK maybe?

Most helpful comment

I agree with @poke, the VS UI gets awkward (makes it look like a web project, adds connected services node, and adds launchsettings.json, I can "publish" my unit tests to Azure, etc.). Improved documentation and a template does helps people get it right though. It just would be nice if the VS UI wasn't misleading too.

@DamianEdwards Naive suggestion: why not have <Project Sdk="Microsoft.NET.Sdk.Test"> and <Project Sdk="Microsoft.NET.Sdk.Web.Test"> so that project nodes "definitively" look like test projects and work in their respective environments, netcoreapp testing and aspnetcoreapp testing? Additionally, the hypothetical test SDK could make <PackageReference Include="Microsoft.NET.Test.Sdk" Version="x.x.x" /> automagical, so users don't have to reference it from NuGet themselves. It would ideally just match the version of Visual Studio you are running.

All 23 comments

We've discussed this quite a bit, and it's currently by-design. You need to add a PackageReference to your test projects that have a ProjectReference to web apps.

I agree it is less than ideal. We explored lots of options, but were unable to find a default that we felt comfortable with that would have fixed this without breaking other scenarios. The root of the problem is the default behavior of NuGet and ProjectReference. By default, NuGet will transitively flow compilation references but exclude the build and analyzer assets.

Some things that could be done: changing the default for PrivateAssets (https://github.com/NuGet/Home/issues/6091), require manually setting PrivateAssets="None" on your PackageReference to Microsoft.AspNetCore.App, or use the Web SDK to update PrivateAssets. As we considered the impact of these, though, we didn't feel confident that these solved the problem any better. Some of these changes would have had other unintended consequences, so we decided to keep the current experience.

I see. was thinking about PrivateAssets before but I see that this could blow up easily as well..

Have there been discussions about adding a template for asp.net core integration tests? or "asp.net core app with integration test" multi-project templates

Doesn鈥檛 this completely defeat a major benefit of using the shared Microsoft.AspNetCore.App then? If I change the web project to have a _versioned_ reference instead, e.g.:

<PackageReference Include="Microsoft.AspNetCore.App" Version="2.1.0-rc1-final" />

Then the test project is able to pull in the dependencies transitively and everything works.

To me that seems much better than having to make the test project a Web SDK project, or having to add an explicitly versioned dependency in just the test project that I need to update properly so it鈥檚 still in sync to the web application project.

If I change the web project to have a versioned reference instead, ... then the test project is able to pull in the dependencies transitively and everything works.

Actually, this is close, but still not quite right. You're getting the compilation references, but your not actually running on the aspnet shared framework. Your test project will run on the Microsoft.NETCore.App shared framework, not Microsoft.AspNetCore.App. (checkout the runtimeconfig.json file in your bin folder).

Have there been discussions about adding a template for asp.net core integration tests? or "asp.net core app with integration test" multi-project templates

It came up, but we haven't committed to it. We were unsure how much demand there is for this.

But would that be a problem? If not running on the AspNetCore shared framework makes everything easier to grasp, then maybe that鈥檚 a better experience for end users of the framework?

It could be. If you are running from the package cache, you don't get the same roll-forward and security-patch semantics as a shared framework would.

There are lots of problems here to consider. The shared framework aspect is one of them, but there are more. Nuspec generation, target framework compatibility, NuGet UI, minor version rollforward for shared frameworks, etc. I appreciate your feedback, and it's something we'll continue to consider. But I agree with @dasMulli - this is a problematic and tricky area in which there do not seem to be good options.

Building a first-class

<TargetFramework>aspnetcoreapp2.1</TargetFramework>

could help but that has lots of other implications as well..

In the long run, this probably isn't going to get easier if a similar pattern is used for desktop apps on .net core 3.0.

FYI adding a template for tests is being tracked in https://github.com/aspnet/templating/issues/511 though it has not yet been triaged.

After working with this for a while now, there are two things that are really bothering me about having to make the test project a Web SDK project:

  • In Visual Studio, the test project has a normal web application icon; so I cannot quickly see that this is a test project.
  • This being a web project makes Visual Studio add a launchSettings.json to it by default, which is really unneded there. So one has to use <NoDefaultLaunchSettingsFile>True</NoDefaultLaunchSettingsFile> explicitly to disable that.

We're going to look at adding a project template for this in 2.2.0. Thanks for the feedback folks.

I agree with @poke, the VS UI gets awkward (makes it look like a web project, adds connected services node, and adds launchsettings.json, I can "publish" my unit tests to Azure, etc.). Improved documentation and a template does helps people get it right though. It just would be nice if the VS UI wasn't misleading too.

@DamianEdwards Naive suggestion: why not have <Project Sdk="Microsoft.NET.Sdk.Test"> and <Project Sdk="Microsoft.NET.Sdk.Web.Test"> so that project nodes "definitively" look like test projects and work in their respective environments, netcoreapp testing and aspnetcoreapp testing? Additionally, the hypothetical test SDK could make <PackageReference Include="Microsoft.NET.Test.Sdk" Version="x.x.x" /> automagical, so users don't have to reference it from NuGet themselves. It would ideally just match the version of Visual Studio you are running.

@natemcmaster - regarding your earlier comments in this thread, does some of the new thinking around the runtime resolve the issue here?

(There's already an issue tracking the creation of a test project template: https://github.com/aspnet/Templating/issues/511)

It doesn't exactly resolve it. If we move forward with removing the PackageReference from the .csproj (#3307), users will still need to set the RuntimeFrameworkName property in each test project. Flowing the value of this property across ProjectReference is something we could investigate doing in 2.2 as a follow up to https://github.com/dotnet/sdk/pull/2394, but that piece will be tricky to get right.

FYI - I've opened https://github.com/dotnet/sdk/issues/2420 to track some SDK changes we need to address the issue of mismatch shared runtimes

@natemcmaster Just so I understand that new issue correctly: If that one is resolved, then test projects essentially only need to reference the web project and they don鈥檛 need to set up anything on their own anymore (e.g. Web SDK and an explicit AspNetCore.App reference)? If that鈥檚 the case, that would be perfect!

That's only one piece of the puzzle, but that change alone isn't enough to really resolve it. I expect a full solution will invoke both SDK changes, something along the lines of #3307, and probably creating some templates which demonstrate that its all working well.

Alright, thank you! I鈥檓 looking forward to seeing how this all turns out in the end.

I believe this is now fixed by https://github.com/dotnet/sdk/pull/2533 in 2.2 tooling 馃帀

TL;DR: Test projects can now have a versionless asp.net core reference that will match the referenced project (provided there isn't any conflict due to self-contained roll-forward).

Side note: It鈥檚 not perfectly resolved since testing the app should ideally pull in the needed references transitively, so a reference to the shared framework should not be necessary.

But if I understand it correctly, with 3.0鈥檚 framework references, the dependencies can be pulled transitively, so this should work out of the box again.

let's see about framework reference. AFAIK it is now a NuGet but probably won't stay that way so we'll see, I'd suggest we open a new issue for that

Was this page helpful?
0 / 5 - 0 ratings