Opened 10 years ago
Closed 10 years ago
#22843 closed Uncategorized (wontfix)
contrib.auth.models.User not comparable
Reported by: | ukl | Owned by: | nobody |
---|---|---|---|
Component: | Uncategorized | Version: | 1.6 |
Severity: | Normal | Keywords: | |
Cc: | Triage Stage: | Unreviewed | |
Has patch: | no | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description (last modified by )
I stumbled over User objects not being comparable as expected in Python code. While I can order by the related User object in a query just fine this fails when sorting in Python.
The reason can be seen in the following manage.py shell session:
>>> from django.contrib.auth.models import User >>> u1 = User.objects.get(id=1) >>> u2 = User.objects.get(id=2) >>> u1_ = User.objects.get(id=1) >>> u1 < u2 >>> True >>> u2 < u1_ >>> True >>> u1 < u1_ >>> True >>> u1 == u1_ >>> True
I think < compares using the object address while == uses the id attribute to compare.
A more complex example that shows the failure is:
>>> sum(1 for _ in groupby(map(attrgetter('user'), MyModel.objects.order_by('user')))) 16 >>> sum(1 for _ in groupby(map(attrgetter('user'), sorted(MyModel.objects.all(), key=attrgetter('user'))))) 90
This means that in the first example there are 16 unique related Users in the MyModel objects. The second line finds considerably more Users because the sorting fails and so groupby doesn't detect duplicates.
I expected that both iterators MyModel.objects.order_by('user') and sorted(MyModel.objects.all(), key=attrgetter('user')) yield the same ordering.
Change History (1)
comment:1 by , 10 years ago
Description: | modified (diff) |
---|---|
Resolution: | → wontfix |
Status: | new → closed |
It isn't clear to me at all why this is the behaviour your would expect. No other Python object has an automatic and implied ordering on an arbitrary attribute, and there's nothing on the User model to say that "username" is the best attribute for this ordering (and I presume you meant "username", not "user"). The default User model doesn't have any predefined ordering defined. The order returned by the database is unpredictable (although, for small data sets, it will generally match the order of database insertion).
In theory, I suppose it would be possible for any object with an ordering defined to automatically gain a
__cmp__
method that implements that search order. However, I'm not sure I agree that this sort of implicit behaviour would be desirable. It would also be complicated to implement in a way that doesn't override manually defined__cmp__
methods on a model.It would also be possible to make a special case of the User object and manually define a
__cmp__
method specifically for User; however, that would also act as an encouragement for people to do sorting in Django, rather than in the database (where the operation can be optimised).So - marking this wontfix. If you want to argue for this change, please start a thread on django-developers.