Report and update deprecated APIs
Last modified: 13 July 2022When evolving an API, you have to introduce breaking changes sooner or later. The traditional way of dealing with that is to mark the deprecated types and members with the [Obsolete]
attribute and explain how to migrate to the new API using the attribute's message.
JetBrains Rider provides a more elegant solution that allows API users to find and automatically convert the old API to a new one. As the API author, you need to mark the obsolete type or member with the [CodeTemplate] attribute from JetBrains.Annotations where you can specify a search pattern to match the old API and a replacement pattern for it. The attribute will act as a custom code inspection with the corresponding quick-fix and fixes in scope.
[CodeTemplateAttribute]
is recognized by all JetBrains products that analyze C#, for example, JetBrains ReSharper JetBrains Fleet and InspectCode Command-Line Tool.
Let's see how it works using an example:
public class MyAssert
{
// Deprecated API. Usages look like:
// MyAssert.IsTrue(args.Length > 0);
public static void IsTrue(bool condition)
{
if (!condition)
throw new Exception("Assertion failed");
}
// New API. Usages should look like:
// MyAssert.That(args.Length > 0, Is.True);
public static void That<T>(T value, Constraint<T> constraint)
{
// ...
}
}
public class Constraint<T> { }
class Is
{
public static Constraint<bool> True => null;
public static Constraint<bool> False => null;
}
Let's annotate the deprecated IsTrue()
method with the [CodeTemplate]
attribute:
[CodeTemplate(
searchTemplate: "$member$($expr$)",
Message = "The API is deprecated, use 'MyAssert.That' instead",
ReplaceTemplate = "MyAssert.That($expr$, Is.True)",
ReplaceMessage = "Convert to 'MyAssert.That'")]
public static void IsTrue(bool condition)
{
if (!condition)
throw new Exception("Assertion failed");
}
Now JetBrains Rider will report all usages of MyAssert.IsTrue()
and suggest a migration fix:
Suboptimal code
MyAssert.IsTrue(args.Length > 0);
After the quick-fix
MyAssert.That(args.Length > 0, Is.True);
tip
Even if you don't have access to the sources of a specific API, you can still annotate obsolete symbols in that API using the external annotations.