-
Notifications
You must be signed in to change notification settings - Fork 242
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Generic versions of enum.Enum? #535
Comments
+1 on this. It's especially annoying that mypy requires to provide types of values explicitly in some cases, but later disregards this information and type of values becomes Any. class Problematic(enum.Enum):
a = frozenset() mypy output:
|
Changes to the typing module are best discussed at https://github.com/python/typing. @roganov Can you file a separate issue about requiring a type annotation within the body of an enum? |
@JukkaL I'm a bit confused. It looks like this is that repository? |
Although this is potentially possible, I think this adds to much burden for little benefits. For most use cases, enums are just sets of unique constants. Also, type checkers could "remember" the type of |
I don't think the generic class is needed. Enums are "frozen", you cannot add additional values after the class is constructed. They can't be subclassed either. MyPy could infer value as the union of all non-function class attributes. If a specific type annotation is desired, that could be applied to any member (likely the first). |
If type inference for enum attributes is possible, this could be an alternative solution to this problem. The main issue is that |
I think it is possible to implement this as a mypy plugin. But it is quite specialised, so it is unlikely that anyone from mypy core team will write the plugin. You of course can write the plugin yourself. |
A generic would be a cleaner solution, but this works: from enum import Enum
from typing import TYPE_CHECKING
class Foo(Enum):
value: int
a = 1
b = 2
if TYPE_CHECKING:
reveal_type(Foo.a)
reveal_type(Foo.a.value)
print(Foo.a)
print(Foo.a.value)
print(list(Foo)) |
I sort of found a solution that works for me. My problem is that I want to define some utility methods on my enum base class. However, I wasn't able to type the return value of those methods properly. First I tried this:
I tried making
Admittedly this is a hack and I was surprised it works. Maybe it shouldn't? Someone with more experience in this might be able to tell better. |
@Photonios What if |
@Photonios your use case doesn't require generic enums, as suggested above you can simply write: from enum import Enum
from typing import Type, List, Tuple, TypeVar
T = TypeVar('T')
class MyEnum(Enum):
@classmethod
def choices(cls: Type[T]) -> List[Tuple[T, str]]:
return [(item, item.name) for item in cls] # error here is a mypy bug, ignore it
class Foo(MyEnum):
A = 1
B = 2
reveal_type(Foo.choices()) # Revealed type is 'builtins.list[Tuple[test.Foo*, builtins.str]]' Also your original example type-checks because Also it looks like there is a bug in your code (uncaught because of |
Just bit by this. I was very surprised to find that |
I believe that it would make sense to make |
Ideally a type checker should be able to infer the generic type from the type of the values inside the enum. But that's not something we can do in typeshed. |
The above solution to declare "value" as having "int" type did not work for me, as my type checker was noticing that "value" never got initialized. What worked for me was declaring a "value" property and calling super.
|
(First of all, apologies if this is not the proper place for suggesting this, or if this has already been discussed, I couldn't find anything related to this so I opened an issue here.)
So I've recently come across a situation like this:
I've defined an enum
and need to use the value of an enum member later on:
Analyzing this with mypy highlights the value of
MyEnum.a.value
as Any. That makes sense, as enum member values could be of any type.However, in this case I know that all members of my enum should have int values, and no other type. In general, in most cases where I have used an enum the values have all been of a single type. I'd like to communicate this to the typing system somehow, but it doesn't seem possible with enum.Enum.
Changing the enum's type to enum.IntEnum lets mypy identify the value as int, however, using IntEnum is discouraged by the enum module documentation since it also makes enums comparable to other enums and to integers, which wouldn't actually be necessary for this use case. So as far as I understand using IntEnum wouldn't be ideal either.
Casting the .value to an int works, but it's more a workaround than a solution, and I hope this could somehow be done without casting.
To me, it seems this could be solved by introducing a generic Enum type in typing, similar to Sequence[T] and the other generics defined there.
With this, the code would look like
This could make the intent of the enum more clear, and would allow people to catch errors that would be introduced by defining an enum member with a different value type. It also would allow static checkers to infer the type of the enum member's value without having to use casts. Comparison operations would work the same way as for enum.Enum, and unlike enum.IntEnum.
If something like this could be considered for the typing module I'd be very grateful.
(Also, lastly, thanks for all the work on static typing in Python. It's helped me catch several bugs in my code so far, and I'm working on fully converting my project to make use of static typing, since it's been such a great help so far.)
The text was updated successfully, but these errors were encountered: