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

Make classes with __getitem__ work in for context #10386

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

Conversation

hatal175
Copy link
Contributor

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 :)

@github-actions

This comment has been minimized.

Copy link
Member

@sobolevn sobolevn left a 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."""
Copy link
Member

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.
Copy link
Member

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
Copy link
Member

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:
Copy link
Member

@sobolevn sobolevn Dec 9, 2021

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 statements
  • list(iter(A())) and list(iter(B()))


if iter_msg_builder.is_errors():
# We couldn't find __iter__ so let's try __getitem__
getitem_msg_builder = temp_message_builder()
Copy link
Member

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.

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.

3 participants