-
Notifications
You must be signed in to change notification settings - Fork 1.2k
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
[flake8-pyi
] Allow overloaded __exit__
and __aexit__
definitions (PYI036
)
#11057
Conversation
8142db8
to
5324a57
Compare
|
Oh, good to see it! I've been having an issue with this recently. 👍👍👍 |
5324a57
to
ecb2b74
Compare
let non_self_positional_args: SmallVec<[&ParameterWithDefault; 3]> = parameters | ||
.posonlyargs | ||
.iter() | ||
.chain(parameters.posonlyargs.iter()) | ||
.collect::<SmallVec<[&ParameterWithDefault; 4]>>(); | ||
.chain(parameters.args.iter()) | ||
.skip(1) | ||
.collect(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think the logic here has been wrong for a while. It should have always been parameters.posonlyargs.iter().chain(parameters.args.iter())
, not parameters.args.iter().chain(parameters.posonlyargs.iter())
. I think it didn't come up because it the bug lead to false negatives rather than false positives, so nobody noticed!
21769ae
to
a72e54f
Compare
}; | ||
|
||
// Collect all the overloads for this method into a SmallVec | ||
let function_overloads: SmallVec<[&StmtFunctionDef; 2]> = parent_class_def |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would just use a normal Vec
here personally. It's not a very hot path, right? This is exactly if you're defining one of a small set of methods, and it has an overload?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It shouldn't be very hot, no. I mainly used a SmallVec
because it was already being used elsewhere in this file, and it is exceedingly unlikely that anybody would use more than two overloads for __exit__
or __aexit__
. Is there a disadvantage to using a SmallVec here?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think if it were me making this decision on a desert island, I'd probably just skip overloads since I wouldn't find the complexity-to-value tradeoff to be good enough (we're trading ~400 lines of code for a scenario that seems somewhat rare). But I defer to you as the owner of the rule, and it looks like you have intuition that this is worth supporting.
Yeah I know what you mean. My initial thought when I saw the issue was "well, that's going to add extra complexity to the check, and it's a less common case, so we may as well skip it". But (although we rarely bother with it in typeshed, it seems like extra complexity for little gain), Adam's recent blog post is strictly-speaking correct that it's more accurate to use overloads for this case, and that blog seems to be making the rounds at the moment. And if you are going to use overloads, then I think it's really easy to get it subtly wrong by e.g. using This may be partly sunk-cost fallacy on my part though, given that I've gone ahead an implemented it now... I think I'll add a few more test cases to make sure that every conceivable scenario is covered, then go ahead with this. |
a72e54f
to
1f7d294
Compare
Summary
Fixes #11044
We should definitely allow overloaded definitions for
__exit__
and__aexit__
. Nonetheless, validating overloads is pretty tricky without the full machinery of a type checker, so this PR deliberately takes a pretty conservative approach. If there are exactly two overloads, both overloads have exactly three annotated non-self non-keyword-only arguments each, and both overloads have no keyword-only arguments or variadic arguments -- then, for both overloads, we check to see if the overload matches one of two recognised overload variants (permitting small variations such as if one parameter is hinted withobject
or_typeshed.Unused
). If there's an unexpected number of arguments, however, we just don't check the overload.Test plan
cargo test
.