Я пытаюсь использовать аннотации типа Python с абстрактными базовыми классами для написания некоторых интерфейсов. Есть ли способ аннотировать возможные типы *args
и **kwargs
?
Например, как я могу выразить, что разумные аргументы функции являются либо int
, либо двумя int
s? type(args)
дает Tuple
, поэтому я предполагал, что аннотировать этот тип как Union[Tuple[int, int], Tuple[int]]
, но это не работает.
from typing import Union, Tuple
def foo(*args: Union[Tuple[int, int], Tuple[int]]):
try:
i, j = args
return i + j
except ValueError:
assert len(args) == 1
i = args[0]
return i
# ok
print(foo((1,)))
print(foo((1, 2)))
# mypy doesn't like this
print(foo(1))
print(foo(1, 2))
Сообщения об ошибках от mypy:
t.py: note: In function "foo":
t.py:6: error: Unsupported operand types for + ("tuple" and "Union[Tuple[int, int], Tuple[int]]")
t.py: note: At top level:
t.py:12: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:14: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:15: error: Argument 1 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
t.py:15: error: Argument 2 to "foo" has incompatible type "int"; expected "Union[Tuple[int, int], Tuple[int]]"
Имеет смысл, что mypy не нравится это для вызова функции, потому что он ожидает, что в самом вызове будет Tuple
. Добавление после распаковки также дает ошибку ввода, которую я не понимаю.
Как аннотировать понятные типы для *args
и **kwargs
?