first commit
This commit is contained in:
132
.venv/lib/python3.12/site-packages/django/utils/inspect.py
Normal file
132
.venv/lib/python3.12/site-packages/django/utils/inspect.py
Normal file
@@ -0,0 +1,132 @@
|
||||
import functools
|
||||
import inspect
|
||||
import threading
|
||||
from contextlib import contextmanager
|
||||
|
||||
from django.utils.version import PY314
|
||||
|
||||
if PY314:
|
||||
import annotationlib
|
||||
|
||||
lock = threading.Lock()
|
||||
safe_signature_from_callable = functools.partial(
|
||||
inspect._signature_from_callable,
|
||||
annotation_format=annotationlib.Format.FORWARDREF,
|
||||
)
|
||||
|
||||
|
||||
@functools.lru_cache(maxsize=512)
|
||||
def _get_func_parameters(func, remove_first):
|
||||
# As the annotations are not used in any case, inspect the signature with
|
||||
# FORWARDREF to leave any deferred annotations unevaluated.
|
||||
if PY314:
|
||||
signature = inspect.signature(
|
||||
func, annotation_format=annotationlib.Format.FORWARDREF
|
||||
)
|
||||
else:
|
||||
signature = inspect.signature(func)
|
||||
|
||||
parameters = tuple(signature.parameters.values())
|
||||
if remove_first:
|
||||
parameters = parameters[1:]
|
||||
return parameters
|
||||
|
||||
|
||||
def _get_callable_parameters(meth_or_func):
|
||||
is_method = inspect.ismethod(meth_or_func)
|
||||
func = meth_or_func.__func__ if is_method else meth_or_func
|
||||
return _get_func_parameters(func, remove_first=is_method)
|
||||
|
||||
|
||||
ARG_KINDS = frozenset(
|
||||
{
|
||||
inspect.Parameter.POSITIONAL_ONLY,
|
||||
inspect.Parameter.KEYWORD_ONLY,
|
||||
inspect.Parameter.POSITIONAL_OR_KEYWORD,
|
||||
}
|
||||
)
|
||||
|
||||
|
||||
def get_func_args(func):
|
||||
params = _get_callable_parameters(func)
|
||||
return [param.name for param in params if param.kind in ARG_KINDS]
|
||||
|
||||
|
||||
def get_func_full_args(func):
|
||||
"""
|
||||
Return a list of (argument name, default value) tuples. If the argument
|
||||
does not have a default value, omit it in the tuple. Arguments such as
|
||||
*args and **kwargs are also included.
|
||||
"""
|
||||
params = _get_callable_parameters(func)
|
||||
args = []
|
||||
for param in params:
|
||||
name = param.name
|
||||
# Ignore 'self'
|
||||
if name == "self":
|
||||
continue
|
||||
if param.kind == inspect.Parameter.VAR_POSITIONAL:
|
||||
name = "*" + name
|
||||
elif param.kind == inspect.Parameter.VAR_KEYWORD:
|
||||
name = "**" + name
|
||||
if param.default != inspect.Parameter.empty:
|
||||
args.append((name, param.default))
|
||||
else:
|
||||
args.append((name,))
|
||||
return args
|
||||
|
||||
|
||||
def func_accepts_kwargs(func):
|
||||
"""Return True if function 'func' accepts keyword arguments **kwargs."""
|
||||
return any(p for p in _get_callable_parameters(func) if p.kind == p.VAR_KEYWORD)
|
||||
|
||||
|
||||
def func_accepts_var_args(func):
|
||||
"""
|
||||
Return True if function 'func' accepts positional arguments *args.
|
||||
"""
|
||||
return any(p for p in _get_callable_parameters(func) if p.kind == p.VAR_POSITIONAL)
|
||||
|
||||
|
||||
def method_has_no_args(meth):
|
||||
"""Return True if a method only accepts 'self'."""
|
||||
count = len([p for p in _get_callable_parameters(meth) if p.kind in ARG_KINDS])
|
||||
return count == 0 if inspect.ismethod(meth) else count == 1
|
||||
|
||||
|
||||
def func_supports_parameter(func, name):
|
||||
return any(param.name == name for param in _get_callable_parameters(func))
|
||||
|
||||
|
||||
def is_module_level_function(func):
|
||||
if not inspect.isfunction(func) or inspect.isbuiltin(func):
|
||||
return False
|
||||
|
||||
if "<locals>" in func.__qualname__:
|
||||
return False
|
||||
|
||||
return True
|
||||
|
||||
|
||||
@contextmanager
|
||||
def lazy_annotations():
|
||||
"""
|
||||
inspect.getfullargspec eagerly evaluates type annotations. To add
|
||||
compatibility with Python 3.14+ deferred evaluation, patch the module-level
|
||||
helper to provide the annotation_format that we are using elsewhere.
|
||||
|
||||
This private helper could be removed when there is an upstream solution for
|
||||
https://github.com/python/cpython/issues/141560.
|
||||
|
||||
This context manager is not reentrant.
|
||||
"""
|
||||
if not PY314:
|
||||
yield
|
||||
return
|
||||
with lock:
|
||||
original_helper = inspect._signature_from_callable
|
||||
inspect._signature_from_callable = safe_signature_from_callable
|
||||
try:
|
||||
yield
|
||||
finally:
|
||||
inspect._signature_from_callable = original_helper
|
||||
Reference in New Issue
Block a user