What's the correct way to check if an object is a typing.Generic?

后端 未结 3 886
你的背包
你的背包 2021-01-01 09:55

I\'m trying to write code that validates type hints, and in order to do so I have to find out what kind of object the annotation is. For example, consider this snippet that\

相关标签:
3条回答
  • 2021-01-01 10:45

    There is no official way to obtain this information. The typing module is still in heavy development, and has no public API to speak of. (In fact, it will probably never have one.)

    All we can do is to look at the module's internals and find the least gross way to get the information we're after. And because the module is still being worked on, its internals will change. A lot.


    In python 3.5 and 3.6, generics had an __origin__ attribute that held a reference to the original generic base class (i.e. List[int].__origin__ would've been List), but this was changed in 3.7. Now the easiest way to find out if something is a generic is probably to check its __parameters__ and __args__ attributes.

    Here is a set of functions that can be used to detect generics:

    import typing
    
    
    if hasattr(typing, '_GenericAlias'):
        # python 3.7
        def _is_generic(cls):
            if isinstance(cls, typing._GenericAlias):
                return True
    
            if isinstance(cls, typing._SpecialForm):
                return cls not in {typing.Any}
    
            return False
    
    
        def _is_base_generic(cls):
            if isinstance(cls, typing._GenericAlias):
                if cls.__origin__ in {typing.Generic, typing._Protocol}:
                    return False
    
                if isinstance(cls, typing._VariadicGenericAlias):
                    return True
    
                return len(cls.__parameters__) > 0
    
            if isinstance(cls, typing._SpecialForm):
                return cls._name in {'ClassVar', 'Union', 'Optional'}
    
            return False
    else:
        # python <3.7
        if hasattr(typing, '_Union'):
            # python 3.6
            def _is_generic(cls):
                if isinstance(cls, (typing.GenericMeta, typing._Union, typing._Optional, typing._ClassVar)):
                    return True
    
                return False
    
    
            def _is_base_generic(cls):
                if isinstance(cls, (typing.GenericMeta, typing._Union)):
                    return cls.__args__ in {None, ()}
    
                if isinstance(cls, typing._Optional):
                    return True
    
                return False
        else:
            # python 3.5
            def _is_generic(cls):
                if isinstance(cls, (typing.GenericMeta, typing.UnionMeta, typing.OptionalMeta, typing.CallableMeta, typing.TupleMeta)):
                    return True
    
                return False
    
    
            def _is_base_generic(cls):
                if isinstance(cls, typing.GenericMeta):
                    return all(isinstance(arg, typing.TypeVar) for arg in cls.__parameters__)
    
                if isinstance(cls, typing.UnionMeta):
                    return cls.__union_params__ is None
    
                if isinstance(cls, typing.TupleMeta):
                    return cls.__tuple_params__ is None
    
                if isinstance(cls, typing.CallableMeta):
                    return cls.__args__ is None
    
                if isinstance(cls, typing.OptionalMeta):
                    return True
    
                return False
    
    
    def is_generic(cls):
        """
        Detects any kind of generic, for example `List` or `List[int]`. This includes "special" types like
        Union and Tuple - anything that's subscriptable, basically.
        """
        return _is_generic(cls)
    
    
    def is_base_generic(cls):
        """
        Detects generic base classes, for example `List` (but not `List[int]`)
        """
        return _is_base_generic(cls)
    
    
    def is_qualified_generic(cls):
        """
        Detects generics with arguments, for example `List[int]` (but not `List`)
        """
        return is_generic(cls) and not is_base_generic(cls)
    

    All of these functions should work in all python versions <= 3.7 (including anything <3.5 that uses the typing module backport).

    0 讨论(0)
  • 2021-01-01 10:47

    You may be looking for __origin__:

    # * __origin__ keeps a reference to a type that was subscripted,
    #   e.g., Union[T, int].__origin__ == Union;`
    
    import typing
    
    typ = typing.Union[int, str]
    
    if typ.__origin__ is typing.Union:
        print('value type should be one of', typ.__args__)
    elif typ.__origin__ is typing.Generic:
        print('value type should be a structure of', typ.__args__[0])
    else:
        print('value type should be', typ)
    
    >>>value type should be one of (<class 'int'>, <class 'str'>)
    

    The best I could find to advocate the use of this undocumented attribute is this reassuring quote from Guido Van Rossum (2 years ago):

    The best I can recommend is using __origin__ -- if we were to change this attribute there would still have to be some other way to access the same information, and it would be easy to grep your code for occurrences of __origin__. (I'd be less worried about changes to __origin__ than to __extra__.) You may also look at the internal functions _gorg() and _geqv() (these names will not be part of any public API, obviously, but their implementations are very simple and conceptually useful).

    This caveat in the documentation seem to indicate that nothing is set in marble yet:

    New features might be added and API may change even between minor releases if deemed necessary by the core developers.

    0 讨论(0)
  • 2021-01-01 10:51

    The most you could do, I think, is to use your typ on a variable, use typing.get_type_hints on it and extract the info you need from the returned __annotations__-like dictionary.

    PEP-484 says:

    get_type_hints(), a utility function to retrieve the type hints from a function or method. Given a function or method object, it returns a dict with the same format as __annotations__, but evaluating forward references (which are given as string literals) as expressions in the context of the original function or method definition.

    26.1.7. Classes, functions, and decorators says:

    At runtime, isinstance(x, T) will raise TypeError. In general, isinstance() and issubclass() should not be used with types.

    However, PEP-526 says in 'Non-goals':

    While the proposal is accompanied by an extension of the typing.get_type_hints standard library function for runtime retrieval of annotations, variable annotations are not designed for runtime type checking. Third party packages will have to be developed to implement such functionality.

    0 讨论(0)
提交回复
热议问题