Callable with variadic fixed arguments that doesn't mean *args
Suppose I have a type of function called CommandHandler, that is a function that receives a command as its first parameter but then could have other parameters. Examples
class Command:
pass
class RegisterUser(Command):
pass
class DeleteUser(Command):
pass
def register_user(command: RegisterUser, db : Db) -> None:
...
def delete_user(command: DeleteUser, dispatch: EventDispatcher, cache: Cache) -> None:
...
mappings : dict[Command, CommandHandler] = {
RegisterUser: register_user,
DeleteUser: delete_user
}
What would be the definition of CommandHandler?
Ideally it would be:
C= TypeVar("C", bound=Command)
CommandHandler = Callable[[C, ...], None]
But this syntax is not possible.
I've found in the documentation that Concatenate could receive ... at the end,
Concatenate is currently only valid when used as the first argument to a Callable. The last parameter to Concatenate must be a ParamSpec or ellipsis (...).
so it could be:
C= TypeVar("C", bound=Command)
CommandHandler = Callable[Concatenate[C, ...], None]
But this returns an error from Pylance:

One could say Callback Protocol could solve this like:
C= TypeVar("C", bound=Command)
class CommandHandler(Protocol):
def __call__(self, __command: C, *__args: Any) -> None: ...
But it only receives functions that has an implicit variadic argument, it doesn't work with variadic forms of the function.
So far the only way I could solve this is through the Union of many Callables, but it doesn't look like the best solution.
At the end CommandHandler can only be defined as:
C= TypeVar("C", bound=Command)
CommandHandler = Union[
Callable[[C, Any], None],
Callable[[C, Any, Any], None],
Callable[[C, Any, Any, Any], None],
Callable[[C, Any, Any, Any, Any], None],
Callable[[C, Any, Any, Any, Any, Any], None],
Callable[[C, Any, Any, Any, Any, Any, Any], None],
Callable[[C, Any, Any, Any, Any, Any, Any, Any], None],
Callable[
[C, Any, Any, Any, Any, Any, Any, Any, Any], None
],
]
Related: https://github.com/python/cpython/issues/88954 https://stackoverflow.com/questions/57658879/python-type-hint-for-callable-with-variable-number-of-str-same-type-arguments
TypeVarTuple can be used to solve this. What you want is,
Ts = TypeVarTuple("Ts")
C = TypeVar("C", bound=Command)
Callable[[C, *Ts], None]
Minor note, most of the time if you accept a callable with return None you probably don't care about return type and should let it be object (Callable[[C, *Ts], object])
@DrecDroid, PEP 612 appears to indicate that the last argument to Concatenate must be a ParamSpec, not an ellipsis. The section titled valid use locations indicates that the last argument for Concatenate must be a "parameter_specification_variable". This explains why pyright (the type checker upon which pylance is built) considers an ellipsis an error here.
I see that the latest Python 3.11 documentation for Concatenate indicates that an ellipsis is allowed here as well. Either I'm misinterpreting the PEP, this aspect of Concatenate was modified after the PEP was written, or the 3.11 documentation is incorrect. I'm not sure which.
@hmc-cs-mdrissi Thanks, good to know, although it appears to be a feature for 3.11.
Unpack operator in subscript requires Python 3.11 or newer Pylance
@erictraut Yes I was confused for that also.
The Concatenate[P, ...] is an unresolved question in this issue.
The 3.11 typevartuple is not much of a restriction as you can use typing_extensions even on 3.7 and use typevartuple. You'll just need to use Unpack[Ts] instead of *Ts.