-
-
Notifications
You must be signed in to change notification settings - Fork 30.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
Enum regression: AttributeError when accessing class variables on instances #87328
Comments
I believe I found a regression in Enum in Python 3.10.0a5. This is Python 3.9: >>> import enum
>>> class C(enum.Enum):
... A = 0
... B = 1
...
>>> C.A
<C.A: 0>
>>> C.B
<C.B: 1>
>>> C(0).A
<C.A: 0>
>>> C(0).B
<C.B: 1>
>>> The Enum instances can access class-attributes via dot, like normal instances do. While in Python 3.10.0a5: >>> import enum
>>> class C(enum.Enum):
... A = 0
... B = 1
...
>>> C.A
<C.A: 0>
>>> C.B
<C.B: 1>
>>> C(0).A
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/usr/lib64/python3.10/enum.py", line 146, in __get__
raise AttributeError(
AttributeError: C: no attribute 'A'
>>> C(0).B
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/usr/lib64/python3.10/enum.py", line 146, in __get__
raise AttributeError(
AttributeError: C: no attribute 'B' In real word situations, it breaks meson: The __str__ method does: if self is self.EXITCODE: ... And it fails with:
This worked with 3.10.0a4. If this is a deliberate backwards incompatible change of behavior, I don't think it is documented in the changelog or what's new in Python 3.10, nor that it was deprecated in Python 3.9 and 3.8. |
Git bisect: c314e60 is the first new commit
Lib/enum.py | 297 +++++++++++++-------- |
The code for that return self.name ? ----- The issue is not being able to access class attributes, the issue is whether one enum member should be seen as an attribute of another: EnumClass.RED.BLUE and the answer is no. That wasn't possible when Enum was first introduced in 3.4, and was an unfortunate side-effect of speeding up member access in 3.5 (or 3.6). The docs have always warned against it. A deprecation warning is an easier transition, though, so I'll do that for 3.10, and in 3.11 it will become an error. Thank you for reporting! :-) |
Thanks. In the meantime, I've opened mesonbuild/meson#8318 |
Author of said meson code here. I use this pattern when the enum names and values are implementation details, but the string values are user visible. In this case the enum itself is used internally to represent what kind of test we're doing, but we're initializing it from user input. There might be reasons in the future that the names of the enum members and the string values being passed in aren't the same anymore, say because we map two string values to one enum value. I guess I could accomplish the same thing with a staticmethod or helper function, but the code is effectively an alternate initializer, and follows that pattern using a classmethod. |
Ethan, should the depreciation exist for 2 releases prior to removal? Dylan, even in that case, I guess the proper way to access the other members is supposed to be type(self).FOO or ClassName.FOO, not self.FOO. |
Dylan, it's not the def __str__(self):
if self is self.EXITCODE:
return 'exitcode' it should be def __str__(self):
cls = self.__class__
if self is cls.EXITCODE:
return 'exitcode' |
"Wait for the warning to appear in at least two major Python versions. It's fine to wait more than two releases." https://www.python.org/dev/peps/pep-0387/#basic-policy-for-backwards-compatibility So indeed, if you add the warning in 3.10, the behavior can be removed from 3.12 earliest. |
DeprecationWarning will be active in 3.10 and 3.11 with removal in 3.12. |
Thank you, Ethan. |
You're welcome. Thank you for pushing the issue! :-) |
Note: these values reflect the state of the issue at the time it was migrated and might not reflect the current state.
Show more details
GitHub fields:
bugs.python.org fields:
The text was updated successfully, but these errors were encountered: