|
// 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.Threading.Tasks;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.UsePatternMatching;
using Microsoft.CodeAnalysis.Diagnostics;
using Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.Diagnostics;
using Microsoft.CodeAnalysis.Test.Utilities;
using Roslyn.Test.Utilities;
using Xunit;
using Xunit.Abstractions;
namespace Microsoft.CodeAnalysis.Editor.CSharp.UnitTests.UsePatternMatching;
[Trait(Traits.Feature, Traits.Features.CodeActionsInlineTypeCheck)]
public partial class CSharpAsAndNullCheckTests(ITestOutputHelper logger)
: AbstractCSharpDiagnosticProviderBasedUserDiagnosticTest_NoEditor(logger)
{
internal override (DiagnosticAnalyzer, CodeFixProvider) CreateDiagnosticProviderAndFixer(Workspace workspace)
=> (new CSharpAsAndNullCheckDiagnosticAnalyzer(), new CSharpAsAndNullCheckCodeFixProvider());
[Theory]
[InlineData("x != null", "o is string x")]
[InlineData("null != x", "o is string x")]
[InlineData("(object)x != null", "o is string x")]
[InlineData("null != (object)x", "o is string x")]
[InlineData("x is object", "o is string x")]
[InlineData("x == null", "!(o is string x)")]
[InlineData("null == x", "!(o is string x)")]
[InlineData("(object)x == null", "!(o is string x)")]
[InlineData("null == (object)x", "!(o is string x)")]
[InlineData("x is null", "!(o is string x)")]
[InlineData("(x = o as string) != null", "o is string x")]
[InlineData("null != (x = o as string)", "o is string x")]
[InlineData("(x = o as string) is object", "o is string x")]
[InlineData("(x = o as string) == null", "!(o is string x)")]
[InlineData("null == (x = o as string)", "!(o is string x)")]
[InlineData("(x = o as string) is null", "!(o is string x)")]
[InlineData("x == null", "o is not string x", LanguageVersion.CSharp9)]
public async Task InlineTypeCheck1(string input, string output, LanguageVersion version = LanguageVersion.CSharp8)
{
await TestStatement($"if ({input}) {{ }}", $"if ({output}) {{ }}", version);
await TestStatement($"var y = {input};", $"var y = {output};", version);
await TestStatement($"return {input};", $"return {output};", version);
}
[Theory]
[InlineData("(x = o as string) != null", "o is string x")]
[InlineData("null != (x = o as string)", "o is string x")]
[InlineData("(x = o as string) is object", "o is string x")]
[InlineData("(x = o as string) == null", "!(o is string x)")]
[InlineData("null == (x = o as string)", "!(o is string x)")]
public async Task InlineTypeCheck2(string input, string output)
=> await TestStatement($"while ({input}) {{ }}", $"while ({output}) {{ }}");
private async Task TestStatement(string input, string output, LanguageVersion version = LanguageVersion.CSharp8)
{
await TestInRegularAndScript1Async(
$$"""
class C
{
void M(object o)
{
[|var|] x = o as string;
{{input}}
}
}
""",
$$"""
class C
{
void M(object o)
{
{{output}}
}
}
""", new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(version)));
}
[Fact]
public async Task TestMissingInCSharp6()
{
await TestMissingAsync(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null)
{
}
}
}
""", new TestParameters(parseOptions: CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp6)));
}
[Fact]
public async Task TestMissingInWrongName()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|var|] y = o as string;
if (x != null)
{
}
}
}
""");
}
[Fact]
public async Task TestInSwitchSection()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
switch (o)
{
default:
[|var|] x = o as string;
if (x != null)
{
}
}
}
}
""",
"""
class C
{
void M()
{
switch (o)
{
default:
if (o is string x)
{
}
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/33345")]
public async Task TestRemoveNewLinesInSwitchStatement()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
switch (o)
{
default:
[|var|] x = o as string;
//a comment
if (x != null)
{
}
}
}
}
""",
"""
class C
{
void M()
{
switch (o)
{
default:
//a comment
if (o is string x)
{
}
}
}
}
""");
}
[Fact]
public async Task TestMissingOnNonDeclaration()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|y|] = o as string;
if (x != null)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/25237")]
public async Task TestMissingOnReturnStatement()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|return;|]
}
}
""");
}
[Fact]
public async Task TestMissingOnIsExpression()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|var|] x = o is string;
if (x != null)
{
}
}
}
""");
}
[Fact]
public async Task InlineTypeCheckComplexExpression1()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = (o ? z : w) as string;
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
if ((o ? z : w) is string x)
{
}
}
}
""");
}
[Fact]
public async Task TestInlineTypeCheckWithElse()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (null != x)
{
}
else
{
}
}
}
""",
"""
class C
{
void M()
{
if (o is string x)
{
}
else
{
}
}
}
""");
}
[Fact]
public async Task TestComments1()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
// prefix comment
[|var|] x = o as string;
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
// prefix comment
if (o is string x)
{
}
}
}
""");
}
[Fact]
public async Task TestComments2()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string; // suffix comment
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
// suffix comment
if (o is string x)
{
}
}
}
""");
}
[Fact]
public async Task TestComments3()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
// prefix comment
[|var|] x = o as string; // suffix comment
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
// prefix comment
// suffix comment
if (o is string x)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/33345")]
public async Task TestRemoveNewLines()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
//suffix comment
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
//suffix comment
if (o is string x)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/33345")]
public async Task TestRemoveNewLinesWhereBlankLineIsNotEmpty()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
//suffix comment
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
//suffix comment
if (o is string x)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/33345")]
public async Task TestRemoveNewLines2()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
int a = 0;
[|var|] x = o as string;
//suffix comment
if (x != null)
{
}
}
}
""",
"""
class C
{
void M()
{
int a = 0;
//suffix comment
if (o is string x)
{
}
}
}
""");
}
[Fact]
public async Task InlineTypeCheckComplexCondition1()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null ? 0 : 1)
{
}
}
}
""",
"""
class C
{
void M()
{
if (o is string x ? 0 : 1)
{
}
}
}
""");
}
[Fact]
public async Task InlineTypeCheckComplexCondition2()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
if ((x != null))
{
}
}
}
""",
"""
class C
{
void M()
{
if ((o is string x))
{
}
}
}
""");
}
[Fact]
public async Task InlineTypeCheckComplexCondition3()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
}
}
""",
"""
class C
{
void M()
{
if (o is string x && x.Length > 0)
{
}
}
}
""");
}
[Fact]
public async Task TestDefiniteAssignment1()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
else if (x != null)
{
}
}
}
""");
}
[Fact]
public async Task TestDefiniteAssignment2()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
Console.WriteLine(x);
}
}
""");
}
[Fact]
public async Task TestDefiniteAssignment3()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
x = null;
Console.WriteLine(x);
}
}
""",
"""
class C
{
void M()
{
if (o is string x && x.Length > 0)
{
}
x = null;
Console.WriteLine(x);
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21097")]
public async Task TestDefiniteAssignment4()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|var|] s = o as string;
if (s != null)
{
}
else
{
if (o is int?)
s = null;
s.ToString();
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/24286")]
public async Task TestDefiniteAssignment5()
{
await TestMissingInRegularAndScriptAsync(
"""
public class Test
{
public void TestIt(object o1, object o2)
{
[|var|] test = o1 as Test;
if (test != null || o2 != null)
{
var o3 = test ?? o2;
}
}
}
""");
}
[Fact]
public async Task TestDefiniteAssignment6()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
string Use(string x) => x;
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
Console.WriteLine(x = Use(x));
}
}
""");
}
[Fact]
public async Task TestDefiniteAssignment7()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
[|var|] x = o as string;
if (x != null && x.Length > 0)
{
}
Console.WriteLine(x);
x = "writeAfter";
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/28821")]
public async Task TestDefiniteAssignment8()
{
await TestMissingInRegularAndScriptAsync(
"""
class Program
{
static void Goo(System.Activator bar)
{
}
static void Main(string[] args)
{
var a = new object();
[|var|] b = a as System.Activator;
if ((b == null) && false)
{
}
else
{
Goo(b);
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/28866")]
public async Task TestWrittenExpressionBeforeNullCheck()
{
await TestMissingInRegularAndScriptAsync(
"""
class Goo
{
object Data { get; set; }
void DoGoo()
{
[|var|] oldData = this.Data as string;
Data = null;
if (oldData != null)
{
// Do something
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/15957")]
public async Task TestTrivia1()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object y)
{
if (y != null)
{
}
[|var|] x = o as string;
if (x != null)
{
}
}
}
""",
"""
class C
{
void M(object y)
{
if (y != null)
{
}
if (o is string x)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/17129")]
public async Task TestTrivia2()
{
await TestInRegularAndScript1Async(
"""
using System;
namespace N
{
class Program
{
public static void Main()
{
object o = null;
int i = 0;
[|var|] s = o as string;
if (s != null && i == 0 && i == 1 &&
i == 2 && i == 3 &&
i == 4 && i == 5)
{
Console.WriteLine();
}
}
}
}
""",
"""
using System;
namespace N
{
class Program
{
public static void Main()
{
object o = null;
int i = 0;
if (o is string s && i == 0 && i == 1 &&
i == 2 && i == 3 &&
i == 4 && i == 5)
{
Console.WriteLine();
}
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/17122")]
public async Task TestMissingOnNullableType()
{
await TestMissingInRegularAndScriptAsync(
"""
using System;
namespace N
{
class Program
{
public static void Main()
{
object o = null;
[|var|] i = o as int?;
if (i != null)
Console.WriteLine(i);
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/18053")]
public async Task TestMissingWhenTypesDoNotMatch()
{
await TestMissingInRegularAndScriptAsync(
"""
class SyntaxNode
{
public SyntaxNode Parent;
}
class BaseParameterListSyntax : SyntaxNode
{
}
class ParameterSyntax : SyntaxNode
{
}
public static class C
{
static void M(ParameterSyntax parameter)
{
[|SyntaxNode|] parent = parameter.Parent as BaseParameterListSyntax;
if (parent != null)
{
parent = parent.Parent;
}
}
}
""");
}
[Fact]
public async Task TestMissingOnWhileNoInline()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|string|] x = o as string;
while (x != null)
{
}
}
}
""");
}
[Fact]
public async Task TestWhileDefiniteAssignment1()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|string|] x;
while ((x = o as string) != null)
{
}
var readAfterWhile = x;
}
}
""");
}
[Fact]
public async Task TestWhileDefiniteAssignment2()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|string|] x;
while ((x = o as string) != null)
{
}
x = "writeAfterWhile";
}
}
""");
}
[Fact]
public async Task TestWhileDefiniteAssignment3()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|string|] x;
x = "writeBeforeWhile";
while ((x = o as string) != null)
{
}
}
}
""");
}
[Fact]
public async Task TestWhileDefiniteAssignment4()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|string|] x = null;
var readBeforeWhile = x;
while ((x = o as string) != null)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/23504")]
public async Task DoNotChangeOriginalFormatting1()
{
await TestInRegularAndScript1Async(
"""
class Program
{
static void Main(string[] args)
{
object obj = "test";
[|var|] str = obj as string;
var title = str != null
? str
: ";
}
}
""",
"""
class Program
{
static void Main(string[] args)
{
object obj = "test";
var title = obj is string str
? str
: ";
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/23504")]
public async Task DoNotChangeOriginalFormatting2()
{
await TestInRegularAndScript1Async(
"""
class Program
{
static void Main(string[] args)
{
object obj = "test";
[|var|] str = obj as string;
var title = str != null ? str : ";
}
}
""",
"""
class Program
{
static void Main(string[] args)
{
object obj = "test";
var title = obj is string str ? str : ";
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21172")]
public async Task TestMissingWithDynamic()
{
await TestMissingAsync(
"""
class C
{
void M(object o)
{
[|var|] x = o as dynamic;
if (x != null)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/21551")]
public async Task TestOverloadedUserOperator()
{
await TestMissingAsync(
"""
class C
{
public static void Main()
{
object o = new C();
[|var|] c = o as C;
if (c != null)
System.Console.WriteLine();
}
public static bool operator ==(C c1, C c2) => false;
public static bool operator !=(C c1, C c2) => false;
}
""");
}
[Fact]
public async Task TestNegativeDefiniteAssignment1()
{
await TestInRegularAndScript1Async(
"""
class C
{
string M(object o)
{
[|var|] x = o as string;
if (x == null) return null;
return x;
}
}
""",
"""
class C
{
string M(object o)
{
if (!(o is string x)) return null;
return x;
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestNegativeDefiniteAssignment2()
{
await TestInRegularAndScript1Async(
"""
class C
{
string M(object o, bool b)
{
[|var|] x = o as string;
if (((object)x == null) || b)
{
return null;
}
else
{
return x;
}
}
}
""",
"""
class C
{
string M(object o, bool b)
{
if ((!(o is string x)) || b)
{
return null;
}
else
{
return x;
}
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/25993")]
public async Task TestEmbeddedStatement1()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
var fe = e as C;
[|var|] ae = e as C;
if (fe != null)
{
M(fe); // fe is used
}
else if (ae != null)
{
M(ae); // ae is used
}
}
}
""",
"""
class C
{
void M(object e)
{
var fe = e as C;
if (fe != null)
{
M(fe); // fe is used
}
else if (e is C ae)
{
M(ae); // ae is used
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/25993")]
public async Task TestEmbeddedStatement2()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
var fe = e as C;
[|var|] ae = e as C;
if (fe != null)
{
M(ae); // ae is used
}
else if (ae != null)
{
M(ae); // ae is used
}
}
}
""");
}
[Fact]
public async Task TestUseBeforeDeclaration()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
{
{
var x1 = c;
if (c != null)
{
}
}
}
}
}
""");
}
[Fact]
public async Task TestPossiblyUnassigned()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
{
{
if (c != null)
{
}
var x2 = c;
}
// out of scope
var x3 = c;
}
}
}
""");
}
[Fact]
public async Task TestOutOfScope()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
{
{
if (c != null)
{
}
}
// out of scope
var x3 = c;
}
}
}
""");
}
[Fact]
public async Task TestDeclarationOnOuterBlock()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
{
{
if (c != null)
{
}
}
}
}
}
""",
"""
class C
{
void M(object e)
{
{
{
if (e is C c)
{
}
}
}
}
}
""");
}
[Fact]
public async Task TestConditionalExpression()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
M(c != null ? c : null);
}
}
""",
"""
class C
{
void M(object e)
{
M(e is C c ? c : null);
}
}
""");
}
[Fact]
public async Task TestConditionalExpression_OppositeBranch()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
M(c != null ? c : null, c);
}
}
""");
}
[Fact]
public async Task TestForStatement_NoInlineTypeCheck()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|var|] c = e as C;
for (;(c)!=null;) {}
}
}
""");
}
[Fact]
public async Task TestForStatement_InlineTypeCheck()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[|C|] c;
for (; !((c = e as C)==null);) { }
}
}
""",
"""
class C
{
void M(object e)
{
for (; !(!(e is C c));) { }
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestForStatement_InScope()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[|C|] c = null;
for (; !((c = e as C)==null);)
{
M(c);
}
}
}
""",
"""
class C
{
void M(object e)
{
for (; !(!(e is C c));)
{
M(c);
}
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestForStatement_NotAssignedBeforeAccess()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|C|] c = null;
for (; ((c = e as C)==null);)
{
if (b) c = null;
M(c);
}
}
}
""");
}
[Fact]
public async Task TestForStatement_AssignedBeforeAccess()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e, bool b)
{
[|C|] c = null;
for (; (c = e as C)==null;)
{
if (b) c = null;
else c = null;
M(c);
}
}
}
""",
"""
class C
{
void M(object e, bool b)
{
for (; !(e is C c);)
{
if (b) c = null;
else c = null;
M(c);
}
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestForStatement_MultipleDeclarators()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[|C|] c = null, x = null;
for (; !((c = e as C)==null);)
{
M(c);
}
}
}
""",
"""
class C
{
void M(object e)
{
C x = null;
for (; !(!(e is C c));)
{
M(c);
}
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestForStatement_UseBeforeDeclaration()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|C|] c = null, x = c;
for (; !((c = e as C)==null);)
{
M(c);
}
}
}
""");
}
[Fact]
public async Task TestForStatement_Initializer()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object e)
{
[|C|] c;
for (var i = !((c = e as C)==null); i != null; )
{
M(c);
}
}
}
""");
}
[Fact]
public async Task TestLocalFunction()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[||]var c = e as C;
C F() => c == null ? null : c;
}
}
""",
"""
class C
{
void M(object e)
{
C F() => !(e is C c) ? null : c;
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestLocalFunction_UseOutOfScope()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
C M(object e)
{
[||]var c = e as C;
C F() => c == null ? null : c;
return c;
}
}
""");
}
[Fact]
public async Task TestExpressionLambda()
{
await TestInRegularAndScript1Async(
"""
class C
{
void M(object e)
{
[||]var c = e as C;
System.Func<C> f = () => c == null ? null : c;
}
}
""",
"""
class C
{
void M(object e)
{
System.Func<C> f = () => !(e is C c) ? null : c;
}
}
""", parameters: new TestParameters(CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion.CSharp8)));
}
[Fact]
public async Task TestExpressionLambda_UseOutOfScope()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
C M(object e)
{
[||]var c = e as C;
System.Func<C> f = () => c == null ? null : c;
return c;
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/31388")]
public async Task TestUseBetweenAssignmentAndIfCondition()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M(object o)
{
[|var|] c = o as C;
M2(c != null);
if (c == null)
{
return;
}
}
void M2(bool b) { }
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/40007")]
public async Task TestSpaceAfterGenericType()
{
await TestInRegularAndScript1Async(
"""
#nullable enable
using System.Collections.Generic;
class Program
{
static void Goo<TKey, TValue>(object items)
{
[|var|] itemsAsDictionary = items as IDictionary<TKey, TValue>;
SortedDictionary<TKey, TValue>? dictionary = null;
if (itemsAsDictionary != null)
{
dictionary = new SortedDictionary<TKey, TValue>();
}
return dictionary;
}
}
""",
"""
#nullable enable
using System.Collections.Generic;
class Program
{
static void Goo<TKey, TValue>(object items)
{
SortedDictionary<TKey, TValue>? dictionary = null;
if (items is IDictionary<TKey, TValue> itemsAsDictionary)
{
dictionary = new SortedDictionary<TKey, TValue>();
}
return dictionary;
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/45596")]
public async Task TestMissingInUsingDeclaration()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
using [|var|] x = o as IDisposable;
if (x != null)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/45596")]
public async Task TestMissingInUsingStatement()
{
await TestMissingInRegularAndScriptAsync(
"""
class C
{
void M()
{
using ([|var|] x = o as IDisposable)
{
if (x != null)
{
}
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/37398")]
public async Task TestPrecedingDirectiveTrivia()
{
await TestInRegularAndScript1Async(
"""
class C
{
public static void M(object o)
{
#if DEBUG
Console.WriteLine("in debug");
#endif
[|string|] s = o as string;
if (s != null)
{
}
}
}
""",
"""
class C
{
public static void M(object o)
{
#if DEBUG
Console.WriteLine("in debug");
#endif
if (o is string s)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/40006")]
public async Task TestArrayOfNullables()
{
await TestInRegularAndScript1Async(
"""
#nullable enable
class Program
{
static void Set(object obj, object? item)
{
[|object?[]?|] arr = obj as object[];
if (arr != null)
{
arr[0] = item;
}
}
}
""",
"""
#nullable enable
class Program
{
static void Set(object obj, object? item)
{
if (obj is object?[] arr)
{
arr[0] = item;
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/55782")]
public async Task TestLocalReferencedAcrossScopes1()
{
var code = """
using System.Transactions;
class BaseObject { }
class ObjectFactory
{
internal static BaseObject CreateObject(int x)
{
throw new NotImplementedException();
}
}
struct Repro
{
static int Main(string[] args)
{
int x = 0;
[|BaseObject|] obj;
var tso = new TransactionOptions { IsolationLevel = IsolationLevel.RepeatableRead };
using (var trans = new TransactionScope(TransactionScopeOption.Required, tso))
{
try
{
if ((obj = ObjectFactory.CreateObject(x) as BaseObject) == null)
{
return -1;
}
// uses of obj in the transaction
}
catch (TransactionAbortedException)
{
return -1;
}
}
// local used here.
Console.WriteLine(obj);
return 0;
}
}
""";
await TestMissingInRegularAndScriptAsync(code);
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/55782")]
public async Task TestLocalReferencedAcrossScopes2()
{
await TestInRegularAndScript1Async("""
using System.Transactions;
class BaseObject { }
class ObjectFactory
{
internal static BaseObject CreateObject(int x)
{
throw new NotImplementedException();
}
}
struct Repro
{
static int Main(string[] args)
{
int x = 0;
[|BaseObject|] obj;
var tso = new TransactionOptions { IsolationLevel = IsolationLevel.RepeatableRead };
using (var trans = new TransactionScope(TransactionScopeOption.Required, tso))
{
try
{
if ((obj = ObjectFactory.CreateObject(x) as BaseObject) == null)
{
return -1;
}
// uses of obj in the transaction
}
catch (TransactionAbortedException)
{
return -1;
}
}
// not used
return 0;
}
}
""",
"""
using System.Transactions;
class BaseObject { }
class ObjectFactory
{
internal static BaseObject CreateObject(int x)
{
throw new NotImplementedException();
}
}
struct Repro
{
static int Main(string[] args)
{
int x = 0;
var tso = new TransactionOptions { IsolationLevel = IsolationLevel.RepeatableRead };
using (var trans = new TransactionScope(TransactionScopeOption.Required, tso))
{
try
{
if (ObjectFactory.CreateObject(x) is not BaseObject obj)
{
return -1;
}
// uses of obj in the transaction
}
catch (TransactionAbortedException)
{
return -1;
}
}
// not used
return 0;
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/37875")]
public async Task TestNullableWhenWrittenTo1()
{
await TestInRegularAndScript1Async("""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
[|string?|] s = o1 as string;
if (s == null)
{
}
}
}
""",
"""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
if (o1 is not string s)
{
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/37875")]
public async Task TestNullableWhenWrittenTo2()
{
await TestInRegularAndScript1Async("""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
[|string?|] s = o1 as string;
if (s == null)
{
s = "";
}
}
}
""",
"""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
if (o1 is not string s)
{
s = "";
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/37875")]
public async Task TestNullableWhenWrittenTo3()
{
await TestMissingInRegularAndScriptAsync("""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
[|string?|] s = o1 as string;
if (s == null)
{
s = o2 as string;
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/37875")]
public async Task TestNullableWhenWrittenTo4()
{
await TestMissingInRegularAndScriptAsync("""
#nullable enable
using System;
class Program
{
static void Goo(object o1, object o2)
{
[|string?|] s = o1 as string;
if (s == null)
{
s = null;
}
}
}
""");
}
[Fact, WorkItem("https://github.com/dotnet/roslyn/issues/39600")]
public async Task TestNotWithInterveningMutation()
{
await TestMissingInRegularAndScriptAsync("""
using System;
class Program
{
void Goo(object[] values, int index)
{
[|var|] v1 = values[index++] as string;
index++;
if (v1 != null)
{
Console.WriteLine(v1);
}
}
}
""");
}
}
|