Jump to content
Xtreme .Net Talk

MSTest 3.4 is here with WinUI support and new analyzers!


Recommended Posts

Guest Amaury Levé
Posted

We are excited to announce the new release of MSTest, the popular testing framework for .NET. This release brings many enhancements and bug fixes to MSTest.Analyzers, new features and improvements for MSTest.Sdk, and adds support for WinUI applications to MSTest.Runner.

 

We’ve added [iCODE][Timeout][/iCODE] support to all fixture (initialize and cleanup) methods, and STA thread support for UI tests. Both were long-standing issues that were reported by our users and community. We have also simplified testing with Playwright and Aspire by removing project boilerplate.

 

Here are some of the highlights of this release:

 

[HEADING=1]MSTest.Analyzers[/HEADING]

 

MSTest ships with a set of Roslyn based code analyzers to help you write better tests. In this release we have added 9 new rules that cover common best practices and pitfalls. These rules ensure correct usage of attributes and assertions, help enforce design preferences, and more.

 

  • MSTEST0017: Assertion arguments should be passed in the correct order.
  • MSTEST0019: Prefer [iCODE]TestInitialize[/iCODE] over constructors. This rule suggests using [iCODE]TestInitialize[/iCODE] methods over class constructors to ensure a consistent declaration and initialization across all the test classes, because [iCODE]TestInitialize[/iCODE] method allows async initialization, which class constructors don’t allow.
  • MSTEST0020: Prefer constructors over [iCODE]TestInitialize[/iCODE] methods. This rule recommends using constructors over [iCODE]TestInitialize[/iCODE] methods for non-async initialization, because constructors allow for readonly fields and provide better compiler feedback
  • MSTEST0021: Prefer Dispose over [iCODE]TestCleanup[/iCODE] methods. This rule recommends using [iCODE]Dispose[/iCODE] or [iCODE]DisposeAsync[/iCODE] pattern over [iCODE]TestCleanup[/iCODE] methods to provide easier readability for developers not used to MSTest.
  • MSTEST0022: Prefer [iCODE]TestCleanup[/iCODE] methods over [iCODE]Dispose[/iCODE]. This rule is the opposite of [iCODE]MSTEST0021[/iCODE] and suggests using [iCODE]TestCleanup[/iCODE] methods for the cleanup phase, because it allows for the async pattern even in older versions of .NET. This ensures a consistent declaration no matter what the targeted framework is.
  • MSTEST0023: Do not negate boolean assertions. This rule warns against negating boolean assertions, which can lead to less readable tests and make it harder to understand the test’s intent.
  • MSTEST0024: Do not store [iCODE]TestContext[/iCODE] in static members. The [iCODE]TestContext[/iCODE] instance passed to [iCODE]AssemblyInitialize[/iCODE] and [iCODE]ClassInitialize[/iCODE] method is tailored for these methods only and won’t be updated based on the current stage of the test execution. This means that the instance being stored will not provide any consistent state so it is better not to reuse it outside the scope of its method.
  • MSTEST0025: Use [iCODE]Assert.Fail[/iCODE] instead of an always-failing assert. This rule suggests using [iCODE]Assert.Fail[/iCODE] to explicitly fail a test, which is clearer and more direct than using an assertion that is designed to always fail.

 

You can find the full list of rules and their descriptions in the MSTest code analysis documentation. We have also fixed 3 rules that had false positives or incorrect messages.

 

If you have any ideas for a rule or code-fix that would make your tests easier, please feel free to open a feature request ticket on our microsoft/testfx repository.

 

[HEADING=1]MSTest[/HEADING]

 

We have fixed the long-standing request to add support for STA threads in MSTest for VSTest and MSTest.Runner for all the supported target frameworks, allowing for easier testing of UI elements. To enable STA thread support, set [iCODE]<ExecutionThreadApartmentState>STA</ExecutionThreadApartmentState>[/iCODE] in your runsettings. Full runsettings below:

 

<?xml version="1.0" encoding="utf-8" ?>
<RunSettings>
   <RunConfiguration>
       <ExecutionThreadApartmentState>STA</ExecutionThreadApartmentState>
   </RunConfiguration>
</RunSettings>

 

We also plan to improve the experience by allowing to have only some specific tests or test classes run in STA thread mode, please upvote and track Introduce `[sTATestMethod]` and `[sTATestClass]` · Issue #2688 · microsoft/testfx.

 

Additionally, we have introduced the ability to define timeouts on fixture methods (AssemblyInitialize, AssemblyCleanup, ClassInitialize, ClassCleanup, TestInitialize and TestCleanup), giving you more control over the execution of your tests. In this case I want my [iCODE][ClassCleanup][/iCODE] to time out after 1 second:

 

517bd606-36f8-4096-9e30-e8b88580f46c.png.0c393639577e74f668f4c27cd1d903a0.png

 

Alternatively, timeouts can be specified through runsettings, to be applied as default value. For example, the following runsettings will add a default timeout of 1 second for all ClassCleanup methods:

 

<RunSettings>
 <MSTest>
   <ClassCleanupTimeout>1000</ClassCleanupTimeout>
 </MSTest>
</RunSettings>

 

Finally, we have fixed message and stack trace for exceptions thrown in fixture methods.

 

namespace TestProject80
{
   [TestClass]
   public class UnitTest1
   {
       [AssemblyInitialize]
       public static void Init(TestContext context)
       {
           // This will throw in ctor.
           InitHelper.A();
       }

       [TestMethod]
       public void TestMethod1()
       {
       }
   }

   public static class InitHelper
   {
       static InitHelper()
       {
           throw new Exception("AAAA");
       }

       public static void A() {  }
   }
}

 

Was producing the following message

 

 Assembly Initialization method TestProject80.UnitTest1.Init threw exception. System.TypeInitializationException: The type initializer for 'TestProject80.InitHelper' threw an exception.. Aborting test execution.
Stack Trace:
   at TestProject80.InitHelper..cctor() in C:\src\temp\ConsoleApp4\TestProject1\UnitTest1.cs:line 29
--- End of inner exception stack trace ---
   at TestProject80.InitHelper.A() in C:\src\temp\ConsoleApp4\TestProject1\UnitTest1.cs:line 32
  at TestProject80.UnitTest1.Init(TestContext context) in C:\src\temp\ConsoleApp4\TestProject1\UnitTest1.cs:line 17
  at System.RuntimeMethodHandle.InvokeMethod(Object target, Void** arguments, Signature sig, Boolean isConstructor)
  at System.Reflection.MethodBaseInvoker.InvokeDirectByRefWithFewArgs(Object obj, Span`1 copyOfArgs, BindingFlags invokeAttr)

 

And is now producing

 

  Message: 
Assembly Initialization method TestProject80.UnitTest1.Init threw exception. System.Exception: AAAA. Aborting test execution.
Stack Trace:
   at TestProject80.InitHelper..cctor() in C:\src\temp\ConsoleApp4\TestProject1\UnitTest1.cs:line 29

[HEADING=1]MSTest.Sdk[/HEADING]

 

It’s great to see your positive response to MSTest.Sdk. In case you missed our announcement, please read our Introducing MSTest SDK blog post.

 

We have been carefully listening to your feedback and we are now adding global using of MSTest namespace to simplify further your tests. So starting with MSTest 3.4, you no longer need to explicitly add [iCODE]using Microsoft.VisualStudio.TestTools.UnitTesting[/iCODE] in your files. Note that [iCODE]MSTest.Sdk[/iCODE] does respect your implicit usings preferences.

 

On top of that, we have enhanced our sample codes of how to use the SDK.

 

Lastly, we have made Playwright and Aspire testing easier.

 

[HEADING=2]Playwright[/HEADING]

 

Simply add [iCODE]<EnablePlaywright>true</EnablePlaywright>[/iCODE] to your project to remove all the project boilerplate.

 

Playwright project without SDK:

 

<Project Sdk="Microsoft.NET.Sdk">

 <PropertyGroup>
   <TargetFramework>net8.0</TargetFramework>
   <ImplicitUsings>enable</ImplicitUsings>
   <Nullable>enable</Nullable>

   <IsPackable>false</IsPackable>
   <IsTestProject>true</IsTestProject>
 </PropertyGroup>

 <ItemGroup>
   <PackageReference Include="Microsoft.NET.Test.Sdk" Version="17.10.0" />
   <PackageReference Include="Microsoft.Playwright.MSTest" Version="1.44.0" />
   <PackageReference Include="MSTest.Analyzers" Version="3.4.1" />    
   <PackageReference Include="MSTest.TestAdapter" Version="3.4.1" />
   <PackageReference Include="MSTest.TestFramework" Version="3.4.1" />
 </ItemGroup>

 <ItemGroup>
   <Using Include="Microsoft.Playwright.MSTest" />
   <Using Include="Microsoft.VisualStudio.TestTools.UnitTesting" />
   <Using Include="System.Text.RegularExpressions" />
   <Using Include="System.Threading.Tasks" />
 </ItemGroup>

</Project>

 

Playwright project with SDK:

 

<Project Sdk="MSTest.Sdk/3.4.1">

 <PropertyGroup>
   <TargetFramework>net8.0</TargetFramework>
   <ImplicitUsings>enable</ImplicitUsings>
   <Nullable>enable</Nullable>

   <IsPackable>false</IsPackable>
   <EnablePlaywright>true</EnablePlaywright>
 </PropertyGroup>

 <ItemGroup>
   <Using Include="System.Text.RegularExpressions" />
   <Using Include="System.Threading.Tasks" />
 </ItemGroup>

</Project>

[HEADING=2]Aspire[/HEADING]

 

Simply add [iCODE]<EnableAspireTesting>true</EnableAspireTesting>[/iCODE] to your project to remove all the project boilerplate.

 

Aspire test project without SDK:

 

<Project Sdk="Microsoft.NET.Sdk">

 <PropertyGroup>
   <TargetFramework>net8.0</TargetFramework>
   <ImplicitUsings>enable</ImplicitUsings>
   <Nullable>enable</Nullable>
   <IsPackable>false</IsPackable>
   <IsTestProject>true</IsTestProject>
   <EnableMSTestRunner>true</EnableMSTestRunner>
   <OutputType>Exe</OutputType>
 </PropertyGroup>

 <ItemGroup>
   <PackageReference Include="Aspire.Hosting.Testing" Version="8.0.0-preview.6.24214.1" />
   <PackageReference Include="MSTest" Version="3.4.1" />
 </ItemGroup>

 <ItemGroup>
   <Using Include="Aspire.Hosting.Testing" />
   <Using Include="Microsoft.VisualStudio.TestTools.UnitTesting" />
 </ItemGroup>

</Project>

 

Aspire test project with SDK:

 

<Project Sdk="MSTest.Sdk/3.4.1">

 <PropertyGroup>
   <TargetFramework>net8.0</TargetFramework>
   <ImplicitUsings>enable</ImplicitUsings>
   <Nullable>enable</Nullable>

   <IsPackable>false</IsPackable>
   <EnableAspireTesting>true</EnableAspireTesting>
 </PropertyGroup>

</Project>

[HEADING=1]MSTest.Runner support for WinUI tests[/HEADING]

 

You asked for it, so we made it possible to run WinUI tests with MSTest runner. Check-out our project sample and we are working to simplify testing un-packaged applications.

 

We have as well enhanced the runner’s performance by using the built-in [iCODE]System.Text.Json[/iCODE] for .NET instead of Jsonite and by caching command line options.

 

[HEADING=1]What’s next?[/HEADING]

 

We hope you enjoy this new release of MSTest and find it useful. We would love to hear your feedback and suggestions on how we can make MSTest better browse or create an issue in our repository.

 

A huge thank you to all the amazing contributors who helped make this release even better. Your hard work and awesome ideas are what make MSTest great. Thanks for being a part of this journey!

 

Thank you for using MSTest and happy testing!

 

The post MSTest 3.4 is here with WinUI support and new 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...