Jump to content
Xtreme .Net Talk

Supercharge your testing experience with MSTest.Analyzers


Recommended Posts

Guest Jakub Jareš
Posted

MSTest ships with a set of analyzers that inspect your test code and point out common mistakes and pitfalls. These mistakes can be subtle and lead to your tests being completely ignored by the test framework.

 

We’ve been shipping these analyzers since 3.2.0, but in the latest 3.5.1, we’ve added some that we think you should not miss.

 

[HEADING=1]The missing test[/HEADING]

 

One common problem is when you forget to put [iCODE][TestClass][/iCODE] on your class. MSTest won’t know that there are tests in the class, and it won’t run them:

 

public class MyTests

{
   [TestMethod]
   public async Task TestMethod1()
   {
     Assert.Fail();
   }
}

 

Without MSTest.Analyzers, this code builds without warning or info message. There is no test failure when running the tests either. Because there is no [iCODE][TestClass][/iCODE] attribute on the class, MSTest will skip over the whole class for performance reasons, and your test will never be found.

 

But with analyzers you get an info message during build:

 

[ATTACH type=full" alt="Info message for MSTEST0030 shown in Visual Studio Error List]5912[/ATTACH]

 

We recommend, to upgrade this message to a warning, or even build error, one such way is by adding this line to your [iCODE].editorconfig[/iCODE]:

 

[*.cs]
# MSTEST0030: Type containing '[TestMethod]' should be marked with '[TestClass]'
dotnet_diagnostic.MSTEST0030.severity = warning

 

[ATTACH type=full" alt="Warning message for MSTEST0030 shown in Visual Studio Error List]5913[/ATTACH]

 

[HEADING=2]Why is this not a warning by default?[/HEADING]

 

You might be asking why [iCODE]MSTEST0030[/iCODE] is an info message and not warning by default. The reason is that we cannot introduce breaking changes in MSTest v3, and the code above is a common pattern for re-using tests from a base class.

 

public class MyTestsBase
{
   [TestMethod]
   public async Task CommonTestMethod()
   {
   }
}

[TestClass]
public class MyTests : MyTestsBase
{
   [TestMethod]
   public async Task TestMethod1()
   {
     Assert.Fail();
   }
}

 

In the example above, the test [iCODE]CommonTestMethod[/iCODE] will not run from [iCODE]MyTestsBase[/iCODE], because it does not have [iCODE][TestClass][/iCODE] attribute, but it will be inherited to [iCODE]MyTests[/iCODE], and will run there.

 

We DO NOT recommend this pattern. Instead, we recommend to always mark classes with [iCODE][TestClass][/iCODE], and make the base class abstract, if you don’t want to run tests from it.

 

[TestClass]
public abstract class MyTestsBase
{
   [TestMethod]
   public async Task CommonTestMethod()
   {
   }
}

[TestClass]
public class MyTests : MyTestsBase
{
   [TestMethod]
   public async Task TestMethod1()
   {
     Assert.Fail();
   }
}

 

This approach behaves identically to the one above, but you communicate it clearly to the analyzers and the testing framework that the [iCODE]abstract[/iCODE] base class is holding a shared logic, and should not run by itself.

 

[HEADING=1]Malformed AssemblyInitialize[/HEADING]

 

Another example of a useful analyzer is fixing the signature of [iCODE][AssemblyInitialize][/iCODE], to do a one-time setup for all the tests in the assembly.

 

I don’t know about you, but I can’t remember the signature of this method. And when I get it wrong my tests don’t run at all. This is especially annoying in Visual Studio, where the test simply remains blue, and I need to go into the [iCODE]Tests[/iCODE] output to find the reason.

 

But with the analyzers, I can easily find out what is wrong, and there is even an automatic fix that I can apply to my code.

 

Here I write method Setup, and VisualStudio will underline it with a warning, and I press [iCODE]Ctrl+.[/iCODE] to see the automatic fix and apply it:

 

[TestClass]
public class MyTests
{
 [AssemblyInitialize]
 public void Setup()
 {
 }
}

 

[ATTACH type=full" alt="Warning message for MSTEST0012 shown in Visual Studio Error List]5914[/ATTACH] [ATTACH type=full" alt="Fixing signature via lightbulb menu]5915[/ATTACH]

 

Fixing my code to the correct shape that the testing framework can recognize:

 

 [TestClass]
public class MyTests
{
    [AssemblyInitialize]
    public static void Setup(TestContext context)
    {
    }
}

[HEADING=1]Installation[/HEADING]

 

The recommended way to start using [iCODE]MSTest.Analyzers[/iCODE] is by using the [iCODE]MSTest[/iCODE] nuget package, or the MSTest project SDK, with version 3.2.0 or newer.

 

Get started with MSTest - .NET

 

The analyzers can also be installed separately by referencing the [iCODE]MSTest.Analyzers[/iCODE] NuGet package.

 

We wholeheartedly recommend any project that is using MSTest to upgrade to version 3.2.0 and newer, and enable those analyzers.

 

[HEADING=1]Summary[/HEADING]

 

The two analyzers showcased in this article are the ones that I find the most useful in my day-to-day. But there is a whole lot more, such as:

 

  • MSTEST0003, which ensures that your test method has correct signature, such as being [iCODE]public[/iCODE], and not being [iCODE]async void[/iCODE].
  • MSTEST0001, which recommends enabling test parallelization, because we’ve seen multiple test bases dramatically reduce their test execution time by doing so.
  • MSTEST0017, which ensures that you pass arguments to assertions in the correct order, to avoid confusing test failure messages.

 

And 32 more rules, that are split into Design, Performance, and Usage categories. Helping you to write well formed, performant and error free tests.

 

MSTest code analysis - .NET

 

We are constantly looking for ways to improve these analyzers or new analyzers that our customers are missing. We welcome you to share your feedback, ideas for more analyzers, and your experience, on our repository microsoft/testfx.

 

The post Supercharge your testing experience with MSTest.Analyzers appeared first on .NET Blog.

 

Continue reading...

Join the conversation

You can post now and register later. If you have an account, sign in now to post with your account.

Guest
Reply to this topic...

×   Pasted as rich text.   Paste as plain text instead

  Only 75 emoji are allowed.

×   Your link has been automatically embedded.   Display as a link instead

×   Your previous content has been restored.   Clear editor

×   You cannot paste images directly. Upload or insert images from URL.

×
×
  • Create New...