129

Using the Python Enum class, is there a way to test if an Enum contains a specific int value without using try/catch?

With the following class:

from enum import Enum

class Fruit(Enum):
    Apple = 4
    Orange = 5
    Pear = 6

How can I test for the value 6 (returning true), or the value 7 (returning false)?

Ivan
  • 5,493
  • 2
  • 28
  • 45
Nathan Kovner
  • 2,133
  • 3
  • 16
  • 13
  • Using `try/except` in Python is absolutely okay (although I admit that I had a hard time to get it). So, what is the reason for you not to use it? – mikuszefski Oct 12 '21 at 13:23

14 Answers14

176

test for values

variant 1

note that an Enum has a member called _value2member_map_ (which is undocumented and may be changed/removed in future python versions):

print(Fruit._value2member_map_)
# {4: <Fruit.Apple: 4>, 5: <Fruit.Orange: 5>, 6: <Fruit.Pear: 6>}

you can test if a value is in your Enum against this map:

5 in Fruit._value2member_map_  # True
7 in Fruit._value2member_map_  # False

variant 2

if you do not want to rely on this feature this is an alternative:

values = [item.value for item in Fruit]  # [4, 5, 6]

or (probably better): use a set; the in operator will be more efficient:

values = set(item.value for item in Fruit)  # {4, 5, 6}

then test with

5 in values  # True
7 in values  # False

add has_value to your class

you could then add this as a method to your class:

class Fruit(Enum):
    Apple = 4
    Orange = 5
    Pear = 6

    @classmethod
    def has_value(cls, value):
        return value in cls._value2member_map_ 

print(Fruit.has_value(5))  # True
print(Fruit.has_value(7))  # False

test for keys

if you want to test for the names (and not the values) i would use _member_names_:

'Apple' in Fruit._member_names_  # True
'Mango' in Fruit._member_names_  # False
hiro protagonist
  • 40,708
  • 13
  • 78
  • 98
  • 9
    I personally prefer the second solution. Thank you. – Aliakbar Abbasi Sep 03 '17 at 05:05
  • 3
    Either advice is not particularly memory- or time-efficient, although also not very heavy in these terms. In my case, I am working with the `HTTPStatus` enum from the standard library which as of now contains 57 entries. I would never process a whole list of values from an enumeration, and caching them is something I'd like to avoid. – Acsor Sep 18 '17 at 16:51
  • 2
    @none if you are worried about speed you could create a `set` for the lookup. that would use some additional memory but not a significant amount... – hiro protagonist Sep 18 '17 at 18:05
  • Why the parentheses around the return value? – Konstantin Jan 02 '18 at 11:16
  • Had to use `return any(value == item.value[0] for item in cls)` with Python3 did the Enum structure change? – lony Dec 20 '18 at 15:00
  • Beware: using non-Enums in containment checks will raise TypeError in Python 3.8 – Pedro Lourenço Jul 01 '20 at 11:44
29

There is a way to have all the enums be able to check if an item is present:

import enum 

class MyEnumMeta(enum.EnumMeta): 
    def __contains__(cls, item): 
        return item in [v.value for v in cls.__members__.values()] 

class MyEnum(enum.Enum, metaclass=MyEnumMeta): 
   FOO = "foo" 
   BAR = "bar"

Now you can do an easy check:

>>> "foo" in MyEnum
True

It can even be made simpler if all the enum's values will always be the same type -- for example strings:

import enum 
 
class MyEnumMeta(enum.EnumMeta):  
    def __contains__(cls, item): 
        return item in cls.__members__.values()

class MyEnum(str, enum.Enum, metaclass=MyEnumMeta): 
    FOO = "foo" 
    BAR = "bar"

Edit: Yet another version, technically the most correct one:

import enum 

class MyEnumMeta(enum.EnumMeta): 
    def __contains__(cls, item): 
        try:
            cls(item)
        except ValueError:
            return False
        else:
            return True

class MyEnum(enum.Enum, metaclass=MyEnumMeta): 
   FOO = "foo" 
   BAR = "bar"
Berislav Lopac
  • 15,459
  • 6
  • 67
  • 77
25

You could use Enum.__members__ - an ordered dictionary mapping names to members:

In [12]: 'Apple' in Fruit.__members__
Out[12]: True

In [13]: 'Grape' in Fruit.__members__
Out[13]: False
Reda Maachi
  • 807
  • 6
  • 15
  • 8
    The question is asking to test for an int value, not a string. – Nathan Kovner Apr 26 '17 at 12:55
  • 8
    Is there no member in Enum that contains the values? Isn't that strange? – Konstantin Jan 02 '18 at 11:14
  • 4
    This goes in the opposite direction from what the question asked. – user2357112 Oct 01 '18 at 00:59
  • **This doesn't work for values!** (it's for "keys") answerer probably had `Apple = "Apple"`, so he had the same letters & lettercase used for both key and value ... if you'd have `APPLE = "Apple"` then `.__members__` **will result in `{'APPLE': , 'GRAPE': }`** – jave.web May 13 '21 at 10:17
9

Building on what Reda Maachi started:

6 in Fruit.__members__.values() 

returns True

7 in Fruit.__members__.values()  

returns False

manu3d
  • 901
  • 1
  • 8
  • 22
  • 3
    This looked like a very clean way. Unfortunately, when I tested this (at least in Py3.7) it doesn't work like that. Fruit.__members__.values() evaluates to odict_values([, , ]), and both the tests above for 6 and 7 return False for me. But this returns true: ```>>> Fruit.Orange in Fruit.__members__.values() True``` – nrshapiro Oct 06 '19 at 17:40
  • 6
    I'd like to point out that this works as written for IntEnum only. – Yandros Dec 27 '19 at 01:11
8

If the enum has many members, this approach can be faster because it doesn't make a new list and stops walking the enum when the given value is found:

any(x.value == 5 for x in Fruit)  # True
any(x.value == 7 for x in Fruit)  # False
Finesse
  • 8,378
  • 7
  • 57
  • 72
7

I just convert an IntEnum to a list and test it normally:

from enum import IntEnum
class Foo(IntEnum):
    ONE = 1
    TWO = 2
    THREE = 3

print(1 in list(Foo))
True
print(4 in list(Foo))
False
Ray Salemi
  • 4,017
  • 2
  • 24
  • 50
4

Don't.

If you are using Enum, you can test for enum with

     if isinstance(key, Fruit):

But otherwise, try.. is the pythonic way to test for enum. Indeed, for any break in the duck-typing paradigm.

The correct, and pythonic, way of testing for an int in an IntEnum is to give it a go and to catch a ValueError if there's a failure.

Many of the solutions proposed above are actively deprecated and will be disallowed by 3.8 ( "DeprecationWarning: using non-Enums in containment checks will raise TypeError in Python 3.8" )

If you are really disinterested in keeping your code modern, then you can just use

if key in Fruit:
Konchog
  • 1,690
  • 16
  • 21
  • 2
    This solution is actually wrong. Since OP asked how to check if a value was in an enum, such as 6, or 7. This will always return false, since none of the members of the enum are actually equal to 6 or 7. Try it out. – Joshua Ryan Jan 24 '20 at 21:42
  • 1
    But this does do what most people of googling for, how to check if a value is an enum, upvoted! Thank you! – run_the_race May 30 '22 at 17:57
3

Just check whether it's in Enum. _value2member_map_

In[15]: Fruit._value2member_map_
Out[15]: {4: <Fruit.Apple: 4>, 5: <Fruit.Orange: 5>, 6: <Fruit.Pear: 6>}

In[16]: 6 in Fruit._value2member_map_
Out[16]: True

In[17]: 7 in Fruit._value2member_map_
Out[17]: False
JianWei
  • 111
  • 2
  • 6
3

An EAFP version of the answer:

try: 
    Fruit(val)
    return True
except ValueError:
    return False
ComeOnGetMe
  • 909
  • 7
  • 7
3

You could use __members__ special attribute to iterate over members:

from enum import Enum

class Fruit(Enum):
    Apple = 4
    Orange = 5
    Pear = 6

    @staticmethod
    def has_value(item):
        return item in [v.value for v in Fruit.__members__.values()]
Ignacio Vergara Kausel
  • 4,786
  • 3
  • 34
  • 40
Alex Jolig
  • 12,826
  • 19
  • 125
  • 158
3

There's another one liner solution nobody has mentioned yet:

is_value_in_fruit = any(f.value == value_to_check for f in Fruit)

Also, if you use IntEnum instead of Enum, (class Fruit(IntEnum)) you can just do this

is_value_in_fruit = any(f == value_to_check for f in Fruit) 
Joshua Ryan
  • 588
  • 4
  • 12
2

how about this?

from enum import Enum

class Fruit(Enum):
    Apple = 4
    Orange = 5
    Pear = 6

has_apples = 4 in [n.value for n in Fruit]

This would let you also do:

has_apples = "Apple" in [n.name for n in Fruit]
Jubair
  • 2,701
  • 3
  • 28
  • 27
1

IntEnum + __members__

You could use IntEnum and __members__ to achieve required behaviour:

from enum import IntEnum

class Fruit(IntEnum):
    Apple = 4
    Orange = 5
    Pear = 6

>>> 6 in Fruit.__members__.values()
True
>>> 7 in Fruit.__members__.values()
False

Enum + list comprehension + .value

If you must/want stick to Enum, you can do:

>>> 6 in [f.value for f in Fruit]
True
>>> 7 in [f.value for f in Fruit]
False

EAPF + ValueError

Or you can use easier to ask for forgiveness than permission method:

try:
    Fruit(x)
except ValueError:
    return False
else:
    return True
maciek
  • 2,921
  • 2
  • 25
  • 30
0
class MyEnumMixin:

    raw_values = None  # for IDE autocomplete

    def __new__(cls, value):
        if 'raw_values' not in cls.__dict__:
            cls.raw_values = set()
        cls.raw_values.add(value)
        if cls.__bases__[0] is MyEnumMixin:
            member = object().__new__(cls)
        else:
            member = super().__new__(cls, value)
        member._value_ = value
        return member



class MyEnum(MyEnumMixin, Enum):
    FOO = 1
    BAR = 'bar'

print(1 in MyEnum.raw_values)
print('bar' in MyEnum.raw_values)



class MyStrEnumMixin(MyEnumMixin, str):
    pass

class MyStrEnum(MyStrEnumMixin, Enum):
    FOO = 'foo'
    BAR = 'bar'

print('foo' in MyStrEnum.raw_values)
print('bar' in MyStrEnum.raw_values)

Val K
  • 332
  • 1
  • 9