KEMBAR78
Adds support for `__slots__` assignment, refs #10801 by sobolevn · Pull Request #10864 · python/mypy · GitHub
Skip to content

Conversation

@sobolevn
Copy link
Member

@sobolevn sobolevn commented Jul 24, 2021

Description

We can now detect assignment that are not matching defined __slots__.

Example:

class A:
   __slots__ = ('a',)

class B(A):
   __slots__ = ('b',)
   def __init__(self) -> None:
       self.a = 1  # ok
       self.b = 2  # ok
       self.c = 3  # error

b: B
reveal_type(b.c)

Снимок экрана 2021-07-24 в 18 41 14

In runtime it produces:

Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 6, in __init__
AttributeError: 'B' object has no attribute 'c'

Closes #10801

Test Plan

  • I need to test that single class with __slots__ works
  • I need to test that slotted class with __slots__ in super classes works
  • I need to test that a class can extend slotted class without extra __slots__ definition
  • I need to test that assignment of wrong __slots__ raise an error
  • I need to test that __slots__ with wrong elements raise an error
  • I need to test that __slots__ with dynamic elements are ignored
  • I need to test that __slots__ = 1 outside of class definition context should be fine
  • I need to test __dict__ corner case

Future steps

  • I can also modify attrs plugin to include slots if @attr.s(slots=True) is passed

@github-actions

This comment has been minimized.

@github-actions

This comment has been minimized.

@sobolevn
Copy link
Member Author

Improtant special case: __slots__ = ('__dict__',) which should be supported.

@sobolevn sobolevn marked this pull request as ready for review July 25, 2021 10:50
@sobolevn
Copy link
Member Author

I am not sure why this fails: https://app.travis-ci.com/github/python/mypy/jobs/526676991

_____________________ testOverloadLambdaUnpackingInference _____________________

[gw0] linux -- Python 3.10.0 /home/travis/build/python/mypy/.tox/py/bin/python

data: /home/travis/build/python/mypy/test-data/unit/check-overloading.test:4002:

SystemExit: 1

----------------------------- Captured stderr call -----------------------------

The typed_ast package is not installed.

For Python 2 support, install mypy using `python3 -m pip install "mypy[python2]"`Alternatively, you can install typed_ast with `python3 -m pip install typed-ast`.

It does not look related.

@sobolevn sobolevn changed the title WIP: adds support for __slots__ assignment, refs #10801 Adds support for __slots__ assignment, refs #10801 Jul 25, 2021
@sobolevn
Copy link
Member Author

sobolevn commented Aug 9, 2021

@hauntsaninja @JelleZijlstra friendly ping. Any chance to get this PR reviewed? 🙂

Copy link
Collaborator

@A5rocks A5rocks left a comment

Choose a reason for hiding this comment

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

Just a while ago I came across this behaviour of ignoring __slots__ and it didn't seem very intuitive, thanks for doing this!

@sobolevn
Copy link
Member Author

@ilevkivskyi @JukkaL Hi! Any chance to push this forward? 🙂

Copy link
Member

@emmatyping emmatyping left a comment

Choose a reason for hiding this comment

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

I think this looks pretty reasonable overall. Just a couple of questions/comments

mypy/checker.py Outdated
if isinstance(lvalue, MemberExpr) and lvalue.node:
name = lvalue.node.name
inst = get_proper_type(self.expr_checker.accept(lvalue.expr))
if (isinstance(inst, Instance) and
Copy link
Member

Choose a reason for hiding this comment

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

If a class has a mix of slots and non-slots members, wouldn't this report an error incorrectly?

E.g.

class A:
    __slots__ = ('a',)
    b = 4

Edit: ah based on the tests it looks like this does not report an error incorrectly

Copy link
Member Author

Choose a reason for hiding this comment

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

I will add a test case for this, thank you. It actually was not consistent with Python's runtime behavior:

class A:
    __slots__ = ('a',)
    b = 4

    def __init__(self) -> None:
        self.a = 1
        self.b = 2  # error here in runtime

A()
# AttributeError: 'A' object attribute 'b' is read-only

I will change the code to make this an error with mypy as well.

def __init__(self) -> None:
self.a = 1
self.b = 2
self.missing = 3
Copy link
Member

Choose a reason for hiding this comment

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

Isn't this an error if B didn't inherit from another class? It seems weird to me this would be accepted since it isn't in the __slots__ for B nor in the mro.

Copy link
Member Author

Choose a reason for hiding this comment

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

This is how Python works in runtime. If any base class has __dict__ that the child class itself would have __dict__. So, when using B(A) you will implicitly have __dict__ in place. Try this:

class A:
    pass  # no slots

class B(A):
    __slots__ = ("a", "b")

    def __init__(self) -> None:
        self.a = 1
        self.b = 2
        self.missing = 3

b = B()
b.a = 1
b.b = 2
b.missing = 3
b.extra = 4  # E: "B" has no attribute "extra"

But, without A super class it would be different:

class B:
    __slots__ = ("a", "b")

    def __init__(self) -> None:
        self.a = 1
        self.b = 2
        self.missing = 3  # AttributeError: 'B' object has no attribute 'missing'

b = B()

Copy link
Member Author

Choose a reason for hiding this comment

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

I think, that I need to add this to the docs of mypy. Do you agree?
What would be the best place for it? Common issues?

Copy link
Member

@emmatyping emmatyping Sep 15, 2021

Choose a reason for hiding this comment

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

Aha, I see, that makes more sense. I would probably add it to class basics: https://mypy.readthedocs.io/en/stable/class_basics.html#class-basics

@emmatyping
Copy link
Member

Based on the mypy_primer output, it seems there are also a couple of cases that should be handled:

  • __slots__ is Any or list[Any]
  • the rvalue is tuple() or similar for dict, list, etc.

It also seems to report a few true errors :)

@sobolevn
Copy link
Member Author

sobolevn commented Sep 14, 2021

slots is Any or list[Any]

Like just __slots__ = returns_any()?

the rvalue is tuple() or similar for dict, list, etc.

Will do! 👍
For now, I will just ignore cases like list(('a', 'b', 'c')), because it requires quite a lot of code for extra inference.
We can add this later. Moreover, __slots__ = list(('a', 'b', 'c')) is not quite common thing to write, people usually just write __slots__ = ('a', 'b').

But, tests will be added with no errors reported.

@github-actions

This comment has been minimized.

@github-actions

This comment has been minimized.

@sobolevn
Copy link
Member Author

Ok, this reports too many false positives. I will fix that + add more tests for @property.setter

@sobolevn
Copy link
Member Author

sobolevn commented Sep 14, 2021

One more important case I've missed: __setattr__ support. Adding tests for it as well.

This code works in runtime:

class F:
    __slots__ = ('a',)
    def __init__(self) -> None:
        self.a = {}
    def __setattr__(self, k, v) -> None:
        if k != 'a':
            self.a[k] = v
        else:
            F.__dict__[k].__set__(self, v)

f = F()
f.extra = 2
print(f.a)

https://stackoverflow.com/questions/19566419/can-setattr-can-be-defined-in-a-class-with-slots

@github-actions

This comment has been minimized.

@sobolevn
Copy link
Member Author

sobolevn commented Sep 14, 2021

Now, let's look on cases highlighted by mypy_primer:

  1. dulwich. Seems like a bug, I've opened a PR: Fixes Tag.signature field jelmer/dulwich#901 Nevermind, this was a false positive with using a = custom_property(get_a, set_a). Tests for this case are added
  2. parso. It looks valid: https://github.com/davidhalter/parso/blob/master/parso/tree.py#L374-L385 NodeOrLeaf does not have parent field, that's true. I've opened a PR: Fixes __slots__ definition in NodeOrLeaf davidhalter/parso#199
  3. bidict. Looks like a false-positive. It uses nxt = property(a, b). I've missed this form in tests https://github.com/jab/bidict/blob/56ccfad1d9577c39828f43f8d29f39ccc0bdbb62/bidict/_orderedbase.py#L83 I will fix it today.

@emmatyping
Copy link
Member

@sobolevn wow that's great progress in the mypy_primer output, thank you!

@github-actions

This comment has been minimized.

@sobolevn
Copy link
Member Author

Looks like that mypy_primer now detects a single correct bug from #10864 (comment) 🎉 🎉

@github-actions
Copy link
Contributor

Diff from mypy_primer, showing the effect of this PR on open source code:

parso (https://github.com/davidhalter/parso.git)
+ parso/tree.py:385: error: Trying to assign name "parent" that is not in "__slots__" of type "parso.tree.NodeOrLeaf"

@sobolevn
Copy link
Member Author

@ethanhs any other corner-cases you can think of? 🤔

@emmatyping
Copy link
Member

emmatyping commented Sep 15, 2021

@ethanhs any other corner-cases you can think of? 🤔

Not really, this seems to cover things pretty well!

Do you want to add the docs change in a separate PR?

@sobolevn
Copy link
Member Author

Not really, this seems to cover things pretty well!

Anyways, I am here to help / fix any problems.
The last question: should we hide this behind a configuration flag?

Do you want to add the docs change in a separate PR?

@ethanhs yes, I need to think about good places to fit it 👍

@emmatyping
Copy link
Member

The last question: should we hide this behind a configuration flag?

I don't think it should be. It seems to be a useful check and with the most recent changes I don't think it should raise too many false positives, and we have so many options already...

@sobolevn
Copy link
Member Author

@ethanhs agreed. Anything else I need to do to get this merged? 🙂

@emmatyping emmatyping merged commit 8ac0dc2 into python:master Sep 21, 2021
@emmatyping
Copy link
Member

Thanks!

@sobolevn
Copy link
Member Author

sobolevn commented Sep 21, 2021

Thanks, @ethanhs! I will submit a PR with the docs today 🙂

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.

__slots__ isn't effective for limiting attributes

3 participants