-
-
Notifications
You must be signed in to change notification settings - Fork 2.9k
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
Make classes with __getitem__ work in for context #10386
base: master
Are you sure you want to change the base?
Conversation
This comment has been minimized.
This comment has been minimized.
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.
Thank you!
@@ -3499,20 +3500,40 @@ def analyze_iterable_item_type(self, expr: Expression) -> Tuple[Type, Type]: | |||
"""Analyse iterable expression and return iterator and iterator item types.""" |
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.
Let's explain how iterables are handled in CPython. Priorities and possibilities.
return self.named_generic_type("typing.Iterator", [getitem_type]), getitem_type | ||
|
||
self.msg.add_errors(iter_msg_builder) | ||
# Non-tuple iterable. |
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.
This comment does not really explain what is going on anymore.
|
||
if not getitem_msg_builder.is_errors(): | ||
# We found __getitem__ | ||
return self.named_generic_type("typing.Iterator", [getitem_type]), getitem_type |
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.
Looks like it should be typing.Iterable
and not Iterator
.
@@ -1940,6 +1940,17 @@ reveal_type(list(c for c in C)) # N: Revealed type is "builtins.list[__main__.C | |||
reveal_type(list(C)) # N: Revealed type is "builtins.list[__main__.C*]" | |||
[builtins fixtures/list.pyi] | |||
|
|||
[case testIterableGetItemOnClass] | |||
class A: |
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.
Let's also test:
- Metaclasses with
__getitem__
and__class_getitem__
for
statementslist(iter(A()))
andlist(iter(B()))
|
||
if iter_msg_builder.is_errors(): | ||
# We couldn't find __iter__ so let's try __getitem__ | ||
getitem_msg_builder = temp_message_builder() |
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.
You can try to use .clean_copy()
here.
Description
The for analysis searches for iter to decide if it is iterable. I've added a search for getitem as well.
This is a step in the right direction for #2220 but it doesn't really make classes with getitem be accepted as arguments marked as Iterable - I'm not sure how to solve this without being intrusive in either typeshed or mypy type inference.
Please note that I've made it work with getitem that accepts int (as the docs say) so if you have only getitem with string, this will generate an error.
Test Plan
I've added a few tests and mypy primer seems to have fewer errors :)