@@ -46,26 +46,43 @@ public sealed override void Initialize(AnalysisContext context)
46
46
context . EnableConcurrentExecution ( ) ;
47
47
context . ConfigureGeneratedCodeAnalysis ( GeneratedCodeAnalysisFlags . None ) ;
48
48
49
- context . RegisterOperationAction (
50
- operationAnalysisContext => AnalyzeInvocationExpression ( ( IInvocationOperation ) operationAnalysisContext . Operation , operationAnalysisContext . ReportDiagnostic ) ,
51
- OperationKind . Invocation ) ;
52
-
53
- context . RegisterOperationAction (
54
- operationAnalysisContext => AnalyzeBinaryExpression ( ( IBinaryOperation ) operationAnalysisContext . Operation , operationAnalysisContext . ReportDiagnostic ) ,
55
- OperationKind . BinaryOperator ) ;
49
+ context . RegisterCompilationStartAction ( context =>
50
+ {
51
+ var linqExpressionType = context . Compilation . GetOrCreateTypeByMetadataName ( WellKnownTypeNames . SystemLinqExpressionsExpression1 ) ;
52
+
53
+ context . RegisterOperationAction (
54
+ operationAnalysisContext => AnalyzeInvocationExpression (
55
+ ( IInvocationOperation ) operationAnalysisContext . Operation , linqExpressionType ,
56
+ operationAnalysisContext . ReportDiagnostic ) ,
57
+ OperationKind . Invocation ) ;
58
+
59
+ context . RegisterOperationAction (
60
+ operationAnalysisContext => AnalyzeBinaryExpression (
61
+ ( IBinaryOperation ) operationAnalysisContext . Operation , linqExpressionType ,
62
+ operationAnalysisContext . ReportDiagnostic ) ,
63
+ OperationKind . BinaryOperator ) ;
64
+ } ) ;
56
65
}
57
66
58
67
/// <summary>
59
68
/// Check to see if we have an invocation to string.Equals that has an empty string as an argument.
60
69
/// </summary>
61
- private static void AnalyzeInvocationExpression ( IInvocationOperation invocationOperation , Action < Diagnostic > reportDiagnostic )
70
+ private static void AnalyzeInvocationExpression ( IInvocationOperation invocationOperation ,
71
+ INamedTypeSymbol ? linqExpressionTreeType , Action < Diagnostic > reportDiagnostic )
62
72
{
63
73
if ( ! invocationOperation . Arguments . IsEmpty )
64
74
{
65
75
IMethodSymbol methodSymbol = invocationOperation . TargetMethod ;
66
- if ( methodSymbol != null &&
67
- IsStringEqualsMethod ( methodSymbol ) &&
68
- HasAnEmptyStringArgument ( invocationOperation ) )
76
+ if ( methodSymbol == null
77
+ || ! IsStringEqualsMethod ( methodSymbol )
78
+ || ! HasAnEmptyStringArgument ( invocationOperation ) )
79
+ {
80
+ return ;
81
+ }
82
+
83
+ // Check if we are in a Expression<Func<T...>> context, in which case it is possible
84
+ // that the underlying call doesn't have the helper so we want to bail-out.
85
+ if ( ! invocationOperation . IsWithinExpressionTree ( linqExpressionTreeType ) )
69
86
{
70
87
reportDiagnostic ( invocationOperation . Syntax . CreateDiagnostic ( s_rule ) ) ;
71
88
}
@@ -76,7 +93,8 @@ private static void AnalyzeInvocationExpression(IInvocationOperation invocationO
76
93
/// Check to see if we have a equals or not equals expression where an empty string is being
77
94
/// compared.
78
95
/// </summary>
79
- private static void AnalyzeBinaryExpression ( IBinaryOperation binaryOperation , Action < Diagnostic > reportDiagnostic )
96
+ private static void AnalyzeBinaryExpression ( IBinaryOperation binaryOperation ,
97
+ INamedTypeSymbol ? linqExpressionTreeType , Action < Diagnostic > reportDiagnostic )
80
98
{
81
99
if ( binaryOperation . OperatorKind is not BinaryOperatorKind . Equals and
82
100
not BinaryOperatorKind . NotEquals )
@@ -90,7 +108,15 @@ private static void AnalyzeBinaryExpression(IBinaryOperation binaryOperation, Ac
90
108
return ;
91
109
}
92
110
93
- if ( IsEmptyString ( binaryOperation . LeftOperand ) || IsEmptyString ( binaryOperation . RightOperand ) )
111
+ if ( ! IsEmptyString ( binaryOperation . LeftOperand )
112
+ && ! IsEmptyString ( binaryOperation . RightOperand ) )
113
+ {
114
+ return ;
115
+ }
116
+
117
+ // Check if we are in a Expression<Func<T...>> context, in which case it is possible
118
+ // that the underlying call doesn't have the helper so we want to bail-out.
119
+ if ( ! binaryOperation . IsWithinExpressionTree ( linqExpressionTreeType ) )
94
120
{
95
121
reportDiagnostic ( binaryOperation . Syntax . CreateDiagnostic ( s_rule ) ) ;
96
122
}
0 commit comments