Coverage for /var/srv/projects/api.amasfac.comuna18.com/tmp/venv/lib/python3.9/site-packages/numpy/core/overrides.py: 74%
56 statements
« prev ^ index » next coverage.py v6.4.4, created at 2023-07-17 14:22 -0600
« prev ^ index » next coverage.py v6.4.4, created at 2023-07-17 14:22 -0600
1"""Implementation of __array_function__ overrides from NEP-18."""
2import collections
3import functools
4import os
6from numpy.core._multiarray_umath import (
7 add_docstring, implement_array_function, _get_implementing_args)
8from numpy.compat._inspect import getargspec
11ARRAY_FUNCTION_ENABLED = bool(
12 int(os.environ.get('NUMPY_EXPERIMENTAL_ARRAY_FUNCTION', 1)))
14array_function_like_doc = (
15 """like : array_like, optional
16 Reference object to allow the creation of arrays which are not
17 NumPy arrays. If an array-like passed in as ``like`` supports
18 the ``__array_function__`` protocol, the result will be defined
19 by it. In this case, it ensures the creation of an array object
20 compatible with that passed in via this argument."""
21)
23def set_array_function_like_doc(public_api):
24 if public_api.__doc__ is not None: 24 ↛ 29line 24 didn't jump to line 29, because the condition on line 24 was never false
25 public_api.__doc__ = public_api.__doc__.replace(
26 "${ARRAY_FUNCTION_LIKE}",
27 array_function_like_doc,
28 )
29 return public_api
32add_docstring(
33 implement_array_function,
34 """
35 Implement a function with checks for __array_function__ overrides.
37 All arguments are required, and can only be passed by position.
39 Parameters
40 ----------
41 implementation : function
42 Function that implements the operation on NumPy array without
43 overrides when called like ``implementation(*args, **kwargs)``.
44 public_api : function
45 Function exposed by NumPy's public API originally called like
46 ``public_api(*args, **kwargs)`` on which arguments are now being
47 checked.
48 relevant_args : iterable
49 Iterable of arguments to check for __array_function__ methods.
50 args : tuple
51 Arbitrary positional arguments originally passed into ``public_api``.
52 kwargs : dict
53 Arbitrary keyword arguments originally passed into ``public_api``.
55 Returns
56 -------
57 Result from calling ``implementation()`` or an ``__array_function__``
58 method, as appropriate.
60 Raises
61 ------
62 TypeError : if no implementation is found.
63 """)
66# exposed for testing purposes; used internally by implement_array_function
67add_docstring(
68 _get_implementing_args,
69 """
70 Collect arguments on which to call __array_function__.
72 Parameters
73 ----------
74 relevant_args : iterable of array-like
75 Iterable of possibly array-like arguments to check for
76 __array_function__ methods.
78 Returns
79 -------
80 Sequence of arguments with __array_function__ methods, in the order in
81 which they should be called.
82 """)
85ArgSpec = collections.namedtuple('ArgSpec', 'args varargs keywords defaults')
88def verify_matching_signatures(implementation, dispatcher):
89 """Verify that a dispatcher function has the right signature."""
90 implementation_spec = ArgSpec(*getargspec(implementation))
91 dispatcher_spec = ArgSpec(*getargspec(dispatcher))
93 if (implementation_spec.args != dispatcher_spec.args or 93 ↛ 101line 93 didn't jump to line 101, because the condition on line 93 was never true
94 implementation_spec.varargs != dispatcher_spec.varargs or
95 implementation_spec.keywords != dispatcher_spec.keywords or
96 (bool(implementation_spec.defaults) !=
97 bool(dispatcher_spec.defaults)) or
98 (implementation_spec.defaults is not None and
99 len(implementation_spec.defaults) !=
100 len(dispatcher_spec.defaults))):
101 raise RuntimeError('implementation and dispatcher for %s have '
102 'different function signatures' % implementation)
104 if implementation_spec.defaults is not None:
105 if dispatcher_spec.defaults != (None,) * len(dispatcher_spec.defaults): 105 ↛ 106line 105 didn't jump to line 106, because the condition on line 105 was never true
106 raise RuntimeError('dispatcher functions can only use None for '
107 'default argument values')
110def set_module(module):
111 """Decorator for overriding __module__ on a function or class.
113 Example usage::
115 @set_module('numpy')
116 def example():
117 pass
119 assert example.__module__ == 'numpy'
120 """
121 def decorator(func):
122 if module is not None: 122 ↛ 124line 122 didn't jump to line 124, because the condition on line 122 was never false
123 func.__module__ = module
124 return func
125 return decorator
128def array_function_dispatch(dispatcher, module=None, verify=True,
129 docs_from_dispatcher=False):
130 """Decorator for adding dispatch with the __array_function__ protocol.
132 See NEP-18 for example usage.
134 Parameters
135 ----------
136 dispatcher : callable
137 Function that when called like ``dispatcher(*args, **kwargs)`` with
138 arguments from the NumPy function call returns an iterable of
139 array-like arguments to check for ``__array_function__``.
140 module : str, optional
141 __module__ attribute to set on new function, e.g., ``module='numpy'``.
142 By default, module is copied from the decorated function.
143 verify : bool, optional
144 If True, verify the that the signature of the dispatcher and decorated
145 function signatures match exactly: all required and optional arguments
146 should appear in order with the same names, but the default values for
147 all optional arguments should be ``None``. Only disable verification
148 if the dispatcher's signature needs to deviate for some particular
149 reason, e.g., because the function has a signature like
150 ``func(*args, **kwargs)``.
151 docs_from_dispatcher : bool, optional
152 If True, copy docs from the dispatcher function onto the dispatched
153 function, rather than from the implementation. This is useful for
154 functions defined in C, which otherwise don't have docstrings.
156 Returns
157 -------
158 Function suitable for decorating the implementation of a NumPy function.
159 """
161 if not ARRAY_FUNCTION_ENABLED: 161 ↛ 162line 161 didn't jump to line 162, because the condition on line 161 was never true
162 def decorator(implementation):
163 if docs_from_dispatcher:
164 add_docstring(implementation, dispatcher.__doc__)
165 if module is not None:
166 implementation.__module__ = module
167 return implementation
168 return decorator
170 def decorator(implementation):
171 if verify:
172 verify_matching_signatures(implementation, dispatcher)
174 if docs_from_dispatcher:
175 add_docstring(implementation, dispatcher.__doc__)
177 @functools.wraps(implementation)
178 def public_api(*args, **kwargs):
179 relevant_args = dispatcher(*args, **kwargs)
180 return implement_array_function(
181 implementation, public_api, relevant_args, args, kwargs)
183 public_api.__code__ = public_api.__code__.replace(
184 co_name=implementation.__name__,
185 co_filename='<__array_function__ internals>')
186 if module is not None:
187 public_api.__module__ = module
189 public_api._implementation = implementation
191 return public_api
193 return decorator
196def array_function_from_dispatcher(
197 implementation, module=None, verify=True, docs_from_dispatcher=True):
198 """Like array_function_dispatcher, but with function arguments flipped."""
200 def decorator(dispatcher):
201 return array_function_dispatch(
202 dispatcher, module, verify=verify,
203 docs_from_dispatcher=docs_from_dispatcher)(implementation)
204 return decorator