2
// CallToObjectEqualsViaBaseIssue.cs
5
// Simon Lindgren <simon.n.lindgren@gmail.com>
7
// Copyright (c) 2012 Simon Lindgren
9
// Permission is hereby granted, free of charge, to any person obtaining a copy
10
// of this software and associated documentation files (the "Software"), to deal
11
// in the Software without restriction, including without limitation the rights
12
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
13
// copies of the Software, and to permit persons to whom the Software is
14
// furnished to do so, subject to the following conditions:
16
// The above copyright notice and this permission notice shall be included in
17
// all copies or substantial portions of the Software.
19
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
22
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
27
using System.Collections.Generic;
28
using ICSharpCode.NRefactory.TypeSystem;
29
using ICSharpCode.NRefactory.Semantics;
32
namespace ICSharpCode.NRefactory.CSharp.Refactoring
34
[IssueDescription("Call to base.Equals resolves to Object.Equals, which is reference equality",
35
Description = "Finds potentially erroneous calls to Object.Equals.",
36
Category = IssueCategories.CodeQualityIssues,
37
Severity = Severity.Warning,
38
ResharperDisableKeyword = "BaseObjectEqualsIsObjectEquals")]
39
public class CallToObjectEqualsViaBaseIssue : ICodeIssueProvider
41
public IEnumerable<CodeIssue> GetIssues(BaseRefactoringContext context)
43
return new GatherVisitor(context).GetIssues();
46
class GatherVisitor : GatherVisitorBase<CallToObjectEqualsViaBaseIssue>
48
public GatherVisitor(BaseRefactoringContext context) : base (context)
52
public override void VisitInvocationExpression(InvocationExpression invocationExpression)
54
base.VisitInvocationExpression(invocationExpression);
56
if (invocationExpression.Arguments.Count != 1) {
59
var memberExpression = invocationExpression.Target as MemberReferenceExpression;
60
if (memberExpression == null || memberExpression.MemberName != "Equals" || !(memberExpression.Target is BaseReferenceExpression)) {
63
var resolveResult = ctx.Resolve(invocationExpression) as InvocationResolveResult;
64
if (resolveResult == null || !resolveResult.Member.DeclaringTypeDefinition.IsKnownType(KnownTypeCode.Object)) {
67
var title = ctx.TranslateString("Call to base.Equals resolves to Object.Equals, which is reference equality");
68
AddIssue(invocationExpression, title, GetActions(invocationExpression));
71
IEnumerable<CodeAction> GetActions(InvocationExpression invocationExpression)
73
yield return new CodeAction(ctx.TranslateString("Change invocation to call Object.ReferenceEquals"), script => {
74
var args = Enumerable.Concat(new [] { new ThisReferenceExpression() }, invocationExpression.Arguments.Select(arg => arg.Clone()));
75
var newInvocation = MakeInvocation("object.ReferenceEquals", args);
76
script.Replace(invocationExpression, newInvocation);
77
}, invocationExpression);
78
yield return new CodeAction(ctx.TranslateString("Remove 'base.'"), script => {
79
var newInvocation = MakeInvocation("Equals", invocationExpression.Arguments.Select(arg => arg.Clone()));
80
script.Replace(invocationExpression, newInvocation);
81
}, invocationExpression);
84
static InvocationExpression MakeInvocation(string memberName, IEnumerable<Expression> unClonedArguments)
86
return new InvocationExpression(new IdentifierExpression(memberName), unClonedArguments);