-
-
Notifications
You must be signed in to change notification settings - Fork 12k
Description
There is a complex set of questions around how to handle method resolution in the presence of __numpy_ufunc__. Currently in master is an extremely complicated set of rules that isn't documented and that I don't actually understand (see #5748 for the latest set of changes to this), so it's kinda hard to know whether they are correct, but I suspect not. And this is a blocker for 1.10, b/c whatever we release in 1.10 will be set in stone forever.
I strongly feel that we cannot include __numpy_ufunc__ in a release without at least having a document somewhere describing what the actual dispatch rules are. I hope that doesn't mean we have to defer __numpy_ufunc__ for another release, but if it does then it does.
AFAICT this is how a op b dispatch works for ndarrays, BEFORE __numpy_ufunc__ (i.e., this is how 1.9 works):
- First Python uses the subclass rule to decide whether to invoke
a.__op__(b)orb.__rop__(a). So in the case where one of these objects is a proper subclass of the other, that object always gets to do absolutely anything, so that's fine. The interesting cases are the ones where neither is a proper subclass of the other (either because it's like,matrix + masked array, or because it's likendarray + scipy.sparse). So without loss of generality, let's focus on the case where Python callsa.__op__(b), andais either an instance ofndarrayor else an instance of a subclass ofndarraywhich has not overridden__op__, i.e. we're gettingndarray.__op__(a, b). ndarray.__op__has the following logic (seePyArray_GenericBinaryFunctioninnumber.c):- If
bis not anndarrayat all (even a subclass), andbhas a higher__array_priority__thana, then we returnNotImplementedand let control pass tob.__rop__(a). - Otherwise, we call
np.op(a, b)and let the ufunc machinery take over.
- If
np.op(a, b)does the following (seePyUFunc_GenericFunction,PyUFunc_GeneralizedFunction, inufunc_object.c, and alsoufunc_generic_callwhich converts-2return values from the previous intoNotImplementedso you have to audit their whole call stack):- If
bis not anndarray, and callingnp.array(b)returns an object array (presumably because coercion failed... though I guess this could also be hit ifb.__array__()return an object array or something), ANDbhas a higher__array_priority__thana, andbhas an__rop__method, then returnNotImplemented. - If any of our arrays contain structured dtypes or strings, and there are no special struct ufunc loops registered, but not if any of our arrays contain objects, then return
NotImplemented. (This is buried inget_ufunc_arguments, search forreturn -2.) - Otherwise we return the actual ufunc result.
- If
Now, my suggestion is that the way we would EVENTUALLY like this to look is:
- First, Python uses the subclass rule to decide whether to invoke
a.__op__(b)orb.__rop__(a). As above, let's assume that it invokesndarray.__op__(a, b). ndarray.__op__(a, b)callsnp.op(a, b)(which in turn invokes all the standard ufunc stuff, including__numpy_ufunc__resolution).- There is no step 3.
I submit that it is obvious that IF we can make this work, then it is obviously the ideal outcome, because it is the simplest possible solution. But is it too simple? To determine this we have to answer two questions: (1) Will it adequately address all the relevant use cases? (2) Can we get there from here?
So let's compare the current rules to my dream rules.
First, we observe that everything that currently happens inside the ufunc machinery looks like it's totally wrong. The first check can only be triggered if b is a non-ndarray that has a higher __array_priority__ (among other things), but if we look above, we see that those conditions are sufficient to trigger the check in ndarray.__op__, so checking again at the ufunc level is redundant at best. And the second check is just incoherent nonsense AFAICT. The only reason to return NotImplemented is b/c you want to pass control to another __(r)op__ method, and there's no reason arrays containing structured dtypes in particular should somehow magically have different __(r)op__ methods available than other arrays. So we can just get rid of all the ufunc stuff immediately, great.
That leaves the __array_priority__ stuff. We have two problems here: we can't just drop this immediately b/c of backcompat issues, and we need to have some way to continue to support all the use cases that this currently supports. The first problem is just a matter of having a deprecation period. For the second, observe that a class which defines a __numpy_ufunc__ method gets complete control over what any ufunc call does, so it has almost as much power as a class that currently sets __array_priority__. The only additional power that __array_priority__ currently gives you is that it lets you distinguish between e.g. a call to ndarray.__add(a, b) versus a call to np.add(a, b). So the only code that really loses out from my proposed change is code which wants a + b and add(a, b) to do different things.
AFAIK in the entire history of numpy there is only one situation where this power has been used on purpose: the definition of matrix classes where a * b is matmul, but np.multiply(a, b) is elmul. And we've all agreed that such classes should be deprecated and eventually phased out (cite: PEP 465).
So, I conclude that EVENTUALLY my dream rules should work great. The only problem is that we need some temporary compromises to get us from here to there. Therefore, I propose we use the following dispatch rules in numpy 1.10, with the goal of moving to my "dream rules" in some future version:
- First, Python uses the subclass rule to decide whether to invoke
a.__op__(b)orb.__rop__(a). As above, let's assume that it invokesndarray.__op__(a, b). ndarray.__op__(a, b)does the following:- If
bdoes not define__numpy_ufunc__and is not anndarrayat all (even a subclass), andbhas a higher__array_priority__thana, then we issue a deprecation warning and returnNotImplementedand let control pass tob.__rop__(a). (bolded parts are changes compared to the current behaviour) - If
__op__is__mul__andb->tp_class->tp_name.startswith("scipy.sparse."), then returnNotImplemented. (This rule is necessary in addition to the above, becausescipy.sparsehas already made a release containing__numpy_ufunc__methods, so the exception above doesn't apply.) - Otherwise, we call
np.op(a, b)and let the ufunc machinery take over.
- If
I believe that this is adequate to covers all practical use cases for the current dispatch machinery, and gives us a clean path to better dispatch machinery in the future.
The main alternative proposal is Pauli's, which involves a very complicated check (I won't try to summarize here, see this comment and following code). The goal of that approach is to continue supporting classes where a + b and add(a, b) do different things. I don't think that keeping substantial additional complexity around indefinitely is worth it in order to support functionality that no-one has ever found a use for except in one very specific case (overriding __mul__), and where we generally agree that that one specific case should be phased out as possible.
I would very much appreciate feedback from scipy.sparse and astropy in particular on whether the above covers all their concerns.