mirror of
https://github.com/python/cpython.git
synced 2025-10-24 10:23:58 +00:00
Revert "bpo-40066: [Enum] update str() and format() output (GH-30582)" (GH-30632)
This reverts commit acf7403f9b
.
This commit is contained in:
parent
7f4b69b907
commit
42a64c03ec
14 changed files with 2021 additions and 2087 deletions
|
@ -2,10 +2,15 @@
|
||||||
Enum HOWTO
|
Enum HOWTO
|
||||||
==========
|
==========
|
||||||
|
|
||||||
|
:Author: Ethan Furman <ethan at stoneleaf dot us>
|
||||||
|
|
||||||
.. _enum-basic-tutorial:
|
.. _enum-basic-tutorial:
|
||||||
|
|
||||||
.. currentmodule:: enum
|
.. currentmodule:: enum
|
||||||
|
|
||||||
|
Basic Enum Tutorial
|
||||||
|
-------------------
|
||||||
|
|
||||||
An :class:`Enum` is a set of symbolic names bound to unique values. They are
|
An :class:`Enum` is a set of symbolic names bound to unique values. They are
|
||||||
similar to global variables, but they offer a more useful :func:`repr()`,
|
similar to global variables, but they offer a more useful :func:`repr()`,
|
||||||
grouping, type-safety, and a few other features.
|
grouping, type-safety, and a few other features.
|
||||||
|
@ -23,14 +28,6 @@ selection of values. For example, the days of the week::
|
||||||
... SATURDAY = 6
|
... SATURDAY = 6
|
||||||
... SUNDAY = 7
|
... SUNDAY = 7
|
||||||
|
|
||||||
Or perhaps the RGB primary colors::
|
|
||||||
|
|
||||||
>>> from enum import Enum
|
|
||||||
>>> class Color(Enum):
|
|
||||||
... RED = 1
|
|
||||||
... GREEN = 2
|
|
||||||
... BLUE = 3
|
|
||||||
|
|
||||||
As you can see, creating an :class:`Enum` is as simple as writing a class that
|
As you can see, creating an :class:`Enum` is as simple as writing a class that
|
||||||
inherits from :class:`Enum` itself.
|
inherits from :class:`Enum` itself.
|
||||||
|
|
||||||
|
@ -44,14 +41,13 @@ important, but either way that value can be used to get the corresponding
|
||||||
member::
|
member::
|
||||||
|
|
||||||
>>> Weekday(3)
|
>>> Weekday(3)
|
||||||
<Weekday.WEDNESDAY: 3>
|
Weekday.WEDNESDAY
|
||||||
|
|
||||||
As you can see, the ``repr()`` of a member shows the enum name, the member name,
|
As you can see, the ``repr()`` of a member shows the enum name and the
|
||||||
and the value. The ``str()`` of a member shows only the enum name and member
|
member name. The ``str()`` on a member shows only its name::
|
||||||
name::
|
|
||||||
|
|
||||||
>>> print(Weekday.THURSDAY)
|
>>> print(Weekday.THURSDAY)
|
||||||
Weekday.THURSDAY
|
THURSDAY
|
||||||
|
|
||||||
The *type* of an enumeration member is the enum it belongs to::
|
The *type* of an enumeration member is the enum it belongs to::
|
||||||
|
|
||||||
|
@ -101,8 +97,8 @@ The complete :class:`Weekday` enum now looks like this::
|
||||||
Now we can find out what today is! Observe::
|
Now we can find out what today is! Observe::
|
||||||
|
|
||||||
>>> from datetime import date
|
>>> from datetime import date
|
||||||
>>> Weekday.from_date(date.today()) # doctest: +SKIP
|
>>> Weekday.from_date(date.today())
|
||||||
<Weekday.TUESDAY: 2>
|
Weekday.TUESDAY
|
||||||
|
|
||||||
Of course, if you're reading this on some other day, you'll see that day instead.
|
Of course, if you're reading this on some other day, you'll see that day instead.
|
||||||
|
|
||||||
|
@ -128,21 +124,21 @@ Just like the original :class:`Weekday` enum above, we can have a single selecti
|
||||||
|
|
||||||
>>> first_week_day = Weekday.MONDAY
|
>>> first_week_day = Weekday.MONDAY
|
||||||
>>> first_week_day
|
>>> first_week_day
|
||||||
<Weekday.MONDAY: 1>
|
Weekday.MONDAY
|
||||||
|
|
||||||
But :class:`Flag` also allows us to combine several members into a single
|
But :class:`Flag` also allows us to combine several members into a single
|
||||||
variable::
|
variable::
|
||||||
|
|
||||||
>>> weekend = Weekday.SATURDAY | Weekday.SUNDAY
|
>>> weekend = Weekday.SATURDAY | Weekday.SUNDAY
|
||||||
>>> weekend
|
>>> weekend
|
||||||
<Weekday.SATURDAY|SUNDAY: 96>
|
Weekday.SATURDAY|Weekday.SUNDAY
|
||||||
|
|
||||||
You can even iterate over a :class:`Flag` variable::
|
You can even iterate over a :class:`Flag` variable::
|
||||||
|
|
||||||
>>> for day in weekend:
|
>>> for day in weekend:
|
||||||
... print(day)
|
... print(day)
|
||||||
Weekday.SATURDAY
|
SATURDAY
|
||||||
Weekday.SUNDAY
|
SUNDAY
|
||||||
|
|
||||||
Okay, let's get some chores set up::
|
Okay, let's get some chores set up::
|
||||||
|
|
||||||
|
@ -177,7 +173,6 @@ yourself some work and use :func:`auto()` for the values::
|
||||||
|
|
||||||
.. _enum-advanced-tutorial:
|
.. _enum-advanced-tutorial:
|
||||||
|
|
||||||
|
|
||||||
Programmatic access to enumeration members and their attributes
|
Programmatic access to enumeration members and their attributes
|
||||||
---------------------------------------------------------------
|
---------------------------------------------------------------
|
||||||
|
|
||||||
|
@ -186,16 +181,16 @@ situations where ``Color.RED`` won't do because the exact color is not known
|
||||||
at program-writing time). ``Enum`` allows such access::
|
at program-writing time). ``Enum`` allows such access::
|
||||||
|
|
||||||
>>> Color(1)
|
>>> Color(1)
|
||||||
<Color.RED: 1>
|
Color.RED
|
||||||
>>> Color(3)
|
>>> Color(3)
|
||||||
<Color.BLUE: 3>
|
Color.BLUE
|
||||||
|
|
||||||
If you want to access enum members by *name*, use item access::
|
If you want to access enum members by *name*, use item access::
|
||||||
|
|
||||||
>>> Color['RED']
|
>>> Color['RED']
|
||||||
<Color.RED: 1>
|
Color.RED
|
||||||
>>> Color['GREEN']
|
>>> Color['GREEN']
|
||||||
<Color.GREEN: 2>
|
Color.GREEN
|
||||||
|
|
||||||
If you have an enum member and need its :attr:`name` or :attr:`value`::
|
If you have an enum member and need its :attr:`name` or :attr:`value`::
|
||||||
|
|
||||||
|
@ -217,7 +212,7 @@ Having two enum members with the same name is invalid::
|
||||||
...
|
...
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
TypeError: 'SQUARE' already defined as 2
|
TypeError: 'SQUARE' already defined as: 2
|
||||||
|
|
||||||
However, an enum member can have other names associated with it. Given two
|
However, an enum member can have other names associated with it. Given two
|
||||||
entries ``A`` and ``B`` with the same value (and ``A`` defined first), ``B``
|
entries ``A`` and ``B`` with the same value (and ``A`` defined first), ``B``
|
||||||
|
@ -232,11 +227,11 @@ By-name lookup of ``B`` will also return the member ``A``::
|
||||||
... ALIAS_FOR_SQUARE = 2
|
... ALIAS_FOR_SQUARE = 2
|
||||||
...
|
...
|
||||||
>>> Shape.SQUARE
|
>>> Shape.SQUARE
|
||||||
<Shape.SQUARE: 2>
|
Shape.SQUARE
|
||||||
>>> Shape.ALIAS_FOR_SQUARE
|
>>> Shape.ALIAS_FOR_SQUARE
|
||||||
<Shape.SQUARE: 2>
|
Shape.SQUARE
|
||||||
>>> Shape(2)
|
>>> Shape(2)
|
||||||
<Shape.SQUARE: 2>
|
Shape.SQUARE
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
|
@ -304,7 +299,7 @@ Iteration
|
||||||
Iterating over the members of an enum does not provide the aliases::
|
Iterating over the members of an enum does not provide the aliases::
|
||||||
|
|
||||||
>>> list(Shape)
|
>>> list(Shape)
|
||||||
[<Shape.SQUARE: 2>, <Shape.DIAMOND: 1>, <Shape.CIRCLE: 3>]
|
[Shape.SQUARE, Shape.DIAMOND, Shape.CIRCLE]
|
||||||
|
|
||||||
The special attribute ``__members__`` is a read-only ordered mapping of names
|
The special attribute ``__members__`` is a read-only ordered mapping of names
|
||||||
to members. It includes all names defined in the enumeration, including the
|
to members. It includes all names defined in the enumeration, including the
|
||||||
|
@ -313,10 +308,10 @@ aliases::
|
||||||
>>> for name, member in Shape.__members__.items():
|
>>> for name, member in Shape.__members__.items():
|
||||||
... name, member
|
... name, member
|
||||||
...
|
...
|
||||||
('SQUARE', <Shape.SQUARE: 2>)
|
('SQUARE', Shape.SQUARE)
|
||||||
('DIAMOND', <Shape.DIAMOND: 1>)
|
('DIAMOND', Shape.DIAMOND)
|
||||||
('CIRCLE', <Shape.CIRCLE: 3>)
|
('CIRCLE', Shape.CIRCLE)
|
||||||
('ALIAS_FOR_SQUARE', <Shape.SQUARE: 2>)
|
('ALIAS_FOR_SQUARE', Shape.SQUARE)
|
||||||
|
|
||||||
The ``__members__`` attribute can be used for detailed programmatic access to
|
The ``__members__`` attribute can be used for detailed programmatic access to
|
||||||
the enumeration members. For example, finding all the aliases::
|
the enumeration members. For example, finding all the aliases::
|
||||||
|
@ -365,8 +360,8 @@ below)::
|
||||||
Allowed members and attributes of enumerations
|
Allowed members and attributes of enumerations
|
||||||
----------------------------------------------
|
----------------------------------------------
|
||||||
|
|
||||||
Most of the examples above use integers for enumeration values. Using integers
|
Most of the examples above use integers for enumeration values. Using integers is
|
||||||
is short and handy (and provided by default by the `Functional API`_), but not
|
short and handy (and provided by default by the `Functional API`_), but not
|
||||||
strictly enforced. In the vast majority of use-cases, one doesn't care what
|
strictly enforced. In the vast majority of use-cases, one doesn't care what
|
||||||
the actual value of an enumeration is. But if the value *is* important,
|
the actual value of an enumeration is. But if the value *is* important,
|
||||||
enumerations can have arbitrary values.
|
enumerations can have arbitrary values.
|
||||||
|
@ -394,7 +389,7 @@ usual. If we have this enumeration::
|
||||||
Then::
|
Then::
|
||||||
|
|
||||||
>>> Mood.favorite_mood()
|
>>> Mood.favorite_mood()
|
||||||
<Mood.HAPPY: 3>
|
Mood.HAPPY
|
||||||
>>> Mood.HAPPY.describe()
|
>>> Mood.HAPPY.describe()
|
||||||
('HAPPY', 3)
|
('HAPPY', 3)
|
||||||
>>> str(Mood.FUNKY)
|
>>> str(Mood.FUNKY)
|
||||||
|
@ -430,7 +425,7 @@ any members. So this is forbidden::
|
||||||
...
|
...
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
TypeError: <enum 'MoreColor'> cannot extend <enum 'Color'>
|
TypeError: MoreColor: cannot extend enumeration 'Color'
|
||||||
|
|
||||||
But this is allowed::
|
But this is allowed::
|
||||||
|
|
||||||
|
@ -481,9 +476,11 @@ The :class:`Enum` class is callable, providing the following functional API::
|
||||||
>>> Animal
|
>>> Animal
|
||||||
<enum 'Animal'>
|
<enum 'Animal'>
|
||||||
>>> Animal.ANT
|
>>> Animal.ANT
|
||||||
<Animal.ANT: 1>
|
Animal.ANT
|
||||||
|
>>> Animal.ANT.value
|
||||||
|
1
|
||||||
>>> list(Animal)
|
>>> list(Animal)
|
||||||
[<Animal.ANT: 1>, <Animal.BEE: 2>, <Animal.CAT: 3>, <Animal.DOG: 4>]
|
[Animal.ANT, Animal.BEE, Animal.CAT, Animal.DOG]
|
||||||
|
|
||||||
The semantics of this API resemble :class:`~collections.namedtuple`. The first
|
The semantics of this API resemble :class:`~collections.namedtuple`. The first
|
||||||
argument of the call to :class:`Enum` is the name of the enumeration.
|
argument of the call to :class:`Enum` is the name of the enumeration.
|
||||||
|
@ -628,7 +625,16 @@ StrEnum
|
||||||
The second variation of :class:`Enum` that is provided is also a subclass of
|
The second variation of :class:`Enum` that is provided is also a subclass of
|
||||||
:class:`str`. Members of a :class:`StrEnum` can be compared to strings;
|
:class:`str`. Members of a :class:`StrEnum` can be compared to strings;
|
||||||
by extension, string enumerations of different types can also be compared
|
by extension, string enumerations of different types can also be compared
|
||||||
to each other.
|
to each other. :class:`StrEnum` exists to help avoid the problem of getting
|
||||||
|
an incorrect member::
|
||||||
|
|
||||||
|
>>> from enum import StrEnum
|
||||||
|
>>> class Directions(StrEnum):
|
||||||
|
... NORTH = 'north', # notice the trailing comma
|
||||||
|
... SOUTH = 'south'
|
||||||
|
|
||||||
|
Before :class:`StrEnum`, ``Directions.NORTH`` would have been the :class:`tuple`
|
||||||
|
``('north',)``.
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
|
@ -639,8 +645,9 @@ IntFlag
|
||||||
The next variation of :class:`Enum` provided, :class:`IntFlag`, is also based
|
The next variation of :class:`Enum` provided, :class:`IntFlag`, is also based
|
||||||
on :class:`int`. The difference being :class:`IntFlag` members can be combined
|
on :class:`int`. The difference being :class:`IntFlag` members can be combined
|
||||||
using the bitwise operators (&, \|, ^, ~) and the result is still an
|
using the bitwise operators (&, \|, ^, ~) and the result is still an
|
||||||
:class:`IntFlag` member, if possible. Like :class:`IntEnum`, :class:`IntFlag`
|
:class:`IntFlag` member, if possible. However, as the name implies, :class:`IntFlag`
|
||||||
members are also integers and can be used wherever an :class:`int` is used.
|
members also subclass :class:`int` and can be used wherever an :class:`int` is
|
||||||
|
used.
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
|
@ -663,7 +670,7 @@ Sample :class:`IntFlag` class::
|
||||||
... X = 1
|
... X = 1
|
||||||
...
|
...
|
||||||
>>> Perm.R | Perm.W
|
>>> Perm.R | Perm.W
|
||||||
<Perm.R|W: 6>
|
Perm.R|Perm.W
|
||||||
>>> Perm.R + Perm.W
|
>>> Perm.R + Perm.W
|
||||||
6
|
6
|
||||||
>>> RW = Perm.R | Perm.W
|
>>> RW = Perm.R | Perm.W
|
||||||
|
@ -678,11 +685,11 @@ It is also possible to name the combinations::
|
||||||
... X = 1
|
... X = 1
|
||||||
... RWX = 7
|
... RWX = 7
|
||||||
>>> Perm.RWX
|
>>> Perm.RWX
|
||||||
<Perm.RWX: 7>
|
Perm.RWX
|
||||||
>>> ~Perm.RWX
|
>>> ~Perm.RWX
|
||||||
<Perm: 0>
|
Perm(0)
|
||||||
>>> Perm(7)
|
>>> Perm(7)
|
||||||
<Perm.RWX: 7>
|
Perm.RWX
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
|
@ -695,7 +702,7 @@ Another important difference between :class:`IntFlag` and :class:`Enum` is that
|
||||||
if no flags are set (the value is 0), its boolean evaluation is :data:`False`::
|
if no flags are set (the value is 0), its boolean evaluation is :data:`False`::
|
||||||
|
|
||||||
>>> Perm.R & Perm.X
|
>>> Perm.R & Perm.X
|
||||||
<Perm: 0>
|
Perm(0)
|
||||||
>>> bool(Perm.R & Perm.X)
|
>>> bool(Perm.R & Perm.X)
|
||||||
False
|
False
|
||||||
|
|
||||||
|
@ -703,7 +710,7 @@ Because :class:`IntFlag` members are also subclasses of :class:`int` they can
|
||||||
be combined with them (but may lose :class:`IntFlag` membership::
|
be combined with them (but may lose :class:`IntFlag` membership::
|
||||||
|
|
||||||
>>> Perm.X | 4
|
>>> Perm.X | 4
|
||||||
<Perm.R|X: 5>
|
Perm.R|Perm.X
|
||||||
|
|
||||||
>>> Perm.X | 8
|
>>> Perm.X | 8
|
||||||
9
|
9
|
||||||
|
@ -719,7 +726,7 @@ be combined with them (but may lose :class:`IntFlag` membership::
|
||||||
:class:`IntFlag` members can also be iterated over::
|
:class:`IntFlag` members can also be iterated over::
|
||||||
|
|
||||||
>>> list(RW)
|
>>> list(RW)
|
||||||
[<Perm.R: 4>, <Perm.W: 2>]
|
[Perm.R, Perm.W]
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
|
@ -746,7 +753,7 @@ flags being set, the boolean evaluation is :data:`False`::
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
...
|
...
|
||||||
>>> Color.RED & Color.GREEN
|
>>> Color.RED & Color.GREEN
|
||||||
<Color: 0>
|
Color(0)
|
||||||
>>> bool(Color.RED & Color.GREEN)
|
>>> bool(Color.RED & Color.GREEN)
|
||||||
False
|
False
|
||||||
|
|
||||||
|
@ -760,7 +767,7 @@ while combinations of flags won't::
|
||||||
... WHITE = RED | BLUE | GREEN
|
... WHITE = RED | BLUE | GREEN
|
||||||
...
|
...
|
||||||
>>> Color.WHITE
|
>>> Color.WHITE
|
||||||
<Color.WHITE: 7>
|
Color.WHITE
|
||||||
|
|
||||||
Giving a name to the "no flags set" condition does not change its boolean
|
Giving a name to the "no flags set" condition does not change its boolean
|
||||||
value::
|
value::
|
||||||
|
@ -772,7 +779,7 @@ value::
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
...
|
...
|
||||||
>>> Color.BLACK
|
>>> Color.BLACK
|
||||||
<Color.BLACK: 0>
|
Color.BLACK
|
||||||
>>> bool(Color.BLACK)
|
>>> bool(Color.BLACK)
|
||||||
False
|
False
|
||||||
|
|
||||||
|
@ -780,7 +787,7 @@ value::
|
||||||
|
|
||||||
>>> purple = Color.RED | Color.BLUE
|
>>> purple = Color.RED | Color.BLUE
|
||||||
>>> list(purple)
|
>>> list(purple)
|
||||||
[<Color.RED: 1>, <Color.BLUE: 2>]
|
[Color.RED, Color.BLUE]
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
|
@ -805,16 +812,16 @@ simple to implement independently::
|
||||||
pass
|
pass
|
||||||
|
|
||||||
This demonstrates how similar derived enumerations can be defined; for example
|
This demonstrates how similar derived enumerations can be defined; for example
|
||||||
a :class:`FloatEnum` that mixes in :class:`float` instead of :class:`int`.
|
a :class:`StrEnum` that mixes in :class:`str` instead of :class:`int`.
|
||||||
|
|
||||||
Some rules:
|
Some rules:
|
||||||
|
|
||||||
1. When subclassing :class:`Enum`, mix-in types must appear before
|
1. When subclassing :class:`Enum`, mix-in types must appear before
|
||||||
:class:`Enum` itself in the sequence of bases, as in the :class:`IntEnum`
|
:class:`Enum` itself in the sequence of bases, as in the :class:`IntEnum`
|
||||||
example above.
|
example above.
|
||||||
2. Mix-in types must be subclassable. For example, :class:`bool` and
|
2. Mix-in types must be subclassable. For example,
|
||||||
:class:`range` are not subclassable and will throw an error during Enum
|
:class:`bool` and :class:`range` are not subclassable
|
||||||
creation if used as the mix-in type.
|
and will throw an error during Enum creation if used as the mix-in type.
|
||||||
3. While :class:`Enum` can have members of any type, once you mix in an
|
3. While :class:`Enum` can have members of any type, once you mix in an
|
||||||
additional type, all the members must have values of that type, e.g.
|
additional type, all the members must have values of that type, e.g.
|
||||||
:class:`int` above. This restriction does not apply to mix-ins which only
|
:class:`int` above. This restriction does not apply to mix-ins which only
|
||||||
|
@ -822,18 +829,15 @@ Some rules:
|
||||||
4. When another data type is mixed in, the :attr:`value` attribute is *not the
|
4. When another data type is mixed in, the :attr:`value` attribute is *not the
|
||||||
same* as the enum member itself, although it is equivalent and will compare
|
same* as the enum member itself, although it is equivalent and will compare
|
||||||
equal.
|
equal.
|
||||||
5. %-style formatting: ``%s`` and ``%r`` call the :class:`Enum` class's
|
5. %-style formatting: `%s` and `%r` call the :class:`Enum` class's
|
||||||
:meth:`__str__` and :meth:`__repr__` respectively; other codes (such as
|
:meth:`__str__` and :meth:`__repr__` respectively; other codes (such as
|
||||||
``%i`` or ``%h`` for IntEnum) treat the enum member as its mixed-in type.
|
`%i` or `%h` for IntEnum) treat the enum member as its mixed-in type.
|
||||||
6. :ref:`Formatted string literals <f-strings>`, :meth:`str.format`,
|
6. :ref:`Formatted string literals <f-strings>`, :meth:`str.format`,
|
||||||
and :func:`format` will use the enum's :meth:`__str__` method.
|
and :func:`format` will use the mixed-in type's :meth:`__format__`
|
||||||
|
unless :meth:`__str__` or :meth:`__format__` is overridden in the subclass,
|
||||||
.. note::
|
in which case the overridden methods or :class:`Enum` methods will be used.
|
||||||
|
Use the !s and !r format codes to force usage of the :class:`Enum` class's
|
||||||
Because :class:`IntEnum`, :class:`IntFlag`, and :class:`StrEnum` are
|
:meth:`__str__` and :meth:`__repr__` methods.
|
||||||
designed to be drop-in replacements for existing constants, their
|
|
||||||
:meth:`__str__` method has been reset to their data types
|
|
||||||
:meth:`__str__` method.
|
|
||||||
|
|
||||||
When to use :meth:`__new__` vs. :meth:`__init__`
|
When to use :meth:`__new__` vs. :meth:`__init__`
|
||||||
------------------------------------------------
|
------------------------------------------------
|
||||||
|
@ -862,10 +866,10 @@ want one of them to be the value::
|
||||||
...
|
...
|
||||||
|
|
||||||
>>> print(Coordinate['PY'])
|
>>> print(Coordinate['PY'])
|
||||||
Coordinate.PY
|
PY
|
||||||
|
|
||||||
>>> print(Coordinate(3))
|
>>> print(Coordinate(3))
|
||||||
Coordinate.VY
|
VY
|
||||||
|
|
||||||
|
|
||||||
Finer Points
|
Finer Points
|
||||||
|
@ -945,36 +949,35 @@ but remain normal attributes.
|
||||||
""""""""""""""""""""
|
""""""""""""""""""""
|
||||||
|
|
||||||
Enum members are instances of their enum class, and are normally accessed as
|
Enum members are instances of their enum class, and are normally accessed as
|
||||||
``EnumClass.member``. In Python versions ``3.5`` to ``3.10`` you could access
|
``EnumClass.member``. In Python versions ``3.5`` to ``3.9`` you could access
|
||||||
members from other members -- this practice was discouraged, and in ``3.11``
|
members from other members -- this practice was discouraged, and in ``3.12``
|
||||||
:class:`Enum` returns to not allowing it::
|
:class:`Enum` will return to not allowing it, while in ``3.10`` and ``3.11``
|
||||||
|
it will raise a :exc:`DeprecationWarning`::
|
||||||
|
|
||||||
>>> class FieldTypes(Enum):
|
>>> class FieldTypes(Enum):
|
||||||
... name = 0
|
... name = 0
|
||||||
... value = 1
|
... value = 1
|
||||||
... size = 2
|
... size = 2
|
||||||
...
|
...
|
||||||
>>> FieldTypes.value.size
|
>>> FieldTypes.value.size # doctest: +SKIP
|
||||||
Traceback (most recent call last):
|
DeprecationWarning: accessing one member from another is not supported,
|
||||||
...
|
and will be disabled in 3.12
|
||||||
AttributeError: <enum 'FieldTypes'> member has no attribute 'size'
|
<FieldTypes.size: 2>
|
||||||
|
|
||||||
|
|
||||||
.. versionchanged:: 3.5
|
.. versionchanged:: 3.5
|
||||||
.. versionchanged:: 3.11
|
|
||||||
|
|
||||||
|
|
||||||
Creating members that are mixed with other data types
|
Creating members that are mixed with other data types
|
||||||
"""""""""""""""""""""""""""""""""""""""""""""""""""""
|
"""""""""""""""""""""""""""""""""""""""""""""""""""""
|
||||||
|
|
||||||
When subclassing other data types, such as :class:`int` or :class:`str`, with
|
When subclassing other data types, such as :class:`int` or :class:`str`, with
|
||||||
an :class:`Enum`, all values after the ``=`` are passed to that data type's
|
an :class:`Enum`, all values after the `=` are passed to that data type's
|
||||||
constructor. For example::
|
constructor. For example::
|
||||||
|
|
||||||
>>> class MyEnum(IntEnum): # help(int) -> int(x, base=10) -> integer
|
>>> class MyEnum(IntEnum):
|
||||||
... example = '11', 16 # so x='11' and base=16
|
... example = '11', 16 # '11' will be interpreted as a hexadecimal
|
||||||
...
|
... # number
|
||||||
>>> MyEnum.example.value # and hex(11) is...
|
>>> MyEnum.example.value
|
||||||
17
|
17
|
||||||
|
|
||||||
|
|
||||||
|
@ -997,12 +1000,13 @@ Plain :class:`Enum` classes always evaluate as :data:`True`.
|
||||||
"""""""""""""""""""""""""""""
|
"""""""""""""""""""""""""""""
|
||||||
|
|
||||||
If you give your enum subclass extra methods, like the `Planet`_
|
If you give your enum subclass extra methods, like the `Planet`_
|
||||||
class below, those methods will show up in a :func:`dir` of the member,
|
class below, those methods will show up in a :func:`dir` of the member and the
|
||||||
but not of the class::
|
class. Attributes defined in an :func:`__init__` method will only show up in a
|
||||||
|
:func:`dir` of the member::
|
||||||
|
|
||||||
>>> dir(Planet) # doctest: +SKIP
|
>>> dir(Planet)
|
||||||
['EARTH', 'JUPITER', 'MARS', 'MERCURY', 'NEPTUNE', 'SATURN', 'URANUS', 'VENUS', '__class__', '__doc__', '__members__', '__module__']
|
['EARTH', 'JUPITER', 'MARS', 'MERCURY', 'NEPTUNE', 'SATURN', 'URANUS', 'VENUS', '__class__', '__doc__', '__init__', '__members__', '__module__', 'surface_gravity']
|
||||||
>>> dir(Planet.EARTH) # doctest: +SKIP
|
>>> dir(Planet.EARTH)
|
||||||
['__class__', '__doc__', '__module__', 'mass', 'name', 'radius', 'surface_gravity', 'value']
|
['__class__', '__doc__', '__module__', 'mass', 'name', 'radius', 'surface_gravity', 'value']
|
||||||
|
|
||||||
|
|
||||||
|
@ -1021,10 +1025,19 @@ are comprised of a single bit::
|
||||||
... CYAN = GREEN | BLUE
|
... CYAN = GREEN | BLUE
|
||||||
...
|
...
|
||||||
>>> Color(3) # named combination
|
>>> Color(3) # named combination
|
||||||
<Color.YELLOW: 3>
|
Color.YELLOW
|
||||||
>>> Color(7) # not named combination
|
>>> Color(7) # not named combination
|
||||||
<Color.RED|GREEN|BLUE: 7>
|
Color.RED|Color.GREEN|Color.BLUE
|
||||||
|
|
||||||
|
``StrEnum`` and :meth:`str.__str__`
|
||||||
|
"""""""""""""""""""""""""""""""""""
|
||||||
|
|
||||||
|
An important difference between :class:`StrEnum` and other Enums is the
|
||||||
|
:meth:`__str__` method; because :class:`StrEnum` members are strings, some
|
||||||
|
parts of Python will read the string data directly, while others will call
|
||||||
|
:meth:`str()`. To make those two operations have the same result,
|
||||||
|
:meth:`StrEnum.__str__` will be the same as :meth:`str.__str__` so that
|
||||||
|
``str(StrEnum.member) == StrEnum.member`` is true.
|
||||||
|
|
||||||
``Flag`` and ``IntFlag`` minutia
|
``Flag`` and ``IntFlag`` minutia
|
||||||
""""""""""""""""""""""""""""""""
|
""""""""""""""""""""""""""""""""
|
||||||
|
@ -1047,16 +1060,16 @@ the following are true:
|
||||||
- only canonical flags are returned during iteration::
|
- only canonical flags are returned during iteration::
|
||||||
|
|
||||||
>>> list(Color.WHITE)
|
>>> list(Color.WHITE)
|
||||||
[<Color.RED: 1>, <Color.GREEN: 2>, <Color.BLUE: 4>]
|
[Color.RED, Color.GREEN, Color.BLUE]
|
||||||
|
|
||||||
- negating a flag or flag set returns a new flag/flag set with the
|
- negating a flag or flag set returns a new flag/flag set with the
|
||||||
corresponding positive integer value::
|
corresponding positive integer value::
|
||||||
|
|
||||||
>>> Color.BLUE
|
>>> Color.BLUE
|
||||||
<Color.BLUE: 4>
|
Color.BLUE
|
||||||
|
|
||||||
>>> ~Color.BLUE
|
>>> ~Color.BLUE
|
||||||
<Color.RED|GREEN: 3>
|
Color.RED|Color.GREEN
|
||||||
|
|
||||||
- names of pseudo-flags are constructed from their members' names::
|
- names of pseudo-flags are constructed from their members' names::
|
||||||
|
|
||||||
|
@ -1066,29 +1079,25 @@ the following are true:
|
||||||
- multi-bit flags, aka aliases, can be returned from operations::
|
- multi-bit flags, aka aliases, can be returned from operations::
|
||||||
|
|
||||||
>>> Color.RED | Color.BLUE
|
>>> Color.RED | Color.BLUE
|
||||||
<Color.PURPLE: 5>
|
Color.PURPLE
|
||||||
|
|
||||||
>>> Color(7) # or Color(-1)
|
>>> Color(7) # or Color(-1)
|
||||||
<Color.WHITE: 7>
|
Color.WHITE
|
||||||
|
|
||||||
>>> Color(0)
|
>>> Color(0)
|
||||||
<Color.BLACK: 0>
|
Color.BLACK
|
||||||
|
|
||||||
- membership / containment checking: zero-valued flags are always considered
|
- membership / containment checking has changed slightly -- zero-valued flags
|
||||||
to be contained::
|
are never considered to be contained::
|
||||||
|
|
||||||
>>> Color.BLACK in Color.WHITE
|
>>> Color.BLACK in Color.WHITE
|
||||||
True
|
False
|
||||||
|
|
||||||
otherwise, only if all bits of one flag are in the other flag will True
|
otherwise, if all bits of one flag are in the other flag, True is returned::
|
||||||
be returned::
|
|
||||||
|
|
||||||
>>> Color.PURPLE in Color.WHITE
|
>>> Color.PURPLE in Color.WHITE
|
||||||
True
|
True
|
||||||
|
|
||||||
>>> Color.GREEN in Color.PURPLE
|
|
||||||
False
|
|
||||||
|
|
||||||
There is a new boundary mechanism that controls how out-of-range / invalid
|
There is a new boundary mechanism that controls how out-of-range / invalid
|
||||||
bits are handled: ``STRICT``, ``CONFORM``, ``EJECT``, and ``KEEP``:
|
bits are handled: ``STRICT``, ``CONFORM``, ``EJECT``, and ``KEEP``:
|
||||||
|
|
||||||
|
@ -1172,7 +1181,7 @@ Using :class:`auto` would look like::
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
...
|
...
|
||||||
>>> Color.GREEN
|
>>> Color.GREEN
|
||||||
<Color.GREEN: 3>
|
<Color.GREEN>
|
||||||
|
|
||||||
|
|
||||||
Using :class:`object`
|
Using :class:`object`
|
||||||
|
@ -1185,24 +1194,10 @@ Using :class:`object` would look like::
|
||||||
... GREEN = object()
|
... GREEN = object()
|
||||||
... BLUE = object()
|
... BLUE = object()
|
||||||
...
|
...
|
||||||
>>> Color.GREEN # doctest: +SKIP
|
|
||||||
<Color.GREEN: <object object at 0x...>>
|
|
||||||
|
|
||||||
This is also a good example of why you might want to write your own
|
|
||||||
:meth:`__repr__`::
|
|
||||||
|
|
||||||
>>> class Color(Enum):
|
|
||||||
... RED = object()
|
|
||||||
... GREEN = object()
|
|
||||||
... BLUE = object()
|
|
||||||
... def __repr__(self):
|
|
||||||
... return "<%s.%s>" % (self.__class__.__name__, self._name_)
|
|
||||||
...
|
|
||||||
>>> Color.GREEN
|
>>> Color.GREEN
|
||||||
<Color.GREEN>
|
<Color.GREEN>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Using a descriptive string
|
Using a descriptive string
|
||||||
""""""""""""""""""""""""""
|
""""""""""""""""""""""""""
|
||||||
|
|
||||||
|
@ -1214,7 +1209,9 @@ Using a string as the value would look like::
|
||||||
... BLUE = 'too fast!'
|
... BLUE = 'too fast!'
|
||||||
...
|
...
|
||||||
>>> Color.GREEN
|
>>> Color.GREEN
|
||||||
<Color.GREEN: 'go'>
|
<Color.GREEN>
|
||||||
|
>>> Color.GREEN.value
|
||||||
|
'go'
|
||||||
|
|
||||||
|
|
||||||
Using a custom :meth:`__new__`
|
Using a custom :meth:`__new__`
|
||||||
|
@ -1235,7 +1232,9 @@ Using an auto-numbering :meth:`__new__` would look like::
|
||||||
... BLUE = ()
|
... BLUE = ()
|
||||||
...
|
...
|
||||||
>>> Color.GREEN
|
>>> Color.GREEN
|
||||||
<Color.GREEN: 2>
|
<Color.GREEN>
|
||||||
|
>>> Color.GREEN.value
|
||||||
|
2
|
||||||
|
|
||||||
To make a more general purpose ``AutoNumber``, add ``*args`` to the signature::
|
To make a more general purpose ``AutoNumber``, add ``*args`` to the signature::
|
||||||
|
|
||||||
|
@ -1258,7 +1257,7 @@ to handle any extra arguments::
|
||||||
... BLEACHED_CORAL = () # New color, no Pantone code yet!
|
... BLEACHED_CORAL = () # New color, no Pantone code yet!
|
||||||
...
|
...
|
||||||
>>> Swatch.SEA_GREEN
|
>>> Swatch.SEA_GREEN
|
||||||
<Swatch.SEA_GREEN: 2>
|
<Swatch.SEA_GREEN>
|
||||||
>>> Swatch.SEA_GREEN.pantone
|
>>> Swatch.SEA_GREEN.pantone
|
||||||
'1246'
|
'1246'
|
||||||
>>> Swatch.BLEACHED_CORAL.pantone
|
>>> Swatch.BLEACHED_CORAL.pantone
|
||||||
|
@ -1385,9 +1384,30 @@ An example to show the :attr:`_ignore_` attribute in use::
|
||||||
... Period['day_%d' % i] = i
|
... Period['day_%d' % i] = i
|
||||||
...
|
...
|
||||||
>>> list(Period)[:2]
|
>>> list(Period)[:2]
|
||||||
[<Period.day_0: datetime.timedelta(0)>, <Period.day_1: datetime.timedelta(days=1)>]
|
[Period.day_0, Period.day_1]
|
||||||
>>> list(Period)[-2:]
|
>>> list(Period)[-2:]
|
||||||
[<Period.day_365: datetime.timedelta(days=365)>, <Period.day_366: datetime.timedelta(days=366)>]
|
[Period.day_365, Period.day_366]
|
||||||
|
|
||||||
|
|
||||||
|
Conforming input to Flag
|
||||||
|
^^^^^^^^^^^^^^^^^^^^^^^^
|
||||||
|
|
||||||
|
To create a :class:`Flag` enum that is more resilient to out-of-bounds results
|
||||||
|
from mathematical operations, you can use the :attr:`FlagBoundary.CONFORM`
|
||||||
|
setting::
|
||||||
|
|
||||||
|
>>> from enum import Flag, CONFORM, auto
|
||||||
|
>>> class Weekday(Flag, boundary=CONFORM):
|
||||||
|
... MONDAY = auto()
|
||||||
|
... TUESDAY = auto()
|
||||||
|
... WEDNESDAY = auto()
|
||||||
|
... THURSDAY = auto()
|
||||||
|
... FRIDAY = auto()
|
||||||
|
... SATURDAY = auto()
|
||||||
|
... SUNDAY = auto()
|
||||||
|
>>> today = Weekday.TUESDAY
|
||||||
|
>>> Weekday(today + 22) # what day is three weeks from tomorrow?
|
||||||
|
>>> Weekday.WEDNESDAY
|
||||||
|
|
||||||
|
|
||||||
.. _enumtype-examples:
|
.. _enumtype-examples:
|
||||||
|
|
|
@ -31,7 +31,7 @@ An enumeration:
|
||||||
* uses *call* syntax to return members by value
|
* uses *call* syntax to return members by value
|
||||||
* uses *index* syntax to return members by name
|
* uses *index* syntax to return members by name
|
||||||
|
|
||||||
Enumerations are created either by using :keyword:`class` syntax, or by
|
Enumerations are created either by using the :keyword:`class` syntax, or by
|
||||||
using function-call syntax::
|
using function-call syntax::
|
||||||
|
|
||||||
>>> from enum import Enum
|
>>> from enum import Enum
|
||||||
|
@ -45,7 +45,7 @@ using function-call syntax::
|
||||||
>>> # functional syntax
|
>>> # functional syntax
|
||||||
>>> Color = Enum('Color', ['RED', 'GREEN', 'BLUE'])
|
>>> Color = Enum('Color', ['RED', 'GREEN', 'BLUE'])
|
||||||
|
|
||||||
Even though we can use :keyword:`class` syntax to create Enums, Enums
|
Even though we can use the :keyword:`class` syntax to create Enums, Enums
|
||||||
are not normal Python classes. See
|
are not normal Python classes. See
|
||||||
:ref:`How are Enums different? <enum-class-differences>` for more details.
|
:ref:`How are Enums different? <enum-class-differences>` for more details.
|
||||||
|
|
||||||
|
@ -53,7 +53,7 @@ are not normal Python classes. See
|
||||||
|
|
||||||
- The class :class:`Color` is an *enumeration* (or *enum*)
|
- The class :class:`Color` is an *enumeration* (or *enum*)
|
||||||
- The attributes :attr:`Color.RED`, :attr:`Color.GREEN`, etc., are
|
- The attributes :attr:`Color.RED`, :attr:`Color.GREEN`, etc., are
|
||||||
*enumeration members* (or *members*) and are functionally constants.
|
*enumeration members* (or *enum members*) and are functionally constants.
|
||||||
- The enum members have *names* and *values* (the name of
|
- The enum members have *names* and *values* (the name of
|
||||||
:attr:`Color.RED` is ``RED``, the value of :attr:`Color.BLUE` is
|
:attr:`Color.RED` is ``RED``, the value of :attr:`Color.BLUE` is
|
||||||
``3``, etc.)
|
``3``, etc.)
|
||||||
|
@ -110,10 +110,15 @@ Module Contents
|
||||||
:class:`StrEnum` defaults to the lower-cased version of the member name,
|
:class:`StrEnum` defaults to the lower-cased version of the member name,
|
||||||
while other Enums default to 1 and increase from there.
|
while other Enums default to 1 and increase from there.
|
||||||
|
|
||||||
:func:`property`
|
:func:`global_enum`
|
||||||
|
|
||||||
|
:class:`Enum` class decorator to apply the appropriate global `__repr__`,
|
||||||
|
and export its members into the global name space.
|
||||||
|
|
||||||
|
:func:`.property`
|
||||||
|
|
||||||
Allows :class:`Enum` members to have attributes without conflicting with
|
Allows :class:`Enum` members to have attributes without conflicting with
|
||||||
member names.
|
other members' names.
|
||||||
|
|
||||||
:func:`unique`
|
:func:`unique`
|
||||||
|
|
||||||
|
@ -126,7 +131,7 @@ Module Contents
|
||||||
|
|
||||||
|
|
||||||
.. versionadded:: 3.6 ``Flag``, ``IntFlag``, ``auto``
|
.. versionadded:: 3.6 ``Flag``, ``IntFlag``, ``auto``
|
||||||
.. versionadded:: 3.11 ``StrEnum``, ``EnumCheck``, ``FlagBoundary``, ``property``
|
.. versionadded:: 3.11 ``StrEnum``, ``EnumCheck``, ``FlagBoundary``
|
||||||
|
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
|
@ -140,11 +145,6 @@ Data Types
|
||||||
to subclass *EnumType* -- see :ref:`Subclassing EnumType <enumtype-examples>`
|
to subclass *EnumType* -- see :ref:`Subclassing EnumType <enumtype-examples>`
|
||||||
for details.
|
for details.
|
||||||
|
|
||||||
*EnumType* is responsible for setting the correct :meth:`__repr__`,
|
|
||||||
:meth:`__str__`, :meth:`__format__`, and :meth:`__reduce__` methods on the
|
|
||||||
final *enum*, as well as creating the enum members, properly handling
|
|
||||||
duplicates, providing iteration over the enum class, etc.
|
|
||||||
|
|
||||||
.. method:: EnumType.__contains__(cls, member)
|
.. method:: EnumType.__contains__(cls, member)
|
||||||
|
|
||||||
Returns ``True`` if member belongs to the ``cls``::
|
Returns ``True`` if member belongs to the ``cls``::
|
||||||
|
@ -162,31 +162,32 @@ Data Types
|
||||||
.. method:: EnumType.__dir__(cls)
|
.. method:: EnumType.__dir__(cls)
|
||||||
|
|
||||||
Returns ``['__class__', '__doc__', '__members__', '__module__']`` and the
|
Returns ``['__class__', '__doc__', '__members__', '__module__']`` and the
|
||||||
names of the members in *cls*::
|
names of the members in ``cls``. User-defined methods and methods from
|
||||||
|
mixin classes will also be included::
|
||||||
|
|
||||||
>>> dir(Color)
|
>>> dir(Color)
|
||||||
['BLUE', 'GREEN', 'RED', '__class__', '__contains__', '__doc__', '__getitem__', '__init_subclass__', '__iter__', '__len__', '__members__', '__module__', '__name__', '__qualname__']
|
['BLUE', 'GREEN', 'RED', '__class__', '__doc__', '__members__', '__module__']
|
||||||
|
|
||||||
.. method:: EnumType.__getattr__(cls, name)
|
.. method:: EnumType.__getattr__(cls, name)
|
||||||
|
|
||||||
Returns the Enum member in *cls* matching *name*, or raises an :exc:`AttributeError`::
|
Returns the Enum member in *cls* matching *name*, or raises an :exc:`AttributeError`::
|
||||||
|
|
||||||
>>> Color.GREEN
|
>>> Color.GREEN
|
||||||
<Color.GREEN: 2>
|
Color.GREEN
|
||||||
|
|
||||||
.. method:: EnumType.__getitem__(cls, name)
|
.. method:: EnumType.__getitem__(cls, name)
|
||||||
|
|
||||||
Returns the Enum member in *cls* matching *name*, or raises an :exc:`KeyError`::
|
Returns the Enum member in *cls* matching *name*, or raises a :exc:`KeyError`::
|
||||||
|
|
||||||
>>> Color['BLUE']
|
>>> Color['BLUE']
|
||||||
<Color.BLUE: 3>
|
Color.BLUE
|
||||||
|
|
||||||
.. method:: EnumType.__iter__(cls)
|
.. method:: EnumType.__iter__(cls)
|
||||||
|
|
||||||
Returns each member in *cls* in definition order::
|
Returns each member in *cls* in definition order::
|
||||||
|
|
||||||
>>> list(Color)
|
>>> list(Color)
|
||||||
[<Color.RED: 1>, <Color.GREEN: 2>, <Color.BLUE: 3>]
|
[Color.RED, Color.GREEN, Color.BLUE]
|
||||||
|
|
||||||
.. method:: EnumType.__len__(cls)
|
.. method:: EnumType.__len__(cls)
|
||||||
|
|
||||||
|
@ -200,7 +201,7 @@ Data Types
|
||||||
Returns each member in *cls* in reverse definition order::
|
Returns each member in *cls* in reverse definition order::
|
||||||
|
|
||||||
>>> list(reversed(Color))
|
>>> list(reversed(Color))
|
||||||
[<Color.BLUE: 3>, <Color.GREEN: 2>, <Color.RED: 1>]
|
[Color.BLUE, Color.GREEN, Color.RED]
|
||||||
|
|
||||||
|
|
||||||
.. class:: Enum
|
.. class:: Enum
|
||||||
|
@ -231,7 +232,7 @@ Data Types
|
||||||
.. attribute:: Enum._ignore_
|
.. attribute:: Enum._ignore_
|
||||||
|
|
||||||
``_ignore_`` is only used during creation and is removed from the
|
``_ignore_`` is only used during creation and is removed from the
|
||||||
enumeration once creation is complete.
|
enumeration once that is complete.
|
||||||
|
|
||||||
``_ignore_`` is a list of names that will not become members, and whose
|
``_ignore_`` is a list of names that will not become members, and whose
|
||||||
names will also be removed from the completed enumeration. See
|
names will also be removed from the completed enumeration. See
|
||||||
|
@ -260,7 +261,7 @@ Data Types
|
||||||
.. method:: Enum.__dir__(self)
|
.. method:: Enum.__dir__(self)
|
||||||
|
|
||||||
Returns ``['__class__', '__doc__', '__module__', 'name', 'value']`` and
|
Returns ``['__class__', '__doc__', '__module__', 'name', 'value']`` and
|
||||||
any public methods defined on *self.__class__*::
|
any public methods defined on ``self.__class__`` or a mixin class::
|
||||||
|
|
||||||
>>> from datetime import date
|
>>> from datetime import date
|
||||||
>>> class Weekday(Enum):
|
>>> class Weekday(Enum):
|
||||||
|
@ -275,7 +276,7 @@ Data Types
|
||||||
... def today(cls):
|
... def today(cls):
|
||||||
... print('today is %s' % cls(date.today().isoweekday()).name)
|
... print('today is %s' % cls(date.today().isoweekday()).name)
|
||||||
>>> dir(Weekday.SATURDAY)
|
>>> dir(Weekday.SATURDAY)
|
||||||
['__class__', '__doc__', '__eq__', '__hash__', '__module__', 'name', 'today', 'value']
|
['__class__', '__doc__', '__module__', 'name', 'today', 'value']
|
||||||
|
|
||||||
.. method:: Enum._generate_next_value_(name, start, count, last_values)
|
.. method:: Enum._generate_next_value_(name, start, count, last_values)
|
||||||
|
|
||||||
|
@ -297,11 +298,6 @@ Data Types
|
||||||
>>> PowersOfThree.SECOND.value
|
>>> PowersOfThree.SECOND.value
|
||||||
6
|
6
|
||||||
|
|
||||||
.. method:: Enum.__init_subclass__(cls, \**kwds)
|
|
||||||
|
|
||||||
A *classmethod* that is used to further configure subsequent subclasses.
|
|
||||||
By default, does nothing.
|
|
||||||
|
|
||||||
.. method:: Enum._missing_(cls, value)
|
.. method:: Enum._missing_(cls, value)
|
||||||
|
|
||||||
A *classmethod* for looking up values not found in *cls*. By default it
|
A *classmethod* for looking up values not found in *cls*. By default it
|
||||||
|
@ -321,52 +317,39 @@ Data Types
|
||||||
>>> Build.DEBUG.value
|
>>> Build.DEBUG.value
|
||||||
'debug'
|
'debug'
|
||||||
>>> Build('deBUG')
|
>>> Build('deBUG')
|
||||||
<Build.DEBUG: 'debug'>
|
Build.DEBUG
|
||||||
|
|
||||||
.. method:: Enum.__repr__(self)
|
.. method:: Enum.__repr__(self)
|
||||||
|
|
||||||
Returns the string used for *repr()* calls. By default, returns the
|
Returns the string used for *repr()* calls. By default, returns the
|
||||||
*Enum* name, member name, and value, but can be overridden::
|
*Enum* name and the member name, but can be overridden::
|
||||||
|
|
||||||
>>> class OtherStyle(Enum):
|
>>> class OldStyle(Enum):
|
||||||
... ALTERNATE = auto()
|
... RETRO = auto()
|
||||||
... OTHER = auto()
|
... OLD_SCHOOl = auto()
|
||||||
... SOMETHING_ELSE = auto()
|
... YESTERYEAR = auto()
|
||||||
... def __repr__(self):
|
... def __repr__(self):
|
||||||
... cls_name = self.__class__.__name__
|
... cls_name = self.__class__.__name__
|
||||||
... return f'{cls_name}.{self.name}'
|
... return f'<{cls_name}.{self.name}: {self.value}>'
|
||||||
>>> OtherStyle.ALTERNATE, str(OtherStyle.ALTERNATE), f"{OtherStyle.ALTERNATE}"
|
>>> OldStyle.RETRO
|
||||||
(OtherStyle.ALTERNATE, 'OtherStyle.ALTERNATE', 'OtherStyle.ALTERNATE')
|
<OldStyle.RETRO: 1>
|
||||||
|
|
||||||
.. method:: Enum.__str__(self)
|
.. method:: Enum.__str__(self)
|
||||||
|
|
||||||
Returns the string used for *str()* calls. By default, returns the
|
Returns the string used for *str()* calls. By default, returns the
|
||||||
*Enum* name and member name, but can be overridden::
|
member name, but can be overridden::
|
||||||
|
|
||||||
>>> class OtherStyle(Enum):
|
>>> class OldStyle(Enum):
|
||||||
... ALTERNATE = auto()
|
... RETRO = auto()
|
||||||
... OTHER = auto()
|
... OLD_SCHOOl = auto()
|
||||||
... SOMETHING_ELSE = auto()
|
... YESTERYEAR = auto()
|
||||||
... def __str__(self):
|
... def __str__(self):
|
||||||
... return f'{self.name}'
|
... cls_name = self.__class__.__name__
|
||||||
>>> OtherStyle.ALTERNATE, str(OtherStyle.ALTERNATE), f"{OtherStyle.ALTERNATE}"
|
... return f'{cls_name}.{self.name}'
|
||||||
(<OtherStyle.ALTERNATE: 1>, 'ALTERNATE', 'ALTERNATE')
|
>>> OldStyle.RETRO
|
||||||
|
OldStyle.RETRO
|
||||||
|
|
||||||
.. method:: Enum.__format__(self)
|
.. note::
|
||||||
|
|
||||||
Returns the string used for *format()* and *f-string* calls. By default,
|
|
||||||
returns :meth:`__str__` returns, but can be overridden::
|
|
||||||
|
|
||||||
>>> class OtherStyle(Enum):
|
|
||||||
... ALTERNATE = auto()
|
|
||||||
... OTHER = auto()
|
|
||||||
... SOMETHING_ELSE = auto()
|
|
||||||
... def __format__(self, spec):
|
|
||||||
... return f'{self.name}'
|
|
||||||
>>> OtherStyle.ALTERNATE, str(OtherStyle.ALTERNATE), f"{OtherStyle.ALTERNATE}"
|
|
||||||
(<OtherStyle.ALTERNATE: 1>, 'OtherStyle.ALTERNATE', 'ALTERNATE')
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
|
|
||||||
Using :class:`auto` with :class:`Enum` results in integers of increasing value,
|
Using :class:`auto` with :class:`Enum` results in integers of increasing value,
|
||||||
starting with ``1``.
|
starting with ``1``.
|
||||||
|
@ -384,7 +367,7 @@ Data Types
|
||||||
... TWO = 2
|
... TWO = 2
|
||||||
... THREE = 3
|
... THREE = 3
|
||||||
>>> Numbers.THREE
|
>>> Numbers.THREE
|
||||||
<Numbers.THREE: 3>
|
Numbers.THREE
|
||||||
>>> Numbers.ONE + Numbers.TWO
|
>>> Numbers.ONE + Numbers.TWO
|
||||||
3
|
3
|
||||||
>>> Numbers.THREE + 5
|
>>> Numbers.THREE + 5
|
||||||
|
@ -392,14 +375,10 @@ Data Types
|
||||||
>>> Numbers.THREE == 3
|
>>> Numbers.THREE == 3
|
||||||
True
|
True
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
Using :class:`auto` with :class:`IntEnum` results in integers of increasing
|
Using :class:`auto` with :class:`IntEnum` results in integers of increasing value,
|
||||||
value, starting with ``1``.
|
starting with ``1``.
|
||||||
|
|
||||||
.. versionchanged:: 3.11 :meth:`__str__` is now :func:`int.__str__` to
|
|
||||||
better support the *replacement of existing constants* use-case.
|
|
||||||
:meth:`__format__` was already :func:`int.__format__` for that same reason.
|
|
||||||
|
|
||||||
|
|
||||||
.. class:: StrEnum
|
.. class:: StrEnum
|
||||||
|
@ -413,16 +392,13 @@ Data Types
|
||||||
instead of ``isinstance(str, unknown)``), and in those locations you
|
instead of ``isinstance(str, unknown)``), and in those locations you
|
||||||
will need to use ``str(StrEnum.member)``.
|
will need to use ``str(StrEnum.member)``.
|
||||||
|
|
||||||
.. note::
|
|
||||||
|
|
||||||
Using :class:`auto` with :class:`StrEnum` results in the lower-cased member
|
.. note::
|
||||||
name as the value.
|
|
||||||
|
|
||||||
.. note:: :meth:`__str__` is :func:`str.__str__` to better support the
|
Using :class:`auto` with :class:`StrEnum` results in values of the member name,
|
||||||
*replacement of existing constants* use-case. :meth:`__format__` is likewise
|
lower-cased.
|
||||||
:func:`int.__format__` for that same reason.
|
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
.. class:: Flag
|
.. class:: Flag
|
||||||
|
|
||||||
|
@ -455,9 +431,9 @@ Data Types
|
||||||
Returns all contained members::
|
Returns all contained members::
|
||||||
|
|
||||||
>>> list(Color.RED)
|
>>> list(Color.RED)
|
||||||
[<Color.RED: 1>]
|
[Color.RED]
|
||||||
>>> list(purple)
|
>>> list(purple)
|
||||||
[<Color.RED: 1>, <Color.BLUE: 4>]
|
[Color.RED, Color.BLUE]
|
||||||
|
|
||||||
.. method:: __len__(self):
|
.. method:: __len__(self):
|
||||||
|
|
||||||
|
@ -485,52 +461,42 @@ Data Types
|
||||||
Returns current flag binary or'ed with other::
|
Returns current flag binary or'ed with other::
|
||||||
|
|
||||||
>>> Color.RED | Color.GREEN
|
>>> Color.RED | Color.GREEN
|
||||||
<Color.RED|GREEN: 3>
|
Color.RED|Color.GREEN
|
||||||
|
|
||||||
.. method:: __and__(self, other)
|
.. method:: __and__(self, other)
|
||||||
|
|
||||||
Returns current flag binary and'ed with other::
|
Returns current flag binary and'ed with other::
|
||||||
|
|
||||||
>>> purple & white
|
>>> purple & white
|
||||||
<Color.RED|BLUE: 5>
|
Color.RED|Color.BLUE
|
||||||
>>> purple & Color.GREEN
|
>>> purple & Color.GREEN
|
||||||
<Color: 0>
|
0x0
|
||||||
|
|
||||||
.. method:: __xor__(self, other)
|
.. method:: __xor__(self, other)
|
||||||
|
|
||||||
Returns current flag binary xor'ed with other::
|
Returns current flag binary xor'ed with other::
|
||||||
|
|
||||||
>>> purple ^ white
|
>>> purple ^ white
|
||||||
<Color.GREEN: 2>
|
Color.GREEN
|
||||||
>>> purple ^ Color.GREEN
|
>>> purple ^ Color.GREEN
|
||||||
<Color.RED|GREEN|BLUE: 7>
|
Color.RED|Color.GREEN|Color.BLUE
|
||||||
|
|
||||||
.. method:: __invert__(self):
|
.. method:: __invert__(self):
|
||||||
|
|
||||||
Returns all the flags in *type(self)* that are not in self::
|
Returns all the flags in *type(self)* that are not in self::
|
||||||
|
|
||||||
>>> ~white
|
>>> ~white
|
||||||
<Color: 0>
|
0x0
|
||||||
>>> ~purple
|
>>> ~purple
|
||||||
<Color.GREEN: 2>
|
Color.GREEN
|
||||||
>>> ~Color.RED
|
>>> ~Color.RED
|
||||||
<Color.GREEN|BLUE: 6>
|
Color.GREEN|Color.BLUE
|
||||||
|
|
||||||
.. method:: _numeric_repr_
|
.. note::
|
||||||
|
|
||||||
Function used to format any remaining unnamed numeric values. Default is
|
|
||||||
the value's repr; common choices are :func:`hex` and :func:`oct`.
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
|
|
||||||
Using :class:`auto` with :class:`Flag` results in integers that are powers
|
Using :class:`auto` with :class:`Flag` results in integers that are powers
|
||||||
of two, starting with ``1``.
|
of two, starting with ``1``.
|
||||||
|
|
||||||
.. versionchanged:: 3.11 The *repr()* of zero-valued flags has changed. It
|
|
||||||
is now::
|
|
||||||
|
|
||||||
>>> Color(0)
|
|
||||||
<Color: 0>
|
|
||||||
|
|
||||||
.. class:: IntFlag
|
.. class:: IntFlag
|
||||||
|
|
||||||
|
@ -543,9 +509,9 @@ Data Types
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
... BLUE = auto()
|
... BLUE = auto()
|
||||||
>>> Color.RED & 2
|
>>> Color.RED & 2
|
||||||
<Color: 0>
|
0x0
|
||||||
>>> Color.RED | 2
|
>>> Color.RED | 2
|
||||||
<Color.RED|GREEN: 3>
|
Color.RED|Color.GREEN
|
||||||
|
|
||||||
If any integer operation is performed with an *IntFlag* member, the result is
|
If any integer operation is performed with an *IntFlag* member, the result is
|
||||||
not an *IntFlag*::
|
not an *IntFlag*::
|
||||||
|
@ -558,25 +524,15 @@ Data Types
|
||||||
* the result is a valid *IntFlag*: an *IntFlag* is returned
|
* the result is a valid *IntFlag*: an *IntFlag* is returned
|
||||||
* the result is not a valid *IntFlag*: the result depends on the *FlagBoundary* setting
|
* the result is not a valid *IntFlag*: the result depends on the *FlagBoundary* setting
|
||||||
|
|
||||||
The *repr()* of unnamed zero-valued flags has changed. It is now:
|
.. note::
|
||||||
|
|
||||||
>>> Color(0)
|
|
||||||
<Color: 0>
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
|
|
||||||
Using :class:`auto` with :class:`IntFlag` results in integers that are powers
|
Using :class:`auto` with :class:`IntFlag` results in integers that are powers
|
||||||
of two, starting with ``1``.
|
of two, starting with ``1``.
|
||||||
|
|
||||||
.. versionchanged:: 3.11 :meth:`__str__` is now :func:`int.__str__` to
|
|
||||||
better support the *replacement of existing constants* use-case.
|
|
||||||
:meth:`__format__` was already :func:`int.__format__` for that same reason.
|
|
||||||
|
|
||||||
|
|
||||||
.. class:: EnumCheck
|
.. class:: EnumCheck
|
||||||
|
|
||||||
*EnumCheck* contains the options used by the :func:`verify` decorator to ensure
|
*EnumCheck* contains the options used by the :func:`verify` decorator to ensure
|
||||||
various constraints; failed constraints result in a :exc:`ValueError`.
|
various constraints; failed constraints result in a :exc:`TypeError`.
|
||||||
|
|
||||||
.. attribute:: UNIQUE
|
.. attribute:: UNIQUE
|
||||||
|
|
||||||
|
@ -626,11 +582,11 @@ Data Types
|
||||||
...
|
...
|
||||||
ValueError: invalid Flag 'Color': aliases WHITE and NEON are missing combined values of 0x18 [use enum.show_flag_values(value) for details]
|
ValueError: invalid Flag 'Color': aliases WHITE and NEON are missing combined values of 0x18 [use enum.show_flag_values(value) for details]
|
||||||
|
|
||||||
.. note::
|
.. note::
|
||||||
|
|
||||||
CONTINUOUS and NAMED_FLAGS are designed to work with integer-valued members.
|
CONTINUOUS and NAMED_FLAGS are designed to work with integer-valued members.
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
.. class:: FlagBoundary
|
.. class:: FlagBoundary
|
||||||
|
|
||||||
|
@ -650,7 +606,7 @@ Data Types
|
||||||
>>> StrictFlag(2**2 + 2**4)
|
>>> StrictFlag(2**2 + 2**4)
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
ValueError: <flag 'StrictFlag'> invalid value 20
|
ValueError: StrictFlag: invalid value: 20
|
||||||
given 0b0 10100
|
given 0b0 10100
|
||||||
allowed 0b0 00111
|
allowed 0b0 00111
|
||||||
|
|
||||||
|
@ -665,7 +621,7 @@ Data Types
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
... BLUE = auto()
|
... BLUE = auto()
|
||||||
>>> ConformFlag(2**2 + 2**4)
|
>>> ConformFlag(2**2 + 2**4)
|
||||||
<ConformFlag.BLUE: 4>
|
ConformFlag.BLUE
|
||||||
|
|
||||||
.. attribute:: EJECT
|
.. attribute:: EJECT
|
||||||
|
|
||||||
|
@ -691,52 +647,12 @@ Data Types
|
||||||
... GREEN = auto()
|
... GREEN = auto()
|
||||||
... BLUE = auto()
|
... BLUE = auto()
|
||||||
>>> KeepFlag(2**2 + 2**4)
|
>>> KeepFlag(2**2 + 2**4)
|
||||||
<KeepFlag.BLUE|16: 20>
|
KeepFlag.BLUE|0x10
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
Supported ``__dunder__`` names
|
|
||||||
""""""""""""""""""""""""""""""
|
|
||||||
|
|
||||||
:attr:`__members__` is a read-only ordered mapping of ``member_name``:``member``
|
|
||||||
items. It is only available on the class.
|
|
||||||
|
|
||||||
:meth:`__new__`, if specified, must create and return the enum members; it is
|
|
||||||
also a very good idea to set the member's :attr:`_value_` appropriately. Once
|
|
||||||
all the members are created it is no longer used.
|
|
||||||
|
|
||||||
|
|
||||||
Supported ``_sunder_`` names
|
|
||||||
""""""""""""""""""""""""""""
|
|
||||||
|
|
||||||
- ``_name_`` -- name of the member
|
|
||||||
- ``_value_`` -- value of the member; can be set / modified in ``__new__``
|
|
||||||
|
|
||||||
- ``_missing_`` -- a lookup function used when a value is not found; may be
|
|
||||||
overridden
|
|
||||||
- ``_ignore_`` -- a list of names, either as a :class:`list` or a :class:`str`,
|
|
||||||
that will not be transformed into members, and will be removed from the final
|
|
||||||
class
|
|
||||||
- ``_order_`` -- used in Python 2/3 code to ensure member order is consistent
|
|
||||||
(class attribute, removed during class creation)
|
|
||||||
- ``_generate_next_value_`` -- used to get an appropriate value for an enum
|
|
||||||
member; may be overridden
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
|
|
||||||
For standard :class:`Enum` classes the next value chosen is the last value seen
|
|
||||||
incremented by one.
|
|
||||||
|
|
||||||
For :class:`Flag` classes the next value chosen will be the next highest
|
|
||||||
power-of-two, regardless of the last value seen.
|
|
||||||
|
|
||||||
.. versionadded:: 3.6 ``_missing_``, ``_order_``, ``_generate_next_value_``
|
|
||||||
.. versionadded:: 3.7 ``_ignore_``
|
|
||||||
|
|
||||||
---------------
|
|
||||||
|
|
||||||
Utilities and Decorators
|
Utilities and Decorators
|
||||||
------------------------
|
------------------------
|
||||||
|
|
||||||
|
@ -752,6 +668,15 @@ Utilities and Decorators
|
||||||
``_generate_next_value_`` can be overridden to customize the values used by
|
``_generate_next_value_`` can be overridden to customize the values used by
|
||||||
*auto*.
|
*auto*.
|
||||||
|
|
||||||
|
.. decorator:: global_enum
|
||||||
|
|
||||||
|
A :keyword:`class` decorator specifically for enumerations. It replaces the
|
||||||
|
:meth:`__repr__` method with one that shows *module_name*.*member_name*. It
|
||||||
|
also injects the members, and their aliases, into the global namespace they
|
||||||
|
were defined in.
|
||||||
|
|
||||||
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
.. decorator:: property
|
.. decorator:: property
|
||||||
|
|
||||||
A decorator similar to the built-in *property*, but specifically for
|
A decorator similar to the built-in *property*, but specifically for
|
||||||
|
@ -763,7 +688,7 @@ Utilities and Decorators
|
||||||
*Enum* class, and *Enum* subclasses can define members with the
|
*Enum* class, and *Enum* subclasses can define members with the
|
||||||
names ``value`` and ``name``.
|
names ``value`` and ``name``.
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
.. decorator:: unique
|
.. decorator:: unique
|
||||||
|
|
||||||
|
@ -789,7 +714,7 @@ Utilities and Decorators
|
||||||
:class:`EnumCheck` are used to specify which constraints should be checked
|
:class:`EnumCheck` are used to specify which constraints should be checked
|
||||||
on the decorated enumeration.
|
on the decorated enumeration.
|
||||||
|
|
||||||
.. versionadded:: 3.11
|
.. versionadded:: 3.11
|
||||||
|
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
|
@ -801,20 +726,14 @@ Notes
|
||||||
These three enum types are designed to be drop-in replacements for existing
|
These three enum types are designed to be drop-in replacements for existing
|
||||||
integer- and string-based values; as such, they have extra limitations:
|
integer- and string-based values; as such, they have extra limitations:
|
||||||
|
|
||||||
- ``__str__`` uses the value and not the name of the enum member
|
- ``format()`` will use the value of the enum member, unless ``__str__``
|
||||||
|
has been overridden
|
||||||
|
|
||||||
- ``__format__``, because it uses ``__str__``, will also use the value of
|
- ``StrEnum.__str__`` uses the value and not the name of the enum member
|
||||||
the enum member instead of its name
|
|
||||||
|
|
||||||
If you do not need/want those limitations, you can either create your own
|
If you do not need/want those limitations, you can create your own base
|
||||||
base class by mixing in the ``int`` or ``str`` type yourself::
|
class by mixing in the ``int`` or ``str`` type yourself::
|
||||||
|
|
||||||
>>> from enum import Enum
|
>>> from enum import Enum
|
||||||
>>> class MyIntEnum(int, Enum):
|
>>> class MyIntEnum(int, Enum):
|
||||||
... pass
|
... pass
|
||||||
|
|
||||||
or you can reassign the appropriate :meth:`str`, etc., in your enum::
|
|
||||||
|
|
||||||
>>> from enum import IntEnum
|
|
||||||
>>> class MyIntEnum(IntEnum):
|
|
||||||
... __str__ = IntEnum.__str__
|
|
||||||
|
|
|
@ -2070,7 +2070,7 @@ to speed up repeated connections from the same clients.
|
||||||
:attr:`SSLContext.verify_flags` returns :class:`VerifyFlags` flags:
|
:attr:`SSLContext.verify_flags` returns :class:`VerifyFlags` flags:
|
||||||
|
|
||||||
>>> ssl.create_default_context().verify_flags # doctest: +SKIP
|
>>> ssl.create_default_context().verify_flags # doctest: +SKIP
|
||||||
<VerifyFlags.VERIFY_X509_TRUSTED_FIRST: 32768>
|
ssl.VERIFY_X509_TRUSTED_FIRST
|
||||||
|
|
||||||
.. attribute:: SSLContext.verify_mode
|
.. attribute:: SSLContext.verify_mode
|
||||||
|
|
||||||
|
@ -2082,7 +2082,7 @@ to speed up repeated connections from the same clients.
|
||||||
:attr:`SSLContext.verify_mode` returns :class:`VerifyMode` enum:
|
:attr:`SSLContext.verify_mode` returns :class:`VerifyMode` enum:
|
||||||
|
|
||||||
>>> ssl.create_default_context().verify_mode
|
>>> ssl.create_default_context().verify_mode
|
||||||
<VerifyMode.CERT_REQUIRED: 2>
|
ssl.CERT_REQUIRED
|
||||||
|
|
||||||
.. index:: single: certificates
|
.. index:: single: certificates
|
||||||
|
|
||||||
|
|
573
Lib/enum.py
573
Lib/enum.py
|
@ -1,16 +1,16 @@
|
||||||
import sys
|
import sys
|
||||||
import builtins as bltns
|
|
||||||
from types import MappingProxyType, DynamicClassAttribute
|
from types import MappingProxyType, DynamicClassAttribute
|
||||||
from operator import or_ as _or_
|
from operator import or_ as _or_
|
||||||
from functools import reduce
|
from functools import reduce
|
||||||
|
from builtins import property as _bltin_property, bin as _bltin_bin
|
||||||
|
|
||||||
|
|
||||||
__all__ = [
|
__all__ = [
|
||||||
'EnumType', 'EnumMeta',
|
'EnumType', 'EnumMeta',
|
||||||
'Enum', 'IntEnum', 'StrEnum', 'Flag', 'IntFlag', 'ReprEnum',
|
'Enum', 'IntEnum', 'StrEnum', 'Flag', 'IntFlag',
|
||||||
'auto', 'unique', 'property', 'verify',
|
'auto', 'unique', 'property', 'verify',
|
||||||
'FlagBoundary', 'STRICT', 'CONFORM', 'EJECT', 'KEEP',
|
'FlagBoundary', 'STRICT', 'CONFORM', 'EJECT', 'KEEP',
|
||||||
'global_flag_repr', 'global_enum_repr', 'global_str', 'global_enum',
|
'global_flag_repr', 'global_enum_repr', 'global_enum',
|
||||||
'EnumCheck', 'CONTINUOUS', 'NAMED_FLAGS', 'UNIQUE',
|
'EnumCheck', 'CONTINUOUS', 'NAMED_FLAGS', 'UNIQUE',
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@
|
||||||
# Dummy value for Enum and Flag as there are explicit checks for them
|
# Dummy value for Enum and Flag as there are explicit checks for them
|
||||||
# before they have been created.
|
# before they have been created.
|
||||||
# This is also why there are checks in EnumType like `if Enum is not None`
|
# This is also why there are checks in EnumType like `if Enum is not None`
|
||||||
Enum = Flag = EJECT = _stdlib_enums = ReprEnum = None
|
Enum = Flag = EJECT = None
|
||||||
|
|
||||||
def _is_descriptor(obj):
|
def _is_descriptor(obj):
|
||||||
"""
|
"""
|
||||||
|
@ -116,9 +116,9 @@ def bin(num, max_bits=None):
|
||||||
|
|
||||||
ceiling = 2 ** (num).bit_length()
|
ceiling = 2 ** (num).bit_length()
|
||||||
if num >= 0:
|
if num >= 0:
|
||||||
s = bltns.bin(num + ceiling).replace('1', '0', 1)
|
s = _bltin_bin(num + ceiling).replace('1', '0', 1)
|
||||||
else:
|
else:
|
||||||
s = bltns.bin(~num ^ (ceiling - 1) + ceiling)
|
s = _bltin_bin(~num ^ (ceiling - 1) + ceiling)
|
||||||
sign = s[:3]
|
sign = s[:3]
|
||||||
digits = s[3:]
|
digits = s[3:]
|
||||||
if max_bits is not None:
|
if max_bits is not None:
|
||||||
|
@ -126,19 +126,6 @@ def bin(num, max_bits=None):
|
||||||
digits = (sign[-1] * max_bits + digits)[-max_bits:]
|
digits = (sign[-1] * max_bits + digits)[-max_bits:]
|
||||||
return "%s %s" % (sign, digits)
|
return "%s %s" % (sign, digits)
|
||||||
|
|
||||||
def _dedent(text):
|
|
||||||
"""
|
|
||||||
Like textwrap.dedent. Rewritten because we cannot import textwrap.
|
|
||||||
"""
|
|
||||||
lines = text.split('\n')
|
|
||||||
blanks = 0
|
|
||||||
for i, ch in enumerate(lines[0]):
|
|
||||||
if ch != ' ':
|
|
||||||
break
|
|
||||||
for j, l in enumerate(lines):
|
|
||||||
lines[j] = l[i:]
|
|
||||||
return '\n'.join(lines)
|
|
||||||
|
|
||||||
|
|
||||||
_auto_null = object()
|
_auto_null = object()
|
||||||
class auto:
|
class auto:
|
||||||
|
@ -162,12 +149,22 @@ def __get__(self, instance, ownerclass=None):
|
||||||
return ownerclass._member_map_[self.name]
|
return ownerclass._member_map_[self.name]
|
||||||
except KeyError:
|
except KeyError:
|
||||||
raise AttributeError(
|
raise AttributeError(
|
||||||
'%r has no attribute %r' % (ownerclass, self.name)
|
'%s: no class attribute %r' % (ownerclass.__name__, self.name)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
if self.fget is None:
|
if self.fget is None:
|
||||||
|
# check for member
|
||||||
|
if self.name in ownerclass._member_map_:
|
||||||
|
import warnings
|
||||||
|
warnings.warn(
|
||||||
|
"accessing one member from another is not supported, "
|
||||||
|
" and will be disabled in 3.12",
|
||||||
|
DeprecationWarning,
|
||||||
|
stacklevel=2,
|
||||||
|
)
|
||||||
|
return ownerclass._member_map_[self.name]
|
||||||
raise AttributeError(
|
raise AttributeError(
|
||||||
'%r member has no attribute %r' % (ownerclass, self.name)
|
'%s: no instance attribute %r' % (ownerclass.__name__, self.name)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return self.fget(instance)
|
return self.fget(instance)
|
||||||
|
@ -175,7 +172,7 @@ def __get__(self, instance, ownerclass=None):
|
||||||
def __set__(self, instance, value):
|
def __set__(self, instance, value):
|
||||||
if self.fset is None:
|
if self.fset is None:
|
||||||
raise AttributeError(
|
raise AttributeError(
|
||||||
"<enum %r> cannot set attribute %r" % (self.clsname, self.name)
|
"%s: cannot set instance attribute %r" % (self.clsname, self.name)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return self.fset(instance, value)
|
return self.fset(instance, value)
|
||||||
|
@ -183,7 +180,7 @@ def __set__(self, instance, value):
|
||||||
def __delete__(self, instance):
|
def __delete__(self, instance):
|
||||||
if self.fdel is None:
|
if self.fdel is None:
|
||||||
raise AttributeError(
|
raise AttributeError(
|
||||||
"<enum %r> cannot delete attribute %r" % (self.clsname, self.name)
|
"%s: cannot delete instance attribute %r" % (self.clsname, self.name)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
return self.fdel(instance)
|
return self.fdel(instance)
|
||||||
|
@ -331,7 +328,7 @@ def __setitem__(self, key, value):
|
||||||
elif _is_sunder(key):
|
elif _is_sunder(key):
|
||||||
if key not in (
|
if key not in (
|
||||||
'_order_',
|
'_order_',
|
||||||
'_generate_next_value_', '_numeric_repr_', '_missing_', '_ignore_',
|
'_generate_next_value_', '_missing_', '_ignore_',
|
||||||
'_iter_member_', '_iter_member_by_value_', '_iter_member_by_def_',
|
'_iter_member_', '_iter_member_by_value_', '_iter_member_by_def_',
|
||||||
):
|
):
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
|
@ -361,13 +358,13 @@ def __setitem__(self, key, value):
|
||||||
key = '_order_'
|
key = '_order_'
|
||||||
elif key in self._member_names:
|
elif key in self._member_names:
|
||||||
# descriptor overwriting an enum?
|
# descriptor overwriting an enum?
|
||||||
raise TypeError('%r already defined as %r' % (key, self[key]))
|
raise TypeError('%r already defined as: %r' % (key, self[key]))
|
||||||
elif key in self._ignore:
|
elif key in self._ignore:
|
||||||
pass
|
pass
|
||||||
elif not _is_descriptor(value):
|
elif not _is_descriptor(value):
|
||||||
if key in self:
|
if key in self:
|
||||||
# enum overwriting a descriptor?
|
# enum overwriting a descriptor?
|
||||||
raise TypeError('%r already defined as %r' % (key, self[key]))
|
raise TypeError('%r already defined as: %r' % (key, self[key]))
|
||||||
if isinstance(value, auto):
|
if isinstance(value, auto):
|
||||||
if value.value == _auto_null:
|
if value.value == _auto_null:
|
||||||
value.value = self._generate_next_value(
|
value.value = self._generate_next_value(
|
||||||
|
@ -398,7 +395,7 @@ class EnumType(type):
|
||||||
@classmethod
|
@classmethod
|
||||||
def __prepare__(metacls, cls, bases, **kwds):
|
def __prepare__(metacls, cls, bases, **kwds):
|
||||||
# check that previous enum members do not exist
|
# check that previous enum members do not exist
|
||||||
metacls._check_for_existing_members_(cls, bases)
|
metacls._check_for_existing_members(cls, bases)
|
||||||
# create the namespace dict
|
# create the namespace dict
|
||||||
enum_dict = _EnumDict()
|
enum_dict = _EnumDict()
|
||||||
enum_dict._cls_name = cls
|
enum_dict._cls_name = cls
|
||||||
|
@ -416,10 +413,9 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
# inherited __new__ unless a new __new__ is defined (or the resulting
|
# inherited __new__ unless a new __new__ is defined (or the resulting
|
||||||
# class will fail).
|
# class will fail).
|
||||||
#
|
#
|
||||||
|
# remove any keys listed in _ignore_
|
||||||
if _simple:
|
if _simple:
|
||||||
return super().__new__(metacls, cls, bases, classdict, **kwds)
|
return super().__new__(metacls, cls, bases, classdict, **kwds)
|
||||||
#
|
|
||||||
# remove any keys listed in _ignore_
|
|
||||||
classdict.setdefault('_ignore_', []).append('_ignore_')
|
classdict.setdefault('_ignore_', []).append('_ignore_')
|
||||||
ignore = classdict['_ignore_']
|
ignore = classdict['_ignore_']
|
||||||
for key in ignore:
|
for key in ignore:
|
||||||
|
@ -431,8 +427,8 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
# check for illegal enum names (any others?)
|
# check for illegal enum names (any others?)
|
||||||
invalid_names = set(member_names) & {'mro', ''}
|
invalid_names = set(member_names) & {'mro', ''}
|
||||||
if invalid_names:
|
if invalid_names:
|
||||||
raise ValueError('invalid enum member name(s) '.format(
|
raise ValueError('Invalid enum member name: {0}'.format(
|
||||||
','.join(repr(n) for n in invalid_names)))
|
','.join(invalid_names)))
|
||||||
#
|
#
|
||||||
# adjust the sunders
|
# adjust the sunders
|
||||||
_order_ = classdict.pop('_order_', None)
|
_order_ = classdict.pop('_order_', None)
|
||||||
|
@ -462,8 +458,6 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
classdict['_value2member_map_'] = {}
|
classdict['_value2member_map_'] = {}
|
||||||
classdict['_unhashable_values_'] = []
|
classdict['_unhashable_values_'] = []
|
||||||
classdict['_member_type_'] = member_type
|
classdict['_member_type_'] = member_type
|
||||||
# now set the __repr__ for the value
|
|
||||||
classdict['_value_repr_'] = metacls._find_data_repr_(cls, bases)
|
|
||||||
#
|
#
|
||||||
# Flag structures (will be removed if final class is not a Flag
|
# Flag structures (will be removed if final class is not a Flag
|
||||||
classdict['_boundary_'] = (
|
classdict['_boundary_'] = (
|
||||||
|
@ -473,6 +467,10 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
classdict['_flag_mask_'] = flag_mask
|
classdict['_flag_mask_'] = flag_mask
|
||||||
classdict['_all_bits_'] = 2 ** ((flag_mask).bit_length()) - 1
|
classdict['_all_bits_'] = 2 ** ((flag_mask).bit_length()) - 1
|
||||||
classdict['_inverted_'] = None
|
classdict['_inverted_'] = None
|
||||||
|
#
|
||||||
|
# create a default docstring if one has not been provided
|
||||||
|
if '__doc__' not in classdict:
|
||||||
|
classdict['__doc__'] = 'An enumeration.'
|
||||||
try:
|
try:
|
||||||
exc = None
|
exc = None
|
||||||
enum_class = super().__new__(metacls, cls, bases, classdict, **kwds)
|
enum_class = super().__new__(metacls, cls, bases, classdict, **kwds)
|
||||||
|
@ -483,140 +481,18 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
if exc is not None:
|
if exc is not None:
|
||||||
raise exc
|
raise exc
|
||||||
#
|
#
|
||||||
# update classdict with any changes made by __init_subclass__
|
|
||||||
classdict.update(enum_class.__dict__)
|
|
||||||
#
|
|
||||||
# create a default docstring if one has not been provided
|
|
||||||
if enum_class.__doc__ is None:
|
|
||||||
if not member_names:
|
|
||||||
enum_class.__doc__ = classdict['__doc__'] = _dedent("""\
|
|
||||||
Create a collection of name/value pairs.
|
|
||||||
|
|
||||||
Example enumeration:
|
|
||||||
|
|
||||||
>>> class Color(Enum):
|
|
||||||
... RED = 1
|
|
||||||
... BLUE = 2
|
|
||||||
... GREEN = 3
|
|
||||||
|
|
||||||
Access them by:
|
|
||||||
|
|
||||||
- attribute access::
|
|
||||||
|
|
||||||
>>> Color.RED
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
- value lookup:
|
|
||||||
|
|
||||||
>>> Color(1)
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
- name lookup:
|
|
||||||
|
|
||||||
>>> Color['RED']
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
Enumerations can be iterated over, and know how many members they have:
|
|
||||||
|
|
||||||
>>> len(Color)
|
|
||||||
3
|
|
||||||
|
|
||||||
>>> list(Color)
|
|
||||||
[<Color.RED: 1>, <Color.BLUE: 2>, <Color.GREEN: 3>]
|
|
||||||
|
|
||||||
Methods can be added to enumerations, and members can have their own
|
|
||||||
attributes -- see the documentation for details.
|
|
||||||
""")
|
|
||||||
else:
|
|
||||||
member = list(enum_class)[0]
|
|
||||||
enum_length = len(enum_class)
|
|
||||||
cls_name = enum_class.__name__
|
|
||||||
if enum_length == 1:
|
|
||||||
list_line = 'list(%s)' % cls_name
|
|
||||||
list_repr = '[<%s.%s: %r>]' % (cls_name, member.name, member.value)
|
|
||||||
elif enum_length == 2:
|
|
||||||
member2 = list(enum_class)[1]
|
|
||||||
list_line = 'list(%s)' % cls_name
|
|
||||||
list_repr = '[<%s.%s: %r>, <%s.%s: %r>]' % (
|
|
||||||
cls_name, member.name, member.value,
|
|
||||||
cls_name, member2.name, member2.value,
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
member2 = list(enum_class)[1]
|
|
||||||
member3 = list(enum_class)[2]
|
|
||||||
list_line = 'list(%s)%s' % (cls_name, ('','[:3]')[enum_length > 3])
|
|
||||||
list_repr = '[<%s.%s: %r>, <%s.%s: %r>, <%s.%s: %r>]' % (
|
|
||||||
cls_name, member.name, member.value,
|
|
||||||
cls_name, member2.name, member2.value,
|
|
||||||
cls_name, member3.name, member3.value,
|
|
||||||
)
|
|
||||||
enum_class.__doc__ = classdict['__doc__'] = _dedent("""\
|
|
||||||
A collection of name/value pairs.
|
|
||||||
|
|
||||||
Access them by:
|
|
||||||
|
|
||||||
- attribute access::
|
|
||||||
|
|
||||||
>>> %s.%s
|
|
||||||
<%s.%s: %r>
|
|
||||||
|
|
||||||
- value lookup:
|
|
||||||
|
|
||||||
>>> %s(%r)
|
|
||||||
<%s.%s: %r>
|
|
||||||
|
|
||||||
- name lookup:
|
|
||||||
|
|
||||||
>>> %s[%r]
|
|
||||||
<%s.%s: %r>
|
|
||||||
|
|
||||||
Enumerations can be iterated over, and know how many members they have:
|
|
||||||
|
|
||||||
>>> len(%s)
|
|
||||||
%r
|
|
||||||
|
|
||||||
>>> %s
|
|
||||||
%s
|
|
||||||
|
|
||||||
Methods can be added to enumerations, and members can have their own
|
|
||||||
attributes -- see the documentation for details.
|
|
||||||
"""
|
|
||||||
% (cls_name, member.name,
|
|
||||||
cls_name, member.name, member.value,
|
|
||||||
cls_name, member.value,
|
|
||||||
cls_name, member.name, member.value,
|
|
||||||
cls_name, member.name,
|
|
||||||
cls_name, member.name, member.value,
|
|
||||||
cls_name, enum_length,
|
|
||||||
list_line, list_repr,
|
|
||||||
))
|
|
||||||
#
|
|
||||||
# double check that repr and friends are not the mixin's or various
|
# double check that repr and friends are not the mixin's or various
|
||||||
# things break (such as pickle)
|
# things break (such as pickle)
|
||||||
# however, if the method is defined in the Enum itself, don't replace
|
# however, if the method is defined in the Enum itself, don't replace
|
||||||
# it
|
# it
|
||||||
#
|
|
||||||
# Also, special handling for ReprEnum
|
|
||||||
if ReprEnum is not None and ReprEnum in bases:
|
|
||||||
if member_type is object:
|
|
||||||
raise TypeError(
|
|
||||||
'ReprEnum subclasses must be mixed with a data type (i.e.'
|
|
||||||
' int, str, float, etc.)'
|
|
||||||
)
|
|
||||||
if '__format__' not in classdict:
|
|
||||||
enum_class.__format__ = member_type.__format__
|
|
||||||
classdict['__format__'] = enum_class.__format__
|
|
||||||
if '__str__' not in classdict:
|
|
||||||
method = member_type.__str__
|
|
||||||
if method is object.__str__:
|
|
||||||
# if member_type does not define __str__, object.__str__ will use
|
|
||||||
# its __repr__ instead, so we'll also use its __repr__
|
|
||||||
method = member_type.__repr__
|
|
||||||
enum_class.__str__ = method
|
|
||||||
classdict['__str__'] = enum_class.__str__
|
|
||||||
for name in ('__repr__', '__str__', '__format__', '__reduce_ex__'):
|
for name in ('__repr__', '__str__', '__format__', '__reduce_ex__'):
|
||||||
if name not in classdict:
|
if name in classdict:
|
||||||
setattr(enum_class, name, getattr(first_enum, name))
|
continue
|
||||||
|
class_method = getattr(enum_class, name)
|
||||||
|
obj_method = getattr(member_type, name, None)
|
||||||
|
enum_method = getattr(first_enum, name, None)
|
||||||
|
if obj_method is not None and obj_method is class_method:
|
||||||
|
setattr(enum_class, name, enum_method)
|
||||||
#
|
#
|
||||||
# replace any other __new__ with our own (as long as Enum is not None,
|
# replace any other __new__ with our own (as long as Enum is not None,
|
||||||
# anyway) -- again, this is to support pickle
|
# anyway) -- again, this is to support pickle
|
||||||
|
@ -687,13 +563,13 @@ def __new__(metacls, cls, bases, classdict, *, boundary=None, _simple=False, **k
|
||||||
# _order_ step 4: verify that _order_ and _member_names_ match
|
# _order_ step 4: verify that _order_ and _member_names_ match
|
||||||
if _order_ != enum_class._member_names_:
|
if _order_ != enum_class._member_names_:
|
||||||
raise TypeError(
|
raise TypeError(
|
||||||
'member order does not match _order_:\n %r\n %r'
|
'member order does not match _order_:\n%r\n%r'
|
||||||
% (enum_class._member_names_, _order_)
|
% (enum_class._member_names_, _order_)
|
||||||
)
|
)
|
||||||
#
|
#
|
||||||
return enum_class
|
return enum_class
|
||||||
|
|
||||||
def __bool__(cls):
|
def __bool__(self):
|
||||||
"""
|
"""
|
||||||
classes/types should always be True.
|
classes/types should always be True.
|
||||||
"""
|
"""
|
||||||
|
@ -738,13 +614,6 @@ def __call__(cls, value, names=None, *, module=None, qualname=None, type=None, s
|
||||||
)
|
)
|
||||||
|
|
||||||
def __contains__(cls, member):
|
def __contains__(cls, member):
|
||||||
"""
|
|
||||||
Return True if member is a member of this enum
|
|
||||||
raises TypeError if member is not an enum member
|
|
||||||
|
|
||||||
note: in 3.12 TypeError will no longer be raised, and True will also be
|
|
||||||
returned if member is the value of a member in this enum
|
|
||||||
"""
|
|
||||||
if not isinstance(member, Enum):
|
if not isinstance(member, Enum):
|
||||||
import warnings
|
import warnings
|
||||||
warnings.warn(
|
warnings.warn(
|
||||||
|
@ -762,33 +631,60 @@ def __delattr__(cls, attr):
|
||||||
# nicer error message when someone tries to delete an attribute
|
# nicer error message when someone tries to delete an attribute
|
||||||
# (see issue19025).
|
# (see issue19025).
|
||||||
if attr in cls._member_map_:
|
if attr in cls._member_map_:
|
||||||
raise AttributeError("%r cannot delete member %r." % (cls.__name__, attr))
|
raise AttributeError("%s: cannot delete Enum member %r." % (cls.__name__, attr))
|
||||||
super().__delattr__(attr)
|
super().__delattr__(attr)
|
||||||
|
|
||||||
def __dir__(cls):
|
def __dir__(self):
|
||||||
# TODO: check for custom __init__, __new__, __format__, __repr__, __str__, __init_subclass__
|
# Start off with the desired result for dir(Enum)
|
||||||
# on object-based enums
|
cls_dir = {'__class__', '__doc__', '__members__', '__module__'}
|
||||||
if cls._member_type_ is object:
|
add_to_dir = cls_dir.add
|
||||||
interesting = set(cls._member_names_)
|
mro = self.__mro__
|
||||||
if cls._new_member_ is not object.__new__:
|
this_module = globals().values()
|
||||||
interesting.add('__new__')
|
is_from_this_module = lambda cls: any(cls is thing for thing in this_module)
|
||||||
if cls.__init_subclass__ is not object.__init_subclass__:
|
first_enum_base = next(cls for cls in mro if is_from_this_module(cls))
|
||||||
interesting.add('__init_subclass__')
|
enum_dict = Enum.__dict__
|
||||||
for method in ('__init__', '__format__', '__repr__', '__str__'):
|
sentinel = object()
|
||||||
if getattr(cls, method) not in (getattr(Enum, method), getattr(Flag, method)):
|
# special-case __new__
|
||||||
interesting.add(method)
|
ignored = {'__new__', *filter(_is_sunder, enum_dict)}
|
||||||
return sorted(set([
|
add_to_ignored = ignored.add
|
||||||
'__class__', '__contains__', '__doc__', '__getitem__',
|
|
||||||
'__iter__', '__len__', '__members__', '__module__',
|
# We want these added to __dir__
|
||||||
'__name__', '__qualname__',
|
# if and only if they have been user-overridden
|
||||||
]) | interesting
|
enum_dunders = set(filter(_is_dunder, enum_dict))
|
||||||
)
|
|
||||||
|
for cls in mro:
|
||||||
|
# Ignore any classes defined in this module
|
||||||
|
if cls is object or is_from_this_module(cls):
|
||||||
|
continue
|
||||||
|
|
||||||
|
cls_lookup = cls.__dict__
|
||||||
|
|
||||||
|
# If not an instance of EnumType,
|
||||||
|
# ensure all attributes excluded from that class's `dir()` are ignored here.
|
||||||
|
if not isinstance(cls, EnumType):
|
||||||
|
cls_lookup = set(cls_lookup).intersection(dir(cls))
|
||||||
|
|
||||||
|
for attr_name in cls_lookup:
|
||||||
|
# Already seen it? Carry on
|
||||||
|
if attr_name in cls_dir or attr_name in ignored:
|
||||||
|
continue
|
||||||
|
# Sunders defined in Enum.__dict__ are already in `ignored`,
|
||||||
|
# But sunders defined in a subclass won't be (we want all sunders excluded).
|
||||||
|
elif _is_sunder(attr_name):
|
||||||
|
add_to_ignored(attr_name)
|
||||||
|
# Not an "enum dunder"? Add it to dir() output.
|
||||||
|
elif attr_name not in enum_dunders:
|
||||||
|
add_to_dir(attr_name)
|
||||||
|
# Is an "enum dunder", and is defined by a class from enum.py? Ignore it.
|
||||||
|
elif getattr(self, attr_name) is getattr(first_enum_base, attr_name, sentinel):
|
||||||
|
add_to_ignored(attr_name)
|
||||||
|
# Is an "enum dunder", and is either user-defined or defined by a mixin class?
|
||||||
|
# Add it to dir() output.
|
||||||
else:
|
else:
|
||||||
# return whatever mixed-in data type has
|
add_to_dir(attr_name)
|
||||||
return sorted(set(
|
|
||||||
dir(cls._member_type_)
|
# sort the output before returning it, so that the result is deterministic.
|
||||||
+ cls._member_names_
|
return sorted(cls_dir)
|
||||||
))
|
|
||||||
|
|
||||||
def __getattr__(cls, name):
|
def __getattr__(cls, name):
|
||||||
"""
|
"""
|
||||||
|
@ -807,24 +703,18 @@ def __getattr__(cls, name):
|
||||||
raise AttributeError(name) from None
|
raise AttributeError(name) from None
|
||||||
|
|
||||||
def __getitem__(cls, name):
|
def __getitem__(cls, name):
|
||||||
"""
|
|
||||||
Return the member matching `name`.
|
|
||||||
"""
|
|
||||||
return cls._member_map_[name]
|
return cls._member_map_[name]
|
||||||
|
|
||||||
def __iter__(cls):
|
def __iter__(cls):
|
||||||
"""
|
"""
|
||||||
Return members in definition order.
|
Returns members in definition order.
|
||||||
"""
|
"""
|
||||||
return (cls._member_map_[name] for name in cls._member_names_)
|
return (cls._member_map_[name] for name in cls._member_names_)
|
||||||
|
|
||||||
def __len__(cls):
|
def __len__(cls):
|
||||||
"""
|
|
||||||
Return the number of members (no aliases)
|
|
||||||
"""
|
|
||||||
return len(cls._member_names_)
|
return len(cls._member_names_)
|
||||||
|
|
||||||
@bltns.property
|
@_bltin_property
|
||||||
def __members__(cls):
|
def __members__(cls):
|
||||||
"""
|
"""
|
||||||
Returns a mapping of member name->value.
|
Returns a mapping of member name->value.
|
||||||
|
@ -842,7 +732,7 @@ def __repr__(cls):
|
||||||
|
|
||||||
def __reversed__(cls):
|
def __reversed__(cls):
|
||||||
"""
|
"""
|
||||||
Return members in reverse definition order.
|
Returns members in reverse definition order.
|
||||||
"""
|
"""
|
||||||
return (cls._member_map_[name] for name in reversed(cls._member_names_))
|
return (cls._member_map_[name] for name in reversed(cls._member_names_))
|
||||||
|
|
||||||
|
@ -856,7 +746,7 @@ def __setattr__(cls, name, value):
|
||||||
"""
|
"""
|
||||||
member_map = cls.__dict__.get('_member_map_', {})
|
member_map = cls.__dict__.get('_member_map_', {})
|
||||||
if name in member_map:
|
if name in member_map:
|
||||||
raise AttributeError('cannot reassign member %r' % (name, ))
|
raise AttributeError('Cannot reassign member %r.' % (name, ))
|
||||||
super().__setattr__(name, value)
|
super().__setattr__(name, value)
|
||||||
|
|
||||||
def _create_(cls, class_name, names, *, module=None, qualname=None, type=None, start=1, boundary=None):
|
def _create_(cls, class_name, names, *, module=None, qualname=None, type=None, start=1, boundary=None):
|
||||||
|
@ -911,7 +801,8 @@ def _create_(cls, class_name, names, *, module=None, qualname=None, type=None, s
|
||||||
|
|
||||||
return metacls.__new__(metacls, class_name, bases, classdict, boundary=boundary)
|
return metacls.__new__(metacls, class_name, bases, classdict, boundary=boundary)
|
||||||
|
|
||||||
def _convert_(cls, name, module, filter, source=None, *, boundary=None, as_global=False):
|
def _convert_(cls, name, module, filter, source=None, *, boundary=None):
|
||||||
|
|
||||||
"""
|
"""
|
||||||
Create a new Enum subclass that replaces a collection of global constants
|
Create a new Enum subclass that replaces a collection of global constants
|
||||||
"""
|
"""
|
||||||
|
@ -943,25 +834,22 @@ def _convert_(cls, name, module, filter, source=None, *, boundary=None, as_globa
|
||||||
tmp_cls = type(name, (object, ), body)
|
tmp_cls = type(name, (object, ), body)
|
||||||
cls = _simple_enum(etype=cls, boundary=boundary or KEEP)(tmp_cls)
|
cls = _simple_enum(etype=cls, boundary=boundary or KEEP)(tmp_cls)
|
||||||
cls.__reduce_ex__ = _reduce_ex_by_global_name
|
cls.__reduce_ex__ = _reduce_ex_by_global_name
|
||||||
if as_global:
|
|
||||||
global_enum(cls)
|
global_enum(cls)
|
||||||
else:
|
|
||||||
sys.modules[cls.__module__].__dict__.update(cls.__members__)
|
|
||||||
module_globals[name] = cls
|
module_globals[name] = cls
|
||||||
return cls
|
return cls
|
||||||
|
|
||||||
@classmethod
|
@staticmethod
|
||||||
def _check_for_existing_members_(mcls, class_name, bases):
|
def _check_for_existing_members(class_name, bases):
|
||||||
for chain in bases:
|
for chain in bases:
|
||||||
for base in chain.__mro__:
|
for base in chain.__mro__:
|
||||||
if issubclass(base, Enum) and base._member_names_:
|
if issubclass(base, Enum) and base._member_names_:
|
||||||
raise TypeError(
|
raise TypeError(
|
||||||
"<enum %r> cannot extend %r"
|
"%s: cannot extend enumeration %r"
|
||||||
% (class_name, base)
|
% (class_name, base.__name__)
|
||||||
)
|
)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _get_mixins_(mcls, class_name, bases):
|
def _get_mixins_(cls, class_name, bases):
|
||||||
"""
|
"""
|
||||||
Returns the type for creating enum members, and the first inherited
|
Returns the type for creating enum members, and the first inherited
|
||||||
enum class.
|
enum class.
|
||||||
|
@ -971,33 +859,7 @@ def _get_mixins_(mcls, class_name, bases):
|
||||||
if not bases:
|
if not bases:
|
||||||
return object, Enum
|
return object, Enum
|
||||||
|
|
||||||
mcls._check_for_existing_members_(class_name, bases)
|
def _find_data_type(bases):
|
||||||
|
|
||||||
# ensure final parent class is an Enum derivative, find any concrete
|
|
||||||
# data type, and check that Enum has no members
|
|
||||||
first_enum = bases[-1]
|
|
||||||
if not issubclass(first_enum, Enum):
|
|
||||||
raise TypeError("new enumerations should be created as "
|
|
||||||
"`EnumName([mixin_type, ...] [data_type,] enum_type)`")
|
|
||||||
member_type = mcls._find_data_type_(class_name, bases) or object
|
|
||||||
return member_type, first_enum
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _find_data_repr_(mcls, class_name, bases):
|
|
||||||
for chain in bases:
|
|
||||||
for base in chain.__mro__:
|
|
||||||
if base is object:
|
|
||||||
continue
|
|
||||||
elif issubclass(base, Enum):
|
|
||||||
# if we hit an Enum, use it's _value_repr_
|
|
||||||
return base._value_repr_
|
|
||||||
elif '__repr__' in base.__dict__:
|
|
||||||
# this is our data repr
|
|
||||||
return base.__dict__['__repr__']
|
|
||||||
return None
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def _find_data_type_(mcls, class_name, bases):
|
|
||||||
data_types = set()
|
data_types = set()
|
||||||
for chain in bases:
|
for chain in bases:
|
||||||
candidate = None
|
candidate = None
|
||||||
|
@ -1016,14 +878,24 @@ def _find_data_type_(mcls, class_name, bases):
|
||||||
else:
|
else:
|
||||||
candidate = candidate or base
|
candidate = candidate or base
|
||||||
if len(data_types) > 1:
|
if len(data_types) > 1:
|
||||||
raise TypeError('too many data types for %r: %r' % (class_name, data_types))
|
raise TypeError('%r: too many data types: %r' % (class_name, data_types))
|
||||||
elif data_types:
|
elif data_types:
|
||||||
return data_types.pop()
|
return data_types.pop()
|
||||||
else:
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
@classmethod
|
# ensure final parent class is an Enum derivative, find any concrete
|
||||||
def _find_new_(mcls, classdict, member_type, first_enum):
|
# data type, and check that Enum has no members
|
||||||
|
first_enum = bases[-1]
|
||||||
|
if not issubclass(first_enum, Enum):
|
||||||
|
raise TypeError("new enumerations should be created as "
|
||||||
|
"`EnumName([mixin_type, ...] [data_type,] enum_type)`")
|
||||||
|
cls._check_for_existing_members(class_name, bases)
|
||||||
|
member_type = _find_data_type(bases) or object
|
||||||
|
return member_type, first_enum
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def _find_new_(classdict, member_type, first_enum):
|
||||||
"""
|
"""
|
||||||
Returns the __new__ to be used for creating the enum members.
|
Returns the __new__ to be used for creating the enum members.
|
||||||
|
|
||||||
|
@ -1071,42 +943,9 @@ def _find_new_(mcls, classdict, member_type, first_enum):
|
||||||
|
|
||||||
class Enum(metaclass=EnumType):
|
class Enum(metaclass=EnumType):
|
||||||
"""
|
"""
|
||||||
Create a collection of name/value pairs.
|
Generic enumeration.
|
||||||
|
|
||||||
Example enumeration:
|
Derive from this class to define new enumerations.
|
||||||
|
|
||||||
>>> class Color(Enum):
|
|
||||||
... RED = 1
|
|
||||||
... BLUE = 2
|
|
||||||
... GREEN = 3
|
|
||||||
|
|
||||||
Access them by:
|
|
||||||
|
|
||||||
- attribute access::
|
|
||||||
|
|
||||||
>>> Color.RED
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
- value lookup:
|
|
||||||
|
|
||||||
>>> Color(1)
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
- name lookup:
|
|
||||||
|
|
||||||
>>> Color['RED']
|
|
||||||
<Color.RED: 1>
|
|
||||||
|
|
||||||
Enumerations can be iterated over, and know how many members they have:
|
|
||||||
|
|
||||||
>>> len(Color)
|
|
||||||
3
|
|
||||||
|
|
||||||
>>> list(Color)
|
|
||||||
[<Color.RED: 1>, <Color.BLUE: 2>, <Color.GREEN: 3>]
|
|
||||||
|
|
||||||
Methods can be added to enumerations, and members can have their own
|
|
||||||
attributes -- see the documentation for details.
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __new__(cls, value):
|
def __new__(cls, value):
|
||||||
|
@ -1160,9 +999,6 @@ def __new__(cls, value):
|
||||||
exc = None
|
exc = None
|
||||||
ve_exc = None
|
ve_exc = None
|
||||||
|
|
||||||
def __init__(self, *args, **kwds):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _generate_next_value_(name, start, count, last_values):
|
def _generate_next_value_(name, start, count, last_values):
|
||||||
"""
|
"""
|
||||||
Generate the next value when not given.
|
Generate the next value when not given.
|
||||||
|
@ -1185,44 +1021,47 @@ def _missing_(cls, value):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
v_repr = self.__class__._value_repr_ or self._value_.__class__.__repr__
|
return "%s.%s" % ( self.__class__.__name__, self._name_)
|
||||||
return "<%s.%s: %s>" % (self.__class__.__name__, self._name_, v_repr(self._value_))
|
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return "%s.%s" % (self.__class__.__name__, self._name_, )
|
return "%s" % (self._name_, )
|
||||||
|
|
||||||
def __dir__(self):
|
def __dir__(self):
|
||||||
"""
|
"""
|
||||||
Returns all members and all public methods
|
Returns all members and all public methods
|
||||||
"""
|
"""
|
||||||
if self.__class__._member_type_ is object:
|
cls = type(self)
|
||||||
interesting = set(['__class__', '__doc__', '__eq__', '__hash__', '__module__', 'name', 'value'])
|
to_exclude = {'__members__', '__init__', '__new__', *cls._member_names_}
|
||||||
else:
|
filtered_self_dict = (name for name in self.__dict__ if not name.startswith('_'))
|
||||||
interesting = set(object.__dir__(self))
|
return sorted({'name', 'value', *dir(cls), *filtered_self_dict} - to_exclude)
|
||||||
for name in getattr(self, '__dict__', []):
|
|
||||||
if name[0] != '_':
|
|
||||||
interesting.add(name)
|
|
||||||
for cls in self.__class__.mro():
|
|
||||||
for name, obj in cls.__dict__.items():
|
|
||||||
if name[0] == '_':
|
|
||||||
continue
|
|
||||||
if isinstance(obj, property):
|
|
||||||
# that's an enum.property
|
|
||||||
if obj.fget is not None or name not in self._member_map_:
|
|
||||||
interesting.add(name)
|
|
||||||
else:
|
|
||||||
# in case it was added by `dir(self)`
|
|
||||||
interesting.discard(name)
|
|
||||||
else:
|
|
||||||
interesting.add(name)
|
|
||||||
names = sorted(
|
|
||||||
set(['__class__', '__doc__', '__eq__', '__hash__', '__module__'])
|
|
||||||
| interesting
|
|
||||||
)
|
|
||||||
return names
|
|
||||||
|
|
||||||
def __format__(self, format_spec):
|
def __format__(self, format_spec):
|
||||||
return str.__format__(str(self), format_spec)
|
"""
|
||||||
|
Returns format using actual value type unless __str__ has been overridden.
|
||||||
|
"""
|
||||||
|
# mixed-in Enums should use the mixed-in type's __format__, otherwise
|
||||||
|
# we can get strange results with the Enum name showing up instead of
|
||||||
|
# the value
|
||||||
|
#
|
||||||
|
# pure Enum branch, or branch with __str__ explicitly overridden
|
||||||
|
str_overridden = type(self).__str__ not in (Enum.__str__, IntEnum.__str__, Flag.__str__)
|
||||||
|
if self._member_type_ is object or str_overridden:
|
||||||
|
cls = str
|
||||||
|
val = str(self)
|
||||||
|
# mix-in branch
|
||||||
|
else:
|
||||||
|
if not format_spec or format_spec in ('{}','{:}'):
|
||||||
|
import warnings
|
||||||
|
warnings.warn(
|
||||||
|
"in 3.12 format() will use the enum member, not the enum member's value;\n"
|
||||||
|
"use a format specifier, such as :d for an integer-based Enum, to maintain "
|
||||||
|
"the current display",
|
||||||
|
DeprecationWarning,
|
||||||
|
stacklevel=2,
|
||||||
|
)
|
||||||
|
cls = self._member_type_
|
||||||
|
val = self._value_
|
||||||
|
return cls.__format__(val, format_spec)
|
||||||
|
|
||||||
def __hash__(self):
|
def __hash__(self):
|
||||||
return hash(self._name_)
|
return hash(self._name_)
|
||||||
|
@ -1249,25 +1088,34 @@ def value(self):
|
||||||
return self._value_
|
return self._value_
|
||||||
|
|
||||||
|
|
||||||
class ReprEnum(Enum):
|
class IntEnum(int, Enum):
|
||||||
"""
|
|
||||||
Only changes the repr(), leaving str() and format() to the mixed-in type.
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
class IntEnum(int, ReprEnum):
|
|
||||||
"""
|
"""
|
||||||
Enum where members are also (and must be) ints
|
Enum where members are also (and must be) ints
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "%s" % (self._name_, )
|
||||||
|
|
||||||
class StrEnum(str, ReprEnum):
|
def __format__(self, format_spec):
|
||||||
|
"""
|
||||||
|
Returns format using actual value unless __str__ has been overridden.
|
||||||
|
"""
|
||||||
|
str_overridden = type(self).__str__ != IntEnum.__str__
|
||||||
|
if str_overridden:
|
||||||
|
cls = str
|
||||||
|
val = str(self)
|
||||||
|
else:
|
||||||
|
cls = self._member_type_
|
||||||
|
val = self._value_
|
||||||
|
return cls.__format__(val, format_spec)
|
||||||
|
|
||||||
|
|
||||||
|
class StrEnum(str, Enum):
|
||||||
"""
|
"""
|
||||||
Enum where members are also (and must be) strings
|
Enum where members are also (and must be) strings
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __new__(cls, *values):
|
def __new__(cls, *values):
|
||||||
"values must already be of type `str`"
|
|
||||||
if len(values) > 3:
|
if len(values) > 3:
|
||||||
raise TypeError('too many arguments for str(): %r' % (values, ))
|
raise TypeError('too many arguments for str(): %r' % (values, ))
|
||||||
if len(values) == 1:
|
if len(values) == 1:
|
||||||
|
@ -1287,6 +1135,10 @@ def __new__(cls, *values):
|
||||||
member._value_ = value
|
member._value_ = value
|
||||||
return member
|
return member
|
||||||
|
|
||||||
|
__str__ = str.__str__
|
||||||
|
|
||||||
|
__format__ = str.__format__
|
||||||
|
|
||||||
def _generate_next_value_(name, start, count, last_values):
|
def _generate_next_value_(name, start, count, last_values):
|
||||||
"""
|
"""
|
||||||
Return the lower-cased version of the member name.
|
Return the lower-cased version of the member name.
|
||||||
|
@ -1317,8 +1169,6 @@ class Flag(Enum, boundary=STRICT):
|
||||||
Support for flags
|
Support for flags
|
||||||
"""
|
"""
|
||||||
|
|
||||||
_numeric_repr_ = repr
|
|
||||||
|
|
||||||
def _generate_next_value_(name, start, count, last_values):
|
def _generate_next_value_(name, start, count, last_values):
|
||||||
"""
|
"""
|
||||||
Generate the next value when not given.
|
Generate the next value when not given.
|
||||||
|
@ -1334,7 +1184,7 @@ def _generate_next_value_(name, start, count, last_values):
|
||||||
try:
|
try:
|
||||||
high_bit = _high_bit(last_value)
|
high_bit = _high_bit(last_value)
|
||||||
except Exception:
|
except Exception:
|
||||||
raise TypeError('invalid flag value %r' % last_value) from None
|
raise TypeError('Invalid Flag value: %r' % last_value) from None
|
||||||
return 2 ** (high_bit+1)
|
return 2 ** (high_bit+1)
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
|
@ -1382,8 +1232,8 @@ def _missing_(cls, value):
|
||||||
if cls._boundary_ is STRICT:
|
if cls._boundary_ is STRICT:
|
||||||
max_bits = max(value.bit_length(), flag_mask.bit_length())
|
max_bits = max(value.bit_length(), flag_mask.bit_length())
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
"%r invalid value %r\n given %s\n allowed %s" % (
|
"%s: invalid value: %r\n given %s\n allowed %s" % (
|
||||||
cls, value, bin(value, max_bits), bin(flag_mask, max_bits),
|
cls.__name__, value, bin(value, max_bits), bin(flag_mask, max_bits),
|
||||||
))
|
))
|
||||||
elif cls._boundary_ is CONFORM:
|
elif cls._boundary_ is CONFORM:
|
||||||
value = value & flag_mask
|
value = value & flag_mask
|
||||||
|
@ -1397,7 +1247,7 @@ def _missing_(cls, value):
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
'%r unknown flag boundary %r' % (cls, cls._boundary_, )
|
'unknown flag boundary: %r' % (cls._boundary_, )
|
||||||
)
|
)
|
||||||
if value < 0:
|
if value < 0:
|
||||||
neg_value = value
|
neg_value = value
|
||||||
|
@ -1424,7 +1274,7 @@ def _missing_(cls, value):
|
||||||
m._name_ for m in cls._iter_member_(member_value)
|
m._name_ for m in cls._iter_member_(member_value)
|
||||||
])
|
])
|
||||||
if unknown:
|
if unknown:
|
||||||
pseudo_member._name_ += '|%s' % cls._numeric_repr_(unknown)
|
pseudo_member._name_ += '|0x%x' % unknown
|
||||||
else:
|
else:
|
||||||
pseudo_member._name_ = None
|
pseudo_member._name_ = None
|
||||||
# use setdefault in case another thread already created a composite
|
# use setdefault in case another thread already created a composite
|
||||||
|
@ -1442,8 +1292,10 @@ def __contains__(self, other):
|
||||||
"""
|
"""
|
||||||
if not isinstance(other, self.__class__):
|
if not isinstance(other, self.__class__):
|
||||||
raise TypeError(
|
raise TypeError(
|
||||||
"unsupported operand type(s) for 'in': %r and %r" % (
|
"unsupported operand type(s) for 'in': '%s' and '%s'" % (
|
||||||
type(other).__qualname__, self.__class__.__qualname__))
|
type(other).__qualname__, self.__class__.__qualname__))
|
||||||
|
if other._value_ == 0 or self._value_ == 0:
|
||||||
|
return False
|
||||||
return other._value_ & self._value_ == other._value_
|
return other._value_ & self._value_ == other._value_
|
||||||
|
|
||||||
def __iter__(self):
|
def __iter__(self):
|
||||||
|
@ -1457,18 +1309,27 @@ def __len__(self):
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
cls_name = self.__class__.__name__
|
cls_name = self.__class__.__name__
|
||||||
v_repr = self.__class__._value_repr_ or self._value_.__class__.__repr__
|
|
||||||
if self._name_ is None:
|
if self._name_ is None:
|
||||||
return "<%s: %s>" % (cls_name, v_repr(self._value_))
|
return "0x%x" % (self._value_, )
|
||||||
|
if _is_single_bit(self._value_):
|
||||||
|
return '%s.%s' % (cls_name, self._name_)
|
||||||
|
if self._boundary_ is not FlagBoundary.KEEP:
|
||||||
|
return '%s.' % cls_name + ('|%s.' % cls_name).join(self.name.split('|'))
|
||||||
else:
|
else:
|
||||||
return "<%s.%s: %s>" % (cls_name, self._name_, v_repr(self._value_))
|
name = []
|
||||||
|
for n in self._name_.split('|'):
|
||||||
|
if n.startswith('0'):
|
||||||
|
name.append(n)
|
||||||
|
else:
|
||||||
|
name.append('%s.%s' % (cls_name, n))
|
||||||
|
return '|'.join(name)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
cls_name = self.__class__.__name__
|
cls = self.__class__
|
||||||
if self._name_ is None:
|
if self._name_ is None:
|
||||||
return '%s(%r)' % (cls_name, self._value_)
|
return '%s(%x)' % (cls.__name__, self._value_)
|
||||||
else:
|
else:
|
||||||
return "%s.%s" % (cls_name, self._name_)
|
return self._name_
|
||||||
|
|
||||||
def __bool__(self):
|
def __bool__(self):
|
||||||
return bool(self._value_)
|
return bool(self._value_)
|
||||||
|
@ -1501,11 +1362,20 @@ def __invert__(self):
|
||||||
return self._inverted_
|
return self._inverted_
|
||||||
|
|
||||||
|
|
||||||
class IntFlag(int, ReprEnum, Flag, boundary=EJECT):
|
class IntFlag(int, Flag, boundary=EJECT):
|
||||||
"""
|
"""
|
||||||
Support for integer-based Flags
|
Support for integer-based Flags
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
def __format__(self, format_spec):
|
||||||
|
"""
|
||||||
|
Returns format using actual value unless __str__ has been overridden.
|
||||||
|
"""
|
||||||
|
str_overridden = type(self).__str__ != Flag.__str__
|
||||||
|
value = self
|
||||||
|
if not str_overridden:
|
||||||
|
value = self._value_
|
||||||
|
return int.__format__(value, format_spec)
|
||||||
|
|
||||||
def __or__(self, other):
|
def __or__(self, other):
|
||||||
if isinstance(other, self.__class__):
|
if isinstance(other, self.__class__):
|
||||||
|
@ -1542,7 +1412,6 @@ def __xor__(self, other):
|
||||||
__rxor__ = __xor__
|
__rxor__ = __xor__
|
||||||
__invert__ = Flag.__invert__
|
__invert__ = Flag.__invert__
|
||||||
|
|
||||||
|
|
||||||
def _high_bit(value):
|
def _high_bit(value):
|
||||||
"""
|
"""
|
||||||
returns index of highest bit, or -1 if value is zero or negative
|
returns index of highest bit, or -1 if value is zero or negative
|
||||||
|
@ -1587,7 +1456,7 @@ def global_flag_repr(self):
|
||||||
module = self.__class__.__module__.split('.')[-1]
|
module = self.__class__.__module__.split('.')[-1]
|
||||||
cls_name = self.__class__.__name__
|
cls_name = self.__class__.__name__
|
||||||
if self._name_ is None:
|
if self._name_ is None:
|
||||||
return "%s.%s(%r)" % (module, cls_name, self._value_)
|
return "%s.%s(0x%x)" % (module, cls_name, self._value_)
|
||||||
if _is_single_bit(self):
|
if _is_single_bit(self):
|
||||||
return '%s.%s' % (module, self._name_)
|
return '%s.%s' % (module, self._name_)
|
||||||
if self._boundary_ is not FlagBoundary.KEEP:
|
if self._boundary_ is not FlagBoundary.KEEP:
|
||||||
|
@ -1595,22 +1464,14 @@ def global_flag_repr(self):
|
||||||
else:
|
else:
|
||||||
name = []
|
name = []
|
||||||
for n in self._name_.split('|'):
|
for n in self._name_.split('|'):
|
||||||
if n[0].isdigit():
|
if n.startswith('0'):
|
||||||
name.append(n)
|
name.append(n)
|
||||||
else:
|
else:
|
||||||
name.append('%s.%s' % (module, n))
|
name.append('%s.%s' % (module, n))
|
||||||
return '|'.join(name)
|
return '|'.join(name)
|
||||||
|
|
||||||
def global_str(self):
|
|
||||||
"""
|
|
||||||
use enum_name instead of class.enum_name
|
|
||||||
"""
|
|
||||||
if self._name_ is None:
|
|
||||||
return "%s(%r)" % (cls_name, self._value_)
|
|
||||||
else:
|
|
||||||
return self._name_
|
|
||||||
|
|
||||||
def global_enum(cls, update_str=False):
|
def global_enum(cls):
|
||||||
"""
|
"""
|
||||||
decorator that makes the repr() of an enum member reference its module
|
decorator that makes the repr() of an enum member reference its module
|
||||||
instead of its class; also exports all members to the enum's module's
|
instead of its class; also exports all members to the enum's module's
|
||||||
|
@ -1620,8 +1481,6 @@ def global_enum(cls, update_str=False):
|
||||||
cls.__repr__ = global_flag_repr
|
cls.__repr__ = global_flag_repr
|
||||||
else:
|
else:
|
||||||
cls.__repr__ = global_enum_repr
|
cls.__repr__ = global_enum_repr
|
||||||
if not issubclass(cls, ReprEnum) or update_str:
|
|
||||||
cls.__str__ = global_str
|
|
||||||
sys.modules[cls.__module__].__dict__.update(cls.__members__)
|
sys.modules[cls.__module__].__dict__.update(cls.__members__)
|
||||||
return cls
|
return cls
|
||||||
|
|
||||||
|
@ -1663,7 +1522,6 @@ def convert_class(cls):
|
||||||
body['_value2member_map_'] = value2member_map = {}
|
body['_value2member_map_'] = value2member_map = {}
|
||||||
body['_unhashable_values_'] = []
|
body['_unhashable_values_'] = []
|
||||||
body['_member_type_'] = member_type = etype._member_type_
|
body['_member_type_'] = member_type = etype._member_type_
|
||||||
body['_value_repr_'] = etype._value_repr_
|
|
||||||
if issubclass(etype, Flag):
|
if issubclass(etype, Flag):
|
||||||
body['_boundary_'] = boundary or etype._boundary_
|
body['_boundary_'] = boundary or etype._boundary_
|
||||||
body['_flag_mask_'] = None
|
body['_flag_mask_'] = None
|
||||||
|
@ -1685,8 +1543,13 @@ def convert_class(cls):
|
||||||
# it
|
# it
|
||||||
enum_class = type(cls_name, (etype, ), body, boundary=boundary, _simple=True)
|
enum_class = type(cls_name, (etype, ), body, boundary=boundary, _simple=True)
|
||||||
for name in ('__repr__', '__str__', '__format__', '__reduce_ex__'):
|
for name in ('__repr__', '__str__', '__format__', '__reduce_ex__'):
|
||||||
if name not in body:
|
if name in body:
|
||||||
setattr(enum_class, name, getattr(etype, name))
|
continue
|
||||||
|
class_method = getattr(enum_class, name)
|
||||||
|
obj_method = getattr(member_type, name, None)
|
||||||
|
enum_method = getattr(etype, name, None)
|
||||||
|
if obj_method is not None and obj_method is class_method:
|
||||||
|
setattr(enum_class, name, enum_method)
|
||||||
gnv_last_values = []
|
gnv_last_values = []
|
||||||
if issubclass(enum_class, Flag):
|
if issubclass(enum_class, Flag):
|
||||||
# Flag / IntFlag
|
# Flag / IntFlag
|
||||||
|
@ -1897,8 +1760,8 @@ def _test_simple_enum(checked_enum, simple_enum):
|
||||||
+ list(simple_enum._member_map_.keys())
|
+ list(simple_enum._member_map_.keys())
|
||||||
)
|
)
|
||||||
for key in set(checked_keys + simple_keys):
|
for key in set(checked_keys + simple_keys):
|
||||||
if key in ('__module__', '_member_map_', '_value2member_map_', '__doc__'):
|
if key in ('__module__', '_member_map_', '_value2member_map_'):
|
||||||
# keys known to be different, or very long
|
# keys known to be different
|
||||||
continue
|
continue
|
||||||
elif key in member_names:
|
elif key in member_names:
|
||||||
# members are checked below
|
# members are checked below
|
||||||
|
@ -2019,5 +1882,3 @@ def _old_convert_(etype, name, module, filter, source=None, *, boundary=None):
|
||||||
cls.__reduce_ex__ = _reduce_ex_by_global_name
|
cls.__reduce_ex__ = _reduce_ex_by_global_name
|
||||||
cls.__repr__ = global_enum_repr
|
cls.__repr__ = global_enum_repr
|
||||||
return cls
|
return cls
|
||||||
|
|
||||||
_stdlib_enums = IntEnum, StrEnum, IntFlag
|
|
||||||
|
|
|
@ -2567,21 +2567,15 @@ class _empty:
|
||||||
|
|
||||||
|
|
||||||
class _ParameterKind(enum.IntEnum):
|
class _ParameterKind(enum.IntEnum):
|
||||||
POSITIONAL_ONLY = 'positional-only'
|
POSITIONAL_ONLY = 0
|
||||||
POSITIONAL_OR_KEYWORD = 'positional or keyword'
|
POSITIONAL_OR_KEYWORD = 1
|
||||||
VAR_POSITIONAL = 'variadic positional'
|
VAR_POSITIONAL = 2
|
||||||
KEYWORD_ONLY = 'keyword-only'
|
KEYWORD_ONLY = 3
|
||||||
VAR_KEYWORD = 'variadic keyword'
|
VAR_KEYWORD = 4
|
||||||
|
|
||||||
def __new__(cls, description):
|
@property
|
||||||
value = len(cls.__members__)
|
def description(self):
|
||||||
member = int.__new__(cls, value)
|
return _PARAM_NAME_MAPPING[self]
|
||||||
member._value_ = value
|
|
||||||
member.description = description
|
|
||||||
return member
|
|
||||||
|
|
||||||
def __str__(self):
|
|
||||||
return self.name
|
|
||||||
|
|
||||||
_POSITIONAL_ONLY = _ParameterKind.POSITIONAL_ONLY
|
_POSITIONAL_ONLY = _ParameterKind.POSITIONAL_ONLY
|
||||||
_POSITIONAL_OR_KEYWORD = _ParameterKind.POSITIONAL_OR_KEYWORD
|
_POSITIONAL_OR_KEYWORD = _ParameterKind.POSITIONAL_OR_KEYWORD
|
||||||
|
@ -2589,6 +2583,14 @@ def __str__(self):
|
||||||
_KEYWORD_ONLY = _ParameterKind.KEYWORD_ONLY
|
_KEYWORD_ONLY = _ParameterKind.KEYWORD_ONLY
|
||||||
_VAR_KEYWORD = _ParameterKind.VAR_KEYWORD
|
_VAR_KEYWORD = _ParameterKind.VAR_KEYWORD
|
||||||
|
|
||||||
|
_PARAM_NAME_MAPPING = {
|
||||||
|
_POSITIONAL_ONLY: 'positional-only',
|
||||||
|
_POSITIONAL_OR_KEYWORD: 'positional or keyword',
|
||||||
|
_VAR_POSITIONAL: 'variadic positional',
|
||||||
|
_KEYWORD_ONLY: 'keyword-only',
|
||||||
|
_VAR_KEYWORD: 'variadic keyword'
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class Parameter:
|
class Parameter:
|
||||||
"""Represents a parameter in a function signature.
|
"""Represents a parameter in a function signature.
|
||||||
|
|
|
@ -61,8 +61,7 @@
|
||||||
from xml.parsers.expat import ParserCreate
|
from xml.parsers.expat import ParserCreate
|
||||||
|
|
||||||
|
|
||||||
PlistFormat = enum.Enum('PlistFormat', 'FMT_XML FMT_BINARY', module=__name__)
|
PlistFormat = enum.global_enum(enum.Enum('PlistFormat', 'FMT_XML FMT_BINARY', module=__name__))
|
||||||
globals().update(PlistFormat.__members__)
|
|
||||||
|
|
||||||
|
|
||||||
class UID:
|
class UID:
|
||||||
|
|
|
@ -155,8 +155,6 @@ class RegexFlag:
|
||||||
# sre extensions (experimental, don't rely on these)
|
# sre extensions (experimental, don't rely on these)
|
||||||
TEMPLATE = T = sre_compile.SRE_FLAG_TEMPLATE # disable backtracking
|
TEMPLATE = T = sre_compile.SRE_FLAG_TEMPLATE # disable backtracking
|
||||||
DEBUG = sre_compile.SRE_FLAG_DEBUG # dump pattern after compilation
|
DEBUG = sre_compile.SRE_FLAG_DEBUG # dump pattern after compilation
|
||||||
__str__ = object.__str__
|
|
||||||
_numeric_repr_ = hex
|
|
||||||
|
|
||||||
# sre exception
|
# sre exception
|
||||||
error = sre_compile.error
|
error = sre_compile.error
|
||||||
|
|
|
@ -119,6 +119,7 @@
|
||||||
)
|
)
|
||||||
from _ssl import _DEFAULT_CIPHERS, _OPENSSL_API_VERSION
|
from _ssl import _DEFAULT_CIPHERS, _OPENSSL_API_VERSION
|
||||||
|
|
||||||
|
|
||||||
_IntEnum._convert_(
|
_IntEnum._convert_(
|
||||||
'_SSLMethod', __name__,
|
'_SSLMethod', __name__,
|
||||||
lambda name: name.startswith('PROTOCOL_') and name != 'PROTOCOL_SSLv23',
|
lambda name: name.startswith('PROTOCOL_') and name != 'PROTOCOL_SSLv23',
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -908,7 +908,7 @@ def handler(signum, frame):
|
||||||
|
|
||||||
%s
|
%s
|
||||||
|
|
||||||
blocked = %s
|
blocked = %r
|
||||||
signum = signal.SIGALRM
|
signum = signal.SIGALRM
|
||||||
|
|
||||||
# child: block and wait the signal
|
# child: block and wait the signal
|
||||||
|
|
|
@ -1517,11 +1517,9 @@ def testGetaddrinfo(self):
|
||||||
infos = socket.getaddrinfo(HOST, 80, socket.AF_INET, socket.SOCK_STREAM)
|
infos = socket.getaddrinfo(HOST, 80, socket.AF_INET, socket.SOCK_STREAM)
|
||||||
for family, type, _, _, _ in infos:
|
for family, type, _, _, _ in infos:
|
||||||
self.assertEqual(family, socket.AF_INET)
|
self.assertEqual(family, socket.AF_INET)
|
||||||
self.assertEqual(repr(family), '<AddressFamily.AF_INET: 2>')
|
self.assertEqual(str(family), 'AF_INET')
|
||||||
self.assertEqual(str(family), '2')
|
|
||||||
self.assertEqual(type, socket.SOCK_STREAM)
|
self.assertEqual(type, socket.SOCK_STREAM)
|
||||||
self.assertEqual(repr(type), '<SocketKind.SOCK_STREAM: 1>')
|
self.assertEqual(str(type), 'SOCK_STREAM')
|
||||||
self.assertEqual(str(type), '1')
|
|
||||||
infos = socket.getaddrinfo(HOST, None, 0, socket.SOCK_STREAM)
|
infos = socket.getaddrinfo(HOST, None, 0, socket.SOCK_STREAM)
|
||||||
for _, socktype, _, _, _ in infos:
|
for _, socktype, _, _, _ in infos:
|
||||||
self.assertEqual(socktype, socket.SOCK_STREAM)
|
self.assertEqual(socktype, socket.SOCK_STREAM)
|
||||||
|
@ -1795,10 +1793,8 @@ def test_str_for_enums(self):
|
||||||
# Make sure that the AF_* and SOCK_* constants have enum-like string
|
# Make sure that the AF_* and SOCK_* constants have enum-like string
|
||||||
# reprs.
|
# reprs.
|
||||||
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
|
||||||
self.assertEqual(repr(s.family), '<AddressFamily.AF_INET: 2>')
|
self.assertEqual(str(s.family), 'AF_INET')
|
||||||
self.assertEqual(repr(s.type), '<SocketKind.SOCK_STREAM: 1>')
|
self.assertEqual(str(s.type), 'SOCK_STREAM')
|
||||||
self.assertEqual(str(s.family), '2')
|
|
||||||
self.assertEqual(str(s.type), '1')
|
|
||||||
|
|
||||||
def test_socket_consistent_sock_type(self):
|
def test_socket_consistent_sock_type(self):
|
||||||
SOCK_NONBLOCK = getattr(socket, 'SOCK_NONBLOCK', 0)
|
SOCK_NONBLOCK = getattr(socket, 'SOCK_NONBLOCK', 0)
|
||||||
|
|
|
@ -373,8 +373,7 @@ def test_str_for_enums(self):
|
||||||
# Make sure that the PROTOCOL_* constants have enum-like string
|
# Make sure that the PROTOCOL_* constants have enum-like string
|
||||||
# reprs.
|
# reprs.
|
||||||
proto = ssl.PROTOCOL_TLS_CLIENT
|
proto = ssl.PROTOCOL_TLS_CLIENT
|
||||||
self.assertEqual(repr(proto), '<_SSLMethod.PROTOCOL_TLS_CLIENT: 16>')
|
self.assertEqual(str(proto), 'PROTOCOL_TLS_CLIENT')
|
||||||
self.assertEqual(str(proto), '16')
|
|
||||||
ctx = ssl.SSLContext(proto)
|
ctx = ssl.SSLContext(proto)
|
||||||
self.assertIs(ctx.protocol, proto)
|
self.assertIs(ctx.protocol, proto)
|
||||||
|
|
||||||
|
@ -623,7 +622,7 @@ def test_openssl111_deprecations(self):
|
||||||
with self.assertWarns(DeprecationWarning) as cm:
|
with self.assertWarns(DeprecationWarning) as cm:
|
||||||
ssl.SSLContext(protocol)
|
ssl.SSLContext(protocol)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
f'ssl.{protocol.name} is deprecated',
|
f'{protocol!r} is deprecated',
|
||||||
str(cm.warning)
|
str(cm.warning)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -632,9 +631,8 @@ def test_openssl111_deprecations(self):
|
||||||
ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
|
ctx = ssl.SSLContext(ssl.PROTOCOL_TLS_CLIENT)
|
||||||
with self.assertWarns(DeprecationWarning) as cm:
|
with self.assertWarns(DeprecationWarning) as cm:
|
||||||
ctx.minimum_version = version
|
ctx.minimum_version = version
|
||||||
version_text = '%s.%s' % (version.__class__.__name__, version.name)
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
f'ssl.{version_text} is deprecated',
|
f'ssl.{version!r} is deprecated',
|
||||||
str(cm.warning)
|
str(cm.warning)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -1490,10 +1490,8 @@ def test_formatting_with_enum(self):
|
||||||
# issue18780
|
# issue18780
|
||||||
import enum
|
import enum
|
||||||
class Float(float, enum.Enum):
|
class Float(float, enum.Enum):
|
||||||
# a mixed-in type will use the name for %s etc.
|
|
||||||
PI = 3.1415926
|
PI = 3.1415926
|
||||||
class Int(enum.IntEnum):
|
class Int(enum.IntEnum):
|
||||||
# IntEnum uses the value and not the name for %s etc.
|
|
||||||
IDES = 15
|
IDES = 15
|
||||||
class Str(enum.StrEnum):
|
class Str(enum.StrEnum):
|
||||||
# StrEnum uses the value and not the name for %s etc.
|
# StrEnum uses the value and not the name for %s etc.
|
||||||
|
@ -1510,10 +1508,8 @@ class Str(enum.StrEnum):
|
||||||
# formatting jobs delegated from the string implementation:
|
# formatting jobs delegated from the string implementation:
|
||||||
self.assertEqual('...%(foo)s...' % {'foo':Str.ABC},
|
self.assertEqual('...%(foo)s...' % {'foo':Str.ABC},
|
||||||
'...abc...')
|
'...abc...')
|
||||||
self.assertEqual('...%(foo)r...' % {'foo':Int.IDES},
|
|
||||||
'...<Int.IDES: 15>...')
|
|
||||||
self.assertEqual('...%(foo)s...' % {'foo':Int.IDES},
|
self.assertEqual('...%(foo)s...' % {'foo':Int.IDES},
|
||||||
'...15...')
|
'...IDES...')
|
||||||
self.assertEqual('...%(foo)i...' % {'foo':Int.IDES},
|
self.assertEqual('...%(foo)i...' % {'foo':Int.IDES},
|
||||||
'...15...')
|
'...15...')
|
||||||
self.assertEqual('...%(foo)d...' % {'foo':Int.IDES},
|
self.assertEqual('...%(foo)d...' % {'foo':Int.IDES},
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
``IntEnum``, ``IntFlag``, and ``StrEnum`` use the mixed-in type for their
|
|
||||||
``str()`` and ``format()`` output.
|
|
Loading…
Add table
Add a link
Reference in a new issue