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

[ruff] Fix false positive on global keyword (RUF052) #15235

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

Garrett-R
Copy link

@Garrett-R Garrett-R commented Jan 3, 2025

Summary

Fixes #14996.

BTW, the issue correctly reports a false positive given that the docs say "Only local variables in function scopes are flagged by the rule".

Test Plan

I added the false positive to the fixture and verified it indeed caused a false positive before my fix was done.

@@ -1,4 +1,4 @@
# Correct
##################### Correct #####################
Copy link
Author

@Garrett-R Garrett-R Jan 3, 2025

Choose a reason for hiding this comment

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

This might be just personal taste, so happy to remove!

But when I saw just # Correct, I wasn't sure if that was demarcating a section of code. I personally find this more clear.

EDIT: should probably use blocker header format (otherwise conflicts with E266).

@Garrett-R Garrett-R force-pushed the garrett/14996/ruff052-fix-global-keyword branch from ce679c7 to 70f2f89 Compare January 3, 2025 05:12
_num = 1

def print_num():
print(_num)
Copy link
Author

@Garrett-R Garrett-R Jan 3, 2025

Choose a reason for hiding this comment

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

Note: I inserted the test here into the ############### Correct ############# section, seemed right thing to do.

But given it's not being at the end of this file, it means I got some large diffs in the fixture files. I reviewed the diffs and there's no "real" changes to them, as expected.

@@ -37,6 +37,17 @@ def fun():
_x = "reassigned global"
return _x

# (we had a false positive when a global var is used like this in 2 functions)
Copy link
Author

Choose a reason for hiding this comment

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

(am I overcommenting? I was just worried that it might not be clear that this is a single "test" spread across 2 functions, but happy to remove)

Copy link
Contributor

github-actions bot commented Jan 3, 2025

ruff-ecosystem results

Linter (stable)

✅ ecosystem check detected no linter changes.

Linter (preview)

✅ ecosystem check detected no linter changes.

@Garrett-R Garrett-R marked this pull request as ready for review January 3, 2025 05:20
@@ -1498,7 +1498,7 @@ impl<'a> SemanticModel<'a> {
source: self.node_id,
context: self.execution_context(),
exceptions: self.exceptions(),
flags: BindingFlags::empty(),
flags: BindingFlags::GLOBAL,
Copy link
Member

Choose a reason for hiding this comment

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

I don't think this is the right fix. The binding pushed here is for the variable _x in the global scope. But _x isn't considered a global-binding in the global scope. It is only when used in a nested scope. For example, the checker explicitly skips over the global handling when in the global scope here

if !self.semantic.scope_id.is_global() {
for name in names {
let binding_id = self.semantic.global_scope().get(name);
// Mark the binding in the global scope as "rebound" in the current scope.
if let Some(binding_id) = binding_id {
self.semantic
.add_rebinding_scope(binding_id, self.semantic.scope_id);
}
// Add a binding to the current scope.
let binding_id = self.semantic.push_binding(
name.range(),
BindingKind::Global(binding_id),
BindingFlags::GLOBAL,
);
let scope = self.semantic.current_scope_mut();
scope.add(name, binding_id);
}
}
}

But I think you're on the right track. What I find suspicious (and changing it fixes the bug as well) is that scope is self.scope_id but we add the binding to self.global_scope_mut. That's why I think it should actually be ScopeId::global() instead. But I'd like a confirmation from @charliermarsh on this change

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

used-dummy-variable (RUF052) - false positive on private global variable when it has a usage in two functions
2 participants