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

Detect instanceof pattern variables in scope #4066

Merged
merged 5 commits into from
Mar 4, 2024

Conversation

timtebeek
Copy link
Contributor

What's your motivation?

Used in InstanceOfPatternMatch to detect variables that exist in the surrounding context, which now fails to pick up pattern matching for instance of variables, and as such introduces two identically named variables.

@knutwannheden
Copy link
Contributor

This is probably a hood start and we could go ahead and integrate this, if it already fixes some issues. To be on the safe side, the pattern variable should also be added to the scope of the containing block, as it can in fact also be visible there, depending on what the if and else statements contain.

Comment on lines +247 to +254
public J.InstanceOf visitInstanceOf(J.InstanceOf instanceOf, Set<String> strings) {
if (instanceOf.getPattern() instanceof J.Identifier) {
Set<String> names = nameScopes.get(currentScope.peek());
if (names != null) {
names.add(((J.Identifier)instanceOf.getPattern()).getSimpleName());
}
}
return super.visitInstanceOf(instanceOf, strings);
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@knutwannheden is this what you meant by adding it to the parent scope?

Suggested change
public J.InstanceOf visitInstanceOf(J.InstanceOf instanceOf, Set<String> strings) {
if (instanceOf.getPattern() instanceof J.Identifier) {
Set<String> names = nameScopes.get(currentScope.peek());
if (names != null) {
names.add(((J.Identifier)instanceOf.getPattern()).getSimpleName());
}
}
return super.visitInstanceOf(instanceOf, strings);
public J.InstanceOf visitInstanceOf(J.InstanceOf instanceOf, Set<String> namesInScope) {
if (instanceOf.getPattern() instanceof J.Identifier) {
Set<String> names = nameScopes.get(currentScope.peek());
if (names != null) {
String simpleName = ((J.Identifier) instanceOf.getPattern()).getSimpleName();
names.add(simpleName);
namesInScope.add(simpleName);
}
}
return super.visitInstanceOf(instanceOf, namesInScope);

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not exactly sure what the effect would be of this change. But For example the following code must not end up creating two pattern variables with the same name, as that would give a compile error:

        Object o = "";
        if (!(o instanceof String)) {
            throw new IllegalArgumentException(((String) o));
        }
        if (o instanceof String) {
            System.out.println((String) o);
        }

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The given example is very contrived, as it would result in a runtime exception. But I wanted to get the point across regarding the compiler error. I could otherwise also come up with more sensible examples.

So, to be on the safe side, a pattern variable as in the following code should be regarded as belonging to the surrounding block (not just the if statement's then block):

        Object o = "";
        if (o instanceof String s) {
            // ...
        }

That way the VariableNameUtils should not propose s as a variable name in any statements following the if statement.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looking this over again and how it's used in InstanceOfPattern match I think right now we don't need the above suggested fix, given how we aggregate the scope names in postVisit

public @Nullable J postVisit(J tree, Set<String> namesInScope) {
if (!currentScope.isEmpty() && currentScope.peek().getValue().equals(tree)) {
currentScope.pop();
}
if (scope.getValue().equals(tree)) {
Cursor aggregatedScope = getCursor().getValue() instanceof JavaSourceFile ? getCursor() : aggregateNameScope();
// Add names from parent scope.
Set<String> names = nameScopes.get(aggregatedScope);
// Add the names created in the target scope.
namesInScope.addAll(names);
nameScopes.forEach((key, value) -> {
if (key.isScopeInPath(scope.getValue())) {
namesInScope.addAll(value);
}
});
return tree;
}
return super.postVisit(tree, namesInScope);

I'll try to pick this up in rewrite-static-analysis.

@timtebeek timtebeek merged commit 31d4c83 into main Mar 4, 2024
1 check passed
@timtebeek timtebeek deleted the detect-instanceof-pattern-variables-in-scope branch March 4, 2024 11:42
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
Archived in project
Development

Successfully merging this pull request may close these issues.

2 participants