Sugerencia de tipo Python 3 para decorador

Considere el siguiente código:

from typing import Callable, Any

TFunc = Callable[..., Any]

def get_authenticated_user(): return "John"

def require_auth() -> Callable[TFunc, TFunc]:
    def decorator(func: TFunc) -> TFunc:
        def wrapper(*args, **kwargs) -> Any:
            user = get_authenticated_user()
            if user is None:
                raise Exception("Don't!")
            return func(*args, **kwargs)
        return wrapper
    return decorator

@require_auth()
def foo(a: int) -> bool:
    return bool(a % 2)

foo(2)      # Type check OK
foo("no!")  # Type check failing as intended

Este fragmento de código funciona según lo previsto. Ahora imagine que quiero extender esto, y en lugar de solo ejecutarfunc(*args, **kwargs) Quiero inyectar el nombre de usuario en los argumentos. Por lo tanto, modifico la firma de la función.

from typing import Callable, Any

TFunc = Callable[..., Any]

def get_authenticated_user(): return "John"

def inject_user() -> Callable[TFunc, TFunc]:
    def decorator(func: TFunc) -> TFunc:
        def wrapper(*args, **kwargs) -> Any:
            user = get_authenticated_user()
            if user is None:
                raise Exception("Don't!")
            return func(*args, user, **kwargs)  # <- call signature modified

        return wrapper

    return decorator


@inject_user()
def foo(a: int, username: str) -> bool:
    print(username)
    return bool(a % 2)


foo(2)      # Type check OK
foo("no!")  # Type check OK <---- UNEXPECTED

No puedo encontrar una forma correcta de escribir esto. Sé que en este ejemplo, la función decorada y la función devuelta deberían tener técnicamente la misma firma (pero incluso eso no se detecta).

Respuestas a la pregunta(1)

Su respuesta a la pregunta