Skip to content

Commit

Permalink
gh-93910: Fix enum performance regression (GH-94614)
Browse files Browse the repository at this point in the history
This removes the performance regression in 3.11, **at the expense of not fixing
the "bug" that allows accessing values from values** (e.g. `Color.RED.BLUE`).

Using the benchmark @markshannon [presented](#93910 (comment)), the results are:

| Version | Enum | Fast enum | Normal class |
| --- | --- | --- | --- |
| 3.10 | 2.04 | 0.59 | 0.56 |
| 3.11 | 2.78 | 0.31 | 0.15 |
| This PR | 1.30 | 0.32 | 0.16 |

I share this mostly as information about the source of the regression, as this may be useful. It may be that the lower-risk approach for the beta is just to revert to a previously-known working state.
  • Loading branch information
mdboom authored Jul 7, 2022
1 parent 277f55c commit ed136b9
Show file tree
Hide file tree
Showing 3 changed files with 13 additions and 8 deletions.
17 changes: 9 additions & 8 deletions Lib/enum.py
Original file line number Diff line number Diff line change
Expand Up @@ -309,16 +309,17 @@ def __set_name__(self, enum_class, member_name):
if descriptor and not need_override:
# previous enum.property found, no further action needed
pass
else:
elif descriptor and need_override:
redirect = property()
redirect.__set_name__(enum_class, member_name)
if descriptor and need_override:
# previous enum.property found, but some other inherited attribute
# is in the way; copy fget, fset, fdel to this one
redirect.fget = descriptor.fget
redirect.fset = descriptor.fset
redirect.fdel = descriptor.fdel
# Previous enum.property found, but some other inherited attribute
# is in the way; copy fget, fset, fdel to this one.
redirect.fget = descriptor.fget
redirect.fset = descriptor.fset
redirect.fdel = descriptor.fdel
setattr(enum_class, member_name, redirect)
else:
setattr(enum_class, member_name, enum_member)
# now add to _member_map_ (even aliases)
enum_class._member_map_[member_name] = enum_member
try:
Expand Down Expand Up @@ -647,7 +648,7 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
'member order does not match _order_:\n %r\n %r'
% (enum_class._member_names_, _order_)
)
#

return enum_class

def __bool__(cls):
Expand Down
1 change: 1 addition & 0 deletions Lib/test/test_enum.py
Original file line number Diff line number Diff line change
Expand Up @@ -2646,6 +2646,7 @@ class Private(Enum):
self.assertEqual(Private._Private__corporal, 'Radar')
self.assertEqual(Private._Private__major_, 'Hoolihan')

@unittest.skip("Accessing all values retained for performance reasons, see GH-93910")
def test_exception_for_member_from_member_access(self):
with self.assertRaisesRegex(AttributeError, "<enum .Di.> member has no attribute .NO."):
class Di(Enum):
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
The ability to access the other values of an enum on an enum (e.g.
``Color.RED.BLUE``) has been restored in order to fix a performance
regression.

0 comments on commit ed136b9

Please sign in to comment.