Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[java] UnusedPrivateMethod false positive from inner class via external class #1189

Closed
arifogel opened this issue Jun 14, 2018 · 9 comments · Fixed by #3113
Closed

[java] UnusedPrivateMethod false positive from inner class via external class #1189

arifogel opened this issue Jun 14, 2018 · 9 comments · Fixed by #3113
Labels
a:false-positive PMD flags a piece of code that is not problematic
Milestone

Comments

@arifogel
Copy link

Affects PMD Version: 6.4.0

Rule: UnusedPrivateMethod

Description:
A private method of an outer class called in an inner class via a method from an external class returning an object of the outer class type is falsely reported as being unused.

Code Sample demonstrating the issue:

Running PMD through: [Maven]

@arifogel
Copy link
Author

output.log

@jsotuyod jsotuyod added the a:false-positive PMD flags a piece of code that is not problematic label Jun 14, 2018
@jsotuyod
Copy link
Member

@arifogel thanks for the report. The rule is effectively not using type resolution yet, so these non-straightforward scenarios will be missed for the time being.

@arifogel
Copy link
Author

I'd be willing to look into this with a little shepherding. Is there some documentation/code I could review to help me get started? Or is this blocked on something that is currently churning internally?

@arifogel
Copy link
Author

@jsotuyod any progress on this? I'm still willing to look into this with a bit of initial guidance.

@adangel
Copy link
Member

adangel commented Sep 19, 2018

Hi @arifogel,
Thanks for the offer!

First step would be (if not done already): https://github.com/pmd/pmd/wiki/Setup-IDE
Then you can add your case as a new test case into https://github.com/pmd/pmd/blob/master/pmd-java/src/test/resources/net/sourceforge/pmd/lang/java/rule/bestpractices/xml/UnusedPrivateMethod.xml (you would add here only your OuterClass). Since you are dealing with type resolution, you'll need additionally the two classes as java source, so that it is compiled and available during test runtime. You can add your two classes OuterClass and Referrer to the package net.sourceforge.pmd.lang.java.rule.bestpractices under src/test/java.
You can start the unit test then through BestPracticesRulesTest. You should now see a failure. You can also run the test in the debugger and step through the class https://github.com/pmd/pmd/blob/master/pmd-java/src/main/java/net/sourceforge/pmd/lang/java/rule/bestpractices/UnusedPrivateMethodRule.java .

Then the hard part comes: Fixing the issue. We currently don't resolve the return type of a method call - which would be needed here in order to know, that we call the method privateMethod().

I guess, changing the sample code to this one:

public final class OuterClass {
  private static class InnerClass {
    private void callPrivateMethod() {
      OuterClass outer = Referrer.getOuterClass();
      outer.privateMethod();
    }
  }

  private void privateMethod() {}
}

won't trigger this false positive, since we know the type of the variable outer. If this also triggers the rule, then this might be a first easier step to fix.

Let me know, if you run into any problems - either here or on https://gitter.im/pmd/pmd

@arifogel
Copy link
Author

I finally got my IDE set up. Your version of the sample code passes.
I was able to condense the false positive test code to a single file:

public final class OuterClass {
    public static class InnerClass {
        public void callPrivateMethod() {
            Referrer.getOuterClass().privateMethod();
        }
    }

    public static class Referrer {
        public static OuterClass getOuterClass() {
            return new OuterClass();
        }
    }

    private void privateMethod() {
    }
}

Now comes the hard part of processing the return type..

arifogel added a commit to arifogel/pmd that referenced this issue Oct 27, 2018
@arifogel
Copy link
Author

It seems like the failure occurs even earlier than I expected.
I changed net.sourceforge.pmd.lang.java.symboltable.TRACE to true and ran using pmd master for the concise snippet above.
The trace is unexpected:

new search for variable Referrer:4:class net.sourceforge.pmd.lang.java.ast.ASTName
 checking scope MethodScope: for name occurrence Referrer:4:class net.sourceforge.pmd.lang.java.ast.ASTName
 moving up from MethodScope: to ClassScope (InnerClass): Inner Classes ; Method callPrivateMethod, line 3, params = 0(begins at line 3, 0 usages), 
 checking scope ClassScope (InnerClass): Inner Classes ; Method callPrivateMethod, line 3, params = 0(begins at line 3, 0 usages),  for name occurrence Referrer:4:class net.sourceforge.pmd.lang.java.ast.ASTName
 moving up from ClassScope (InnerClass): Inner Classes ; Method callPrivateMethod, line 3, params = 0(begins at line 3, 0 usages),  to ClassScope (OuterClass): Method privateMethod, line 14, params = 0(begins at line 14, 0 usages), 
 checking scope ClassScope (OuterClass): Method privateMethod, line 14, params = 0(begins at line 14, 0 usages),  for name occurrence Referrer:4:class net.sourceforge.pmd.lang.java.ast.ASTName
 moving up from ClassScope (OuterClass): Method privateMethod, line 14, params = 0(begins at line 14, 0 usages),  to SourceFileScope: Class OuterClass
 checking scope SourceFileScope: Class OuterClass for name occurrence Referrer:4:class net.sourceforge.pmd.lang.java.ast.ASTName
found []

AFAICT, even Referrer by itself cannot be found. Is this the same problem of not using type resolution, or is this a separate issue?

@KroArtem
Copy link
Contributor

It seems like we also encountered this issue.

@oowekyala oowekyala added this to the 7.0.0 milestone Nov 10, 2020
oowekyala added a commit to oowekyala/pmd that referenced this issue Nov 10, 2020
@oowekyala oowekyala linked a pull request Feb 19, 2021 that will close this issue
4 tasks
@adangel adangel mentioned this issue Jan 23, 2023
55 tasks
@adangel
Copy link
Member

adangel commented Apr 22, 2023

This has been fixed with PMD 7.0.0-rc1.

@adangel adangel closed this as completed Apr 22, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
a:false-positive PMD flags a piece of code that is not problematic
Projects
None yet
Development

Successfully merging a pull request may close this issue.

5 participants