|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System.Collections.Immutable;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.ReplaceDefaultLiteral;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.Diagnostics;
using Microsoft.CodeAnalysis.Test.Utilities;
using Xunit;
using Xunit.Abstractions;
namespace Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.ReplaceDefaultLiteral;
[Trait(Traits.Feature, Traits.Features.CodeActionsReplaceDefaultLiteral)]
public sealed class ReplaceDefaultLiteralTests(ITestOutputHelper logger) : AbstractCSharpDiagnosticProviderBasedUserDiagnosticTest_NoEditor(logger)
{
internal override (DiagnosticAnalyzer?, CodeFixProvider) CreateDiagnosticProviderAndFixer(Workspace workspace)
=> (null, new CSharpReplaceDefaultLiteralCodeFixProvider());
private static readonly ImmutableArray<LanguageVersion> s_csharp7_1above =
[LanguageVersion.CSharp7_1, LanguageVersion.Latest];
private static readonly ImmutableArray<LanguageVersion> s_csharp7below =
[
LanguageVersion.CSharp7,
LanguageVersion.CSharp6,
LanguageVersion.CSharp5,
LanguageVersion.CSharp4,
LanguageVersion.CSharp3,
LanguageVersion.CSharp2,
LanguageVersion.CSharp1,
];
private async Task TestWithLanguageVersionsAsync(string initialMarkup, string expectedMarkup, ImmutableArray<LanguageVersion> versions)
{
foreach (var version in versions)
{
await TestInRegularAndScriptAsync(initialMarkup, expectedMarkup,
parseOptions: CSharpParseOptions.Default.WithLanguageVersion(version));
}
}
private async Task TestMissingWithLanguageVersionsAsync(string initialMarkup, ImmutableArray<LanguageVersion> versions)
{
foreach (var version in versions)
{
await TestMissingInRegularAndScriptAsync(initialMarkup,
new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(version)));
}
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_Int()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]default: }
}
}
""",
"""
class C
{
void M()
{
switch (1) { case 0: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_Int_InParentheses()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case ([||]default): }
}
}
""",
"""
class C
{
void M()
{
switch (1) { case (0): }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_Int_NotInsideCast()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case (int)[||]default: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_Int_NotOnDefaultExpression()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]default(int): }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_Int_NotOnNumericLiteral()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]0: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_DateTime()
{
// Note that the default value of a struct type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (System.DateTime.Now) { case [||]default: }
}
}
""",
"""
class C
{
void M()
{
switch (System.DateTime.Now) { case default(System.DateTime): }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCaseSwitchLabel_TupleType()
{
// Note that the default value of a tuple type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch ((0, true)) { case [||]default: }
}
}
""",
"""
class C
{
void M()
{
switch ((0, true)) { case default((int, bool)): }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("value")]
[InlineData("null")]
[InlineData("default")]
[InlineData("() => { }")]
[InlineData("")]
public async Task TestCSharp7_1_InCaseSwitchLabel_NotForInvalidType(string expression)
{
await TestMissingWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
switch ({{expression}}) { case [||]default: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_Int()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]default when true: }
}
}
""",
"""
class C
{
void M()
{
switch (1) { case 0 when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_Int_InParentheses()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case ([||]default) when true: }
}
}
""",
"""
class C
{
void M()
{
switch (1) { case (0) when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_Int_NotInsideCast()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case (int)[||]default when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_Int_NotOnDefaultExpression()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]default(int) when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_Int_NotOnNumericLiteral()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (1) { case [||]0 when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_DateTime()
{
// Note that the default value of a struct type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch (System.DateTime.Now) { case [||]default when true: }
}
}
""",
"""
class C
{
void M()
{
switch (System.DateTime.Now) { case default(System.DateTime) when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_TupleType()
{
// Note that the default value of a tuple type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
switch ((0, true)) { case [||]default when true: }
}
}
""",
"""
class C
{
void M()
{
switch ((0, true)) { case default((int, bool)) when true: }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("value")]
[InlineData("null")]
[InlineData("default")]
[InlineData("() => { }")]
[InlineData("")]
public async Task TestCSharp7_1_InCasePatternSwitchLabel_NotForInvalidType(string expression)
{
await TestMissingWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
switch ({{expression}}) { case [||]default when true: }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is [||]default) { }
}
}
""",
"""
class C
{
void M()
{
if (true is false) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool_InParentheses()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is ([||]default)) { }
}
}
""",
"""
class C
{
void M()
{
if (true is (false)) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool_NotInsideCast()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is (bool)[||]default) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool_NotOnDefaultExpression()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is [||]default(bool)) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool_NotOnFalseLiteral()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is [||]false) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("int", "0")]
[InlineData("uint", "0U")]
[InlineData("byte", "0")]
[InlineData("sbyte", "0")]
[InlineData("short", "0")]
[InlineData("ushort", "0")]
[InlineData("long", "0L")]
[InlineData("ulong", "0UL")]
[InlineData("float", "0F")]
[InlineData("double", "0D")]
[InlineData("decimal", "0M")]
[InlineData("char", "'\\0'")]
[InlineData("string", "null")]
[InlineData("object", "null")]
public async Task TestCSharp7_1_InIsPattern_BuiltInType(string type, string expectedLiteral)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
void M({{type}} value)
{
if (value is [||]default) { }
}
}
""",
$$"""
class C
{
void M({{type}} value)
{
if (value is {{expectedLiteral}}) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_DateTime()
{
// Note that the default value of a struct type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (System.DateTime.Now is [||]default) { }
}
}
""",
"""
class C
{
void M()
{
if (System.DateTime.Now is default(System.DateTime)) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_TupleType()
{
// Note that the default value of a tuple type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if ((0, true) is [||]default) { }
}
}
""",
"""
class C
{
void M()
{
if ((0, true) is default((int, bool))) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("class Type { }")]
[InlineData("interface Type { }")]
[InlineData("delegate void Type();")]
public async Task TestCSharp7_1_InIsPattern_CustomReferenceType(string typeDeclaration)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
{{typeDeclaration}}
void M()
{
if (new Type() is [||]default) { }
}
}
""",
$$"""
class C
{
{{typeDeclaration}}
void M()
{
if (new Type() is null) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("enum Enum { }")]
[InlineData("enum Enum { None = 0 }")]
[InlineData("[Flags] enum Enum { None = 0 }")]
[InlineData("[System.Flags] enum Enum { None = 1 }")]
[InlineData("[System.Flags] enum Enum { None = 1, None = 0 }")]
[InlineData("[System.Flags] enum Enum { Some = 0 }")]
public async Task TestCSharp7_1_InIsPattern_CustomEnum_WithoutSpecialMember(string enumDeclaration)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
{{enumDeclaration}}
void M()
{
if (new Enum() is [||]default) { }
}
}
""",
$$"""
class C
{
{{enumDeclaration}}
void M()
{
if (new Enum() is 0) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("[System.Flags] enum Enum : int { None = 0 }")]
[InlineData("[System.Flags] enum Enum : uint { None = 0 }")]
[InlineData("[System.Flags] enum Enum : byte { None = 0 }")]
[InlineData("[System.Flags] enum Enum : sbyte { None = 0 }")]
[InlineData("[System.Flags] enum Enum : short { None = 0 }")]
[InlineData("[System.Flags] enum Enum : ushort { None = 0 }")]
[InlineData("[System.Flags] enum Enum : long { None = 0 }")]
[InlineData("[System.Flags] enum Enum : ulong { None = 0 }")]
[InlineData("[System.Flags] enum Enum { None = default }")]
[InlineData("[System.Flags] enum Enum { Some = 1, None = 0 }")]
[InlineData("[System.FlagsAttribute] enum Enum { None = 0, Some = 1 }")]
public async Task TestCSharp7_1_InIsPattern_CustomEnum_WithSpecialMember(string enumDeclaration)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
{{enumDeclaration}}
void M()
{
if (new Enum() is [||]default) { }
}
}
""",
$$"""
class C
{
{{enumDeclaration}}
void M()
{
if (new Enum() is Enum.None) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_CustomStruct()
{
// Note that the default value of a struct type is not a constant, so this code is incorrect.
await TestWithLanguageVersionsAsync(
"""
class C
{
struct Struct { }
void M()
{
if (new Struct() is [||]default) { }
}
}
""",
"""
class C
{
struct Struct { }
void M()
{
if (new Struct() is default(Struct)) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_AnonymousType()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (new { a = 0 } is [||]default) { }
}
}
""",
"""
class C
{
void M()
{
if (new { a = 0 } is null) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("class Container<T> { }")]
[InlineData("interface Container<T> { }")]
[InlineData("delegate void Container<T>();")]
public async Task TestCSharp7_1_InIsPattern_CustomReferenceTypeOfAnonymousType(string typeDeclaration)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
{{typeDeclaration}}
Container<T> ToContainer<T>(T value) => new Container<T>();
void M()
{
if (ToContainer(new { x = 0 }) is [||]default) { }
}
}
""",
$$"""
class C
{
{{typeDeclaration}}
Container<T> ToContainer<T>(T value) => new Container<T>();
void M()
{
if (ToContainer(new { x = 0 }) is null) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_NotForCustomStructOfAnonymousType()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
struct Container<T> { }
Container<T> ToContainer<T>(T value) => new Container<T>();
void M()
{
if (ToContainer(new { x = 0 }) is [||]default) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("System.Threading", "CancellationToken", "None")]
public async Task TestCSharp7_1_InIsPattern_SpecialTypeQualified1(string @namespace, string type, string member)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
if (default({{@namespace}}.{{type}}) is [||]default) { }
}
}
""",
$$"""
class C
{
void M()
{
if (default({{@namespace}}.{{type}}) is {{@namespace}}.{{type}}.{{member}}) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("System", "IntPtr", "Zero")]
[InlineData("System", "UIntPtr", "Zero")]
public async Task TestCSharp7_1_InIsPattern_SpecialTypeQualified2(string @namespace, string type, string member)
{
await TestWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
if (default({{@namespace}}.{{type}}) is [||]default) { }
}
}
""",
$$"""
class C
{
void M()
{
if (default({{@namespace}}.{{type}}) is {{@namespace}}.{{type}}.{{member}}) { }
}
}
""", [LanguageVersion.CSharp7_1, LanguageVersion.CSharp10, LanguageVersion.CSharp11, LanguageVersion.Latest]);
}
[Theory]
[InlineData("System.Threading", "CancellationToken", "None")]
public async Task TestCSharp7_1_InIsPattern_SpecialTypeUnqualifiedWithUsing1(string @namespace, string type, string member)
{
await TestWithLanguageVersionsAsync(
$$"""
using {{@namespace}};
class C
{
void M()
{
if (default({{type}}) is [||]default) { }
}
}
""",
$$"""
using {{@namespace}};
class C
{
void M()
{
if (default({{type}}) is {{type}}.{{member}}) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("System", "IntPtr", "Zero")]
[InlineData("System", "UIntPtr", "Zero")]
public async Task TestCSharp7_1_InIsPattern_SpecialTypeUnqualifiedWithUsing2(string @namespace, string type, string member)
{
await TestWithLanguageVersionsAsync(
$$"""
using {{@namespace}};
class C
{
void M()
{
if (default({{type}}) is [||]default) { }
}
}
""",
$$"""
using {{@namespace}};
class C
{
void M()
{
if (default({{type}}) is {{type}}.{{member}}) { }
}
}
""", [LanguageVersion.CSharp7_1, LanguageVersion.CSharp10, LanguageVersion.CSharp11, LanguageVersion.Latest]);
}
[Theory]
[InlineData("CancellationToken")]
[InlineData("IntPtr")]
[InlineData("UIntPtr")]
public async Task TestCSharp7_1_InIsPattern_NotForSpecialTypeUnqualifiedWithoutUsing(string type)
{
await TestMissingWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
if (default({{type}}) is [||]default) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_NotForInvalidType1()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
var value;
if (value is [||]default) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("value")]
[InlineData("null")]
[InlineData("default")]
[InlineData("")]
public async Task TestCSharp7_1_InIsPattern_NotForInvalidType2(string expression)
{
await TestMissingWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
var value = {{expression}};
if (value is [||]default) { }
}
}
""", s_csharp7_1above);
}
[Theory]
[InlineData("value")]
[InlineData("null")]
[InlineData("default")]
[InlineData("() => { }")]
[InlineData("")]
public async Task TestCSharp7_1_InIsPattern_NotForInvalidType3(string expression)
{
await TestMissingWithLanguageVersionsAsync(
$$"""
class C
{
void M()
{
if ({{expression}} is [||]default) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Lambda()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
var value = () => { };
if (value is [||]default) { }
}
}
""", [LanguageVersion.CSharp7_1]);
}
[Fact]
public async Task TestCSharpLatest_InIsPattern_Lambda()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
var value = () => { };
if (value is [||]default) { }
}
}
""",
"""
class C
{
void M()
{
var value = () => { };
if (value is null) { }
}
}
""", [LanguageVersion.Latest]);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_Bool_Trivia()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (true is
/*a*/ [||]default /*b*/) { }
}
}
""",
"""
class C
{
void M()
{
if (true is
/*a*/ false /*b*/) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_InIsPattern_DateTime_Trivia()
{
await TestWithLanguageVersionsAsync(
"""
class C
{
void M()
{
if (System.DateTime.Now is
/*a*/ [||]default /*b*/) { }
}
}
""",
"""
class C
{
void M()
{
if (System.DateTime.Now is
/*a*/ default(System.DateTime) /*b*/) { }
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_NotInsideExpression()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
int i = [||]default;
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7_1_NotInsideExpression_InvalidType()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
var v = [||]default;
}
}
""", s_csharp7_1above);
}
[Fact]
public async Task TestCSharp7Lower_NotInsideExpression()
{
await TestMissingWithLanguageVersionsAsync(
"""
class C
{
void M()
{
int i = [||]default;
}
}
""", s_csharp7below);
}
}
|