|
| 1 | +import inspect |
| 2 | +import sys |
| 3 | + |
| 4 | +if sys.version_info >= (3, 10): |
| 5 | + from inspect import signature |
| 6 | +elif sys.version_info >= (3, 8): |
| 7 | + from typing import Any, Callable |
| 8 | + |
| 9 | + from typing_extensions import get_annotations |
| 10 | + |
| 11 | + def signature( |
| 12 | + func: Callable[..., Any], eval_str: bool = False, **kwargs: Any |
| 13 | + ) -> inspect.Signature: |
| 14 | + sig = inspect.signature(func, **kwargs) |
| 15 | + ann = get_annotations( |
| 16 | + func, |
| 17 | + globals=kwargs.get("globals"), |
| 18 | + locals=kwargs.get("locals"), |
| 19 | + eval_str=eval_str, |
| 20 | + ) |
| 21 | + return sig.replace( |
| 22 | + parameters=[ |
| 23 | + param.replace(annotation=ann.get(name, param.annotation)) |
| 24 | + for name, param in sig.parameters.items() |
| 25 | + ], |
| 26 | + return_annotation=ann.get("return", sig.return_annotation), |
| 27 | + ) |
| 28 | +else: |
| 29 | + # Fallback for Python <3.8 to make `inspect.signature` accept the `eval_str` |
| 30 | + # keyword argument as a no-op. We can't backport support for evaluating |
| 31 | + # string annotations because only typing-extensions v4.13.0+ provides a |
| 32 | + # backport of `typing.get_annotations`, which requires Python 3.8+. |
| 33 | + |
| 34 | + from typing import Any, Callable |
| 35 | + |
| 36 | + def signature( |
| 37 | + func: Callable[..., Any], eval_str: bool = False, **kwargs: Any |
| 38 | + ) -> inspect.Signature: |
| 39 | + return inspect.signature(func, **kwargs) |
| 40 | + |
| 41 | + |
| 42 | +__all__ = ["signature"] |
0 commit comments