Skip to content

Latest commit

 

History

History
106 lines (81 loc) · 3.91 KB

001-Framework-Extensibility-Trait-Attributes.md

File metadata and controls

106 lines (81 loc) · 3.91 KB

RFC 001 - Framework Extensibility for Trait Attributes

  • Approved in principle
  • Under discussion
  • Implementation
  • Shipped

Summary

This details the MSTest V2 framework extensibility for attributes that are traits for a test.

Motivation

It is a requirement for teams to have trait attributes which are strongly typed over a test method as opposed to just a KeyValuePair<string, string>. This allows teams to have a standard across the team which is less error prone and more natural to specify. Users can also filter tests based on the values of these attributes.

Detailed Design

Requirements

  1. One should be able to have a strongly typed trait attribute on a test method.
  2. One should be able to filter in VS IDE based on this trait attribute.
  3. One should be able to see this information in Test Reporting.
  4. This extensibility should also guarantee that attributes in MSTest V1 which are brought into MSTest V2 with this extensibility model should be source compatible.

Proposed solution

The test framework currently has a TestProperty attribute which can be used to define custom traits as a KeyValuePair<string, string>. The definition of this attribute is as below:

[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public sealed class TestPropertyAttribute : Attribute
{
    /// <summary>
    /// Initializes a new instance of the <see cref="TestPropertyAttribute"/> class.
    /// </summary>
    /// <param name="name">
    /// The name.
    /// </param>
    /// <param name="value">
    /// The value.
    /// </param>
    public TestPropertyAttribute(string name, string value)
    {
        this.Name = name;
        this.Value = value;
    }

    /// <summary>
    /// Gets the name.
    /// </summary>
    public string Name { get; }

    /// <summary>
    /// Gets the value.
    /// </summary>
    public string Value { get; }
}

This TestProperty is also filled into the TestPlatform's TestCase object which makes it available for reporting in the various loggers that can be plugged into the TestPlatform.

To provide extension writers with the ability to have strongly typed attributes to achieve what TestProperty above achieves, the proposal is to make TestProperty a non-sealed class allowing classes to extend it like below:

[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]
public class CustomTraitPropertyAttribute : TestPropertyAttribute
{
    /// <summary>
    /// Initializes a new instance of the <see cref="CustomTraitPropertyAttribute"/> class.
    /// </summary>
    /// <param name="traitData">
    /// Data associated with trait property
    /// </param>
    public CustomTraitPropertyAttribute(int traitData)
        : base ("CustomTraitProperty", traitData.ToString())
    {
    }
}

And test methods would be decorated in a much more convenient form as below:

[TestMethod]
[CustomTraitProperty(234)]
public void TestMethod()
{
}

Users can then filter tests in VS IDE Test Explorer based on this Test Property. The query string that would filter the test above would look like

Trait:"CustomTraitProperty" Trait:"234"

This would provide extension writers with the same level of functionality that TestProperty already has with filtering and reporting plus the added ease of having a strongly typed trait.

Requirements from Test Platform

  1. Custom attributes should also show up in the trx logger. This is a bigger change since it might require changes in the trx schema.

Unresolved questions

  1. Should this attribute be opened up to be at a class and assembly level like TestCategory Attribute?
  2. Is filtering needed at the console level for these extension attributes? Currently the TestCaseFilter switch at the console level only supports - TestCategory, Priority, FullyQualifiedName, Name, ClassName. This is a finite list. In order to support filtering for extended attributes this list needs to be made dynamic.