Fixed #146 -- Changed order_by and ordering parameters to be less verbose. The old syntax is still supported but will not be supported by first release.

git-svn-id: http://code.djangoproject.com/svn/django/trunk@292 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
Adrian Holovaty 2005-07-22 13:02:27 +00:00
parent 05bdb8d64b
commit 8b3c05afa5
7 changed files with 93 additions and 56 deletions

View File

@ -42,7 +42,7 @@ class Comment(meta.Model):
'RATINGS_OPTIONAL': 'ra', 'RATINGS_OPTIONAL': 'ra',
'IS_PUBLIC': 'ip', 'IS_PUBLIC': 'ip',
} }
ordering = (('submit_date', 'DESC'),) ordering = ('-submit_date',)
admin = meta.Admin( admin = meta.Admin(
fields = ( fields = (
(None, {'fields': ('content_type_id', 'object_id', 'site_id')}), (None, {'fields': ('content_type_id', 'object_id', 'site_id')}),
@ -170,7 +170,7 @@ class FreeComment(meta.Model):
meta.BooleanField('approved', 'approved by staff'), meta.BooleanField('approved', 'approved by staff'),
meta.ForeignKey(core.Site), meta.ForeignKey(core.Site),
) )
ordering = (('submit_date', 'DESC'),) ordering = ('-submit_date',)
admin = meta.Admin( admin = meta.Admin(
fields = ( fields = (
(None, {'fields': ('content_type_id', 'object_id', 'site_id')}), (None, {'fields': ('content_type_id', 'object_id', 'site_id')}),

View File

@ -148,7 +148,7 @@ class CommentListNode(template.Node):
'content_type__python_module_name__exact': self.module, 'content_type__python_module_name__exact': self.module,
'site_id__exact': SITE_ID, 'site_id__exact': SITE_ID,
'select_related': True, 'select_related': True,
'order_by': (('submit_date', self.ordering),), 'order_by': (self.ordering + 'submit_date',),
} }
if not self.free and COMMENTS_BANNED_USERS_GROUP: if not self.free and COMMENTS_BANNED_USERS_GROUP:
kwargs['select'] = {'is_hidden': 'user_id IN (SELECT user_id FROM auth_users_groups WHERE group_id = %s)' % COMMENTS_BANNED_USERS_GROUP} kwargs['select'] = {'is_hidden': 'user_id IN (SELECT user_id FROM auth_users_groups WHERE group_id = %s)' % COMMENTS_BANNED_USERS_GROUP}
@ -170,16 +170,16 @@ class CommentListNode(template.Node):
class DoCommentForm: class DoCommentForm:
""" """
Displays a comment form for the given params. Displays a comment form for the given params.
Syntax:: Syntax::
{% comment_form for [pkg].[py_module_name] [context_var_containing_obj_id] with [list of options] %} {% comment_form for [pkg].[py_module_name] [context_var_containing_obj_id] with [list of options] %}
Example usage:: Example usage::
{% comment_form for lcom.eventtimes event.id with is_public yes photos_optional thumbs,200,400 ratings_optional scale:1-5|first_option|second_option %} {% comment_form for lcom.eventtimes event.id with is_public yes photos_optional thumbs,200,400 ratings_optional scale:1-5|first_option|second_option %}
``[context_var_containing_obj_id]`` can be a hard-coded integer or a variable containing the ID. ``[context_var_containing_obj_id]`` can be a hard-coded integer or a variable containing the ID.
""" """
def __init__(self, free, tag_name): def __init__(self, free, tag_name):
@ -246,18 +246,18 @@ class DoCommentCount:
""" """
Gets comment count for the given params and populates the template context Gets comment count for the given params and populates the template context
with a variable containing that value, whose name is defined by the 'as' with a variable containing that value, whose name is defined by the 'as'
clause. clause.
Syntax:: Syntax::
{% get_comment_count for [pkg].[py_module_name] [context_var_containing_obj_id] as [varname] %} {% get_comment_count for [pkg].[py_module_name] [context_var_containing_obj_id] as [varname] %}
Example usage:: Example usage::
{% get_comment_count for lcom.eventtimes event.id as comment_count %} {% get_comment_count for lcom.eventtimes event.id as comment_count %}
Note: ``[context_var_containing_obj_id]`` can also be a hard-coded integer, like this:: Note: ``[context_var_containing_obj_id]`` can also be a hard-coded integer, like this::
{% get_comment_count for lcom.eventtimes 23 as comment_count %} {% get_comment_count for lcom.eventtimes 23 as comment_count %}
""" """
def __init__(self, free, tag_name): def __init__(self, free, tag_name):
@ -297,22 +297,22 @@ class DoGetCommentList:
Gets comments for the given params and populates the template context with a Gets comments for the given params and populates the template context with a
special comment_package variable, whose name is defined by the ``as`` special comment_package variable, whose name is defined by the ``as``
clause. clause.
Syntax:: Syntax::
{% get_comment_list for [pkg].[py_module_name] [context_var_containing_obj_id] as [varname] (reversed) %} {% get_comment_list for [pkg].[py_module_name] [context_var_containing_obj_id] as [varname] (reversed) %}
Example usage:: Example usage::
{% get_comment_list for lcom.eventtimes event.id as comment_list %} {% get_comment_list for lcom.eventtimes event.id as comment_list %}
Note: ``[context_var_containing_obj_id]`` can also be a hard-coded integer, like this:: Note: ``[context_var_containing_obj_id]`` can also be a hard-coded integer, like this::
{% get_comment_list for lcom.eventtimes 23 as comment_list %} {% get_comment_list for lcom.eventtimes 23 as comment_list %}
To get a list of comments in reverse order -- that is, most recent first -- To get a list of comments in reverse order -- that is, most recent first --
pass ``reversed`` as the last param:: pass ``reversed`` as the last param::
{% get_comment_list for lcom.eventtimes event.id as comment_list reversed %} {% get_comment_list for lcom.eventtimes event.id as comment_list reversed %}
""" """
def __init__(self, free, tag_name): def __init__(self, free, tag_name):
@ -348,9 +348,9 @@ class DoGetCommentList:
if len(tokens) == 7: if len(tokens) == 7:
if tokens[6] != 'reversed': if tokens[6] != 'reversed':
raise template.TemplateSyntaxError, "Final argument in '%s' must be 'reversed' if given" % self.tag_name raise template.TemplateSyntaxError, "Final argument in '%s' must be 'reversed' if given" % self.tag_name
ordering = "DESC" ordering = "-"
else: else:
ordering = "ASC" ordering = ""
return CommentListNode(package, module, var_name, obj_id, tokens[5], self.free, ordering) return CommentListNode(package, module, var_name, obj_id, tokens[5], self.free, ordering)
# registration comments # registration comments

View File

@ -52,6 +52,36 @@ prep_for_like_query = lambda x: str(x).replace("%", "\%").replace("_", "\_")
# returns the <ul> class for a given radio_admin value # returns the <ul> class for a given radio_admin value
get_ul_class = lambda x: 'radiolist%s' % ((x == HORIZONTAL) and ' inline' or '') get_ul_class = lambda x: 'radiolist%s' % ((x == HORIZONTAL) and ' inline' or '')
# Django currently supports two forms of ordering.
# Form 1 (deprecated) example:
# order_by=(('pub_date', 'DESC'), ('headline', 'ASC'), (None, 'RANDOM'))
# Form 2 (new-style) example:
# order_by=('-pub_date', 'headline', '?')
# Form 1 is deprecated and will no longer be supported for Django's first
# official release. The following code converts from Form 1 to Form 2.
LEGACY_ORDERING_MAPPING = {'ASC': '_', 'DESC': '-_', 'RANDOM': '?'}
def handle_legacy_orderlist(order_list):
if not order_list or isinstance(order_list[0], basestring):
return order_list
else:
# import warnings
new_order_list = [LEGACY_ORDERING_MAPPING[j.upper()].replace('_', str(i)) for i, j in order_list]
# warnings.warn("%r ordering syntax is deprecated. Use %r instead." % (order_list, new_order_list), DeprecationWarning)
return new_order_list
def orderlist2sql(order_list, prefix=''):
output = []
for f in handle_legacy_orderlist(order_list):
if f.startswith('-'):
output.append('%s%s DESC' % (prefix, f[1:]))
elif f == '?':
output.append('RANDOM()')
else:
output.append('%s%s ASC' % (prefix, f))
return ', '.join(output)
def curry(*args, **kwargs): def curry(*args, **kwargs):
def _curried(*moreargs, **morekwargs): def _curried(*moreargs, **morekwargs):
return args[0](*(args[1:]+moreargs), **dict(kwargs.items() + morekwargs.items())) return args[0](*(args[1:]+moreargs), **dict(kwargs.items() + morekwargs.items()))
@ -175,7 +205,7 @@ class Options:
self.get_latest_by = get_latest_by self.get_latest_by = get_latest_by
if order_with_respect_to: if order_with_respect_to:
self.order_with_respect_to = self.get_field(order_with_respect_to) self.order_with_respect_to = self.get_field(order_with_respect_to)
self.ordering = (('_order', 'ASC'),) self.ordering = ('_order',)
else: else:
self.order_with_respect_to = None self.order_with_respect_to = None
self.module_constants = module_constants or {} self.module_constants = module_constants or {}
@ -231,7 +261,7 @@ class Options:
"Returns the full 'ORDER BY' clause for this object, according to self.ordering." "Returns the full 'ORDER BY' clause for this object, according to self.ordering."
if not self.ordering: return '' if not self.ordering: return ''
pre = table_prefix and (table_prefix + '.') or '' pre = table_prefix and (table_prefix + '.') or ''
return 'ORDER BY ' + ','.join(['%s%s %s' % (pre, f, order) for f, order in self.ordering]) return 'ORDER BY ' + orderlist2sql(self.ordering, pre)
def get_add_permission(self): def get_add_permission(self):
return 'add_%s' % self.object_name.lower() return 'add_%s' % self.object_name.lower()
@ -770,7 +800,7 @@ def method_delete(opts, self):
def method_get_next_in_order(opts, order_field, self): def method_get_next_in_order(opts, order_field, self):
if not hasattr(self, '_next_in_order_cache'): if not hasattr(self, '_next_in_order_cache'):
self._next_in_order_cache = opts.get_model_module().get_object(order_by=(('_order', 'ASC'),), self._next_in_order_cache = opts.get_model_module().get_object(order_by=('_order',),
where=['_order > (SELECT _order FROM %s WHERE %s=%%s)' % (opts.db_table, opts.pk.name), where=['_order > (SELECT _order FROM %s WHERE %s=%%s)' % (opts.db_table, opts.pk.name),
'%s=%%s' % order_field.name], limit=1, '%s=%%s' % order_field.name], limit=1,
params=[getattr(self, opts.pk.name), getattr(self, order_field.name)]) params=[getattr(self, opts.pk.name), getattr(self, order_field.name)])
@ -778,7 +808,7 @@ def method_get_next_in_order(opts, order_field, self):
def method_get_previous_in_order(opts, order_field, self): def method_get_previous_in_order(opts, order_field, self):
if not hasattr(self, '_previous_in_order_cache'): if not hasattr(self, '_previous_in_order_cache'):
self._previous_in_order_cache = opts.get_model_module().get_object(order_by=(('_order', 'DESC'),), self._previous_in_order_cache = opts.get_model_module().get_object(order_by=('-_order',),
where=['_order < (SELECT _order FROM %s WHERE %s=%%s)' % (opts.db_table, opts.pk.name), where=['_order < (SELECT _order FROM %s WHERE %s=%%s)' % (opts.db_table, opts.pk.name),
'%s=%%s' % order_field.name], limit=1, '%s=%%s' % order_field.name], limit=1,
params=[getattr(self, opts.pk.name), getattr(self, order_field.name)]) params=[getattr(self, opts.pk.name), getattr(self, order_field.name)])
@ -908,7 +938,7 @@ def method_get_order(ordered_obj, self):
def method_get_next_or_previous(get_object_func, field, is_next, self, **kwargs): def method_get_next_or_previous(get_object_func, field, is_next, self, **kwargs):
kwargs.setdefault('where', []).append('%s %s %%s' % (field.name, (is_next and '>' or '<'))) kwargs.setdefault('where', []).append('%s %s %%s' % (field.name, (is_next and '>' or '<')))
kwargs.setdefault('params', []).append(str(getattr(self, field.name))) kwargs.setdefault('params', []).append(str(getattr(self, field.name)))
kwargs['order_by'] = ((field.name, (is_next and 'ASC' or 'DESC')),) kwargs['order_by'] = [(not is_next and '-' or '') + field.name]
kwargs['limit'] = 1 kwargs['limit'] = 1
return get_object_func(**kwargs) return get_object_func(**kwargs)
@ -1216,16 +1246,20 @@ def function_get_sql_clause(opts, **kwargs):
# ORDER BY clause # ORDER BY clause
order_by = [] order_by = []
for i, j in kwargs.get('order_by', opts.ordering): for f in handle_legacy_orderlist(kwargs.get('order_by', opts.ordering)):
if j == "RANDOM": if f == '?': # Special case.
order_by.append("RANDOM()") order_by.append('RANDOM()')
else: else:
# Append the database table as a column prefix if it wasn't given, # Use the database table as a column prefix if it wasn't given,
# and if the requested column isn't a custom SELECT. # and if the requested column isn't a custom SELECT.
if "." not in i and i not in [k[0] for k in kwargs.get('select', [])]: if "." not in f and f not in [k[0] for k in kwargs.get('select', [])]:
order_by.append("%s.%s %s" % (opts.db_table, i, j)) table_prefix = opts.db_table + '.'
else: else:
order_by.append("%s %s" % (i, j)) table_prefix = ''
if f.startswith('-'):
order_by.append('%s%s DESC' % (table_prefix, f[1:]))
else:
order_by.append('%s%s ASC' % (table_prefix, f))
order_by = ", ".join(order_by) order_by = ", ".join(order_by)
# LIMIT and OFFSET clauses # LIMIT and OFFSET clauses
@ -1246,7 +1280,7 @@ def function_get_in_bulk(opts, klass, *args, **kwargs):
return dict([(o.id, o) for o in obj_list]) return dict([(o.id, o) for o in obj_list])
def function_get_latest(opts, klass, does_not_exist_exception, **kwargs): def function_get_latest(opts, klass, does_not_exist_exception, **kwargs):
kwargs['order_by'] = ((opts.get_latest_by, "DESC"),) kwargs['order_by'] = ('-' + opts.get_latest_by,)
kwargs['limit'] = 1 kwargs['limit'] = 1
return function_get_object(opts, klass, does_not_exist_exception, **kwargs) return function_get_object(opts, klass, does_not_exist_exception, **kwargs)

View File

@ -8,7 +8,7 @@ class Permission(meta.Model):
meta.CharField('codename', 'code name', maxlength=100), meta.CharField('codename', 'code name', maxlength=100),
) )
unique_together = (('package', 'codename'),) unique_together = (('package', 'codename'),)
ordering = (('package', 'ASC'), ('codename', 'ASC')) ordering = ('package', 'codename')
def __repr__(self): def __repr__(self):
return "%s | %s" % (self.package, self.name) return "%s | %s" % (self.package, self.name)
@ -18,7 +18,7 @@ class Group(meta.Model):
meta.CharField('name', 'name', maxlength=80, unique=True), meta.CharField('name', 'name', maxlength=80, unique=True),
meta.ManyToManyField(Permission, blank=True, filter_interface=meta.HORIZONTAL), meta.ManyToManyField(Permission, blank=True, filter_interface=meta.HORIZONTAL),
) )
ordering = (('name', 'ASC'),) ordering = ('name',)
admin = meta.Admin( admin = meta.Admin(
search_fields = ('name',), search_fields = ('name',),
) )
@ -44,7 +44,7 @@ class User(meta.Model):
help_text="In addition to the permissions manually assigned, this user will also get all permissions granted to each group he/she is in."), help_text="In addition to the permissions manually assigned, this user will also get all permissions granted to each group he/she is in."),
meta.ManyToManyField(Permission, name='user_permissions', blank=True, filter_interface=meta.HORIZONTAL), meta.ManyToManyField(Permission, name='user_permissions', blank=True, filter_interface=meta.HORIZONTAL),
) )
ordering = (('username', 'ASC'),) ordering = ('username',)
exceptions = ('SiteProfileNotAvailable',) exceptions = ('SiteProfileNotAvailable',)
admin = meta.Admin( admin = meta.Admin(
fields = ( fields = (
@ -253,7 +253,7 @@ class LogEntry(meta.Model):
meta.PositiveSmallIntegerField('action_flag', 'action flag'), meta.PositiveSmallIntegerField('action_flag', 'action flag'),
meta.TextField('change_message', 'change message', blank=True), meta.TextField('change_message', 'change message', blank=True),
) )
ordering = (('action_time', 'DESC'),) ordering = ('-action_time',)
module_constants = { module_constants = {
'ADDITION': 1, 'ADDITION': 1,
'CHANGE': 2, 'CHANGE': 2,

View File

@ -6,7 +6,7 @@ class Site(meta.Model):
meta.CharField('domain', 'domain name', maxlength=100), meta.CharField('domain', 'domain name', maxlength=100),
meta.CharField('name', 'display name', maxlength=50), meta.CharField('name', 'display name', maxlength=50),
) )
ordering = (('domain', 'ASC'),) ordering = ('domain',)
def __repr__(self): def __repr__(self):
return self.domain return self.domain
@ -22,7 +22,7 @@ class Package(meta.Model):
meta.CharField('label', 'label', maxlength=20, primary_key=True), meta.CharField('label', 'label', maxlength=20, primary_key=True),
meta.CharField('name', 'name', maxlength=30, unique=True), meta.CharField('name', 'name', maxlength=30, unique=True),
) )
ordering = (('name', 'ASC'),) ordering = ('name',)
def __repr__(self): def __repr__(self):
return self.name return self.name
@ -34,7 +34,7 @@ class ContentType(meta.Model):
meta.ForeignKey(Package, name='package'), meta.ForeignKey(Package, name='package'),
meta.CharField('python_module_name', 'Python module name', maxlength=50), meta.CharField('python_module_name', 'Python module name', maxlength=50),
) )
ordering = (('package', 'ASC'), ('name', 'ASC'),) ordering = ('package', 'name')
unique_together = (('package', 'python_module_name'),) unique_together = (('package', 'python_module_name'),)
def __repr__(self): def __repr__(self):
@ -63,7 +63,7 @@ class Redirect(meta.Model):
help_text="This can be either an absolute path (as above) or a full URL starting with 'http://'."), help_text="This can be either an absolute path (as above) or a full URL starting with 'http://'."),
) )
unique_together=(('site_id', 'old_path'),) unique_together=(('site_id', 'old_path'),)
ordering = (('old_path', 'ASC'),) ordering = ('old_path',)
admin = meta.Admin( admin = meta.Admin(
list_display = ('__repr__',), list_display = ('__repr__',),
list_filter = ('site_id',), list_filter = ('site_id',),
@ -87,7 +87,7 @@ class FlatFile(meta.Model):
help_text="If this is checked, only logged-in users will be able to view the page."), help_text="If this is checked, only logged-in users will be able to view the page."),
meta.ManyToManyField(Site), meta.ManyToManyField(Site),
) )
ordering = (('url', 'ASC'),) ordering = ('url',)
admin = meta.Admin( admin = meta.Admin(
fields = ( fields = (
(None, {'fields': ('url', 'title', 'content', 'sites')}), (None, {'fields': ('url', 'title', 'content', 'sites')}),

View File

@ -94,12 +94,15 @@ def change_list(request, app_label, module_name):
# then check the object's default ordering. If neither of those exist, # then check the object's default ordering. If neither of those exist,
# order descending by ID by default. Finally, look for manually-specified # order descending by ID by default. Finally, look for manually-specified
# ordering from the query string. # ordering from the query string.
if lookup_opts.admin.ordering is not None: ordering = lookup_opts.admin.ordering or lookup_opts.ordering or ('-' + lookup_opts.pk.name)
order_field, order_type = lookup_opts.admin.ordering
elif lookup_opts.ordering: # Normalize it to new-style ordering.
order_field, order_type = lookup_opts.ordering[0] ordering = meta.handle_legacy_orderlist(ordering)
if ordering[0].startswith('-'):
order_field, order_type = ordering[0][1:], 'DESC'
else: else:
order_field, order_type = lookup_opts.pk.name, 'DESC' order_field, order_type = ordering[0], 'ASC'
if params.has_key(ORDER_VAR): if params.has_key(ORDER_VAR):
try: try:
try: try:
@ -1071,7 +1074,7 @@ def delete_stage(request, app_label, module_name, object_id):
def history(request, app_label, module_name, object_id): def history(request, app_label, module_name, object_id):
mod, opts = _get_mod_opts(app_label, module_name) mod, opts = _get_mod_opts(app_label, module_name)
action_list = log.get_list(object_id__exact=object_id, content_type_id__exact=opts.get_content_type_id(), action_list = log.get_list(object_id__exact=object_id, content_type_id__exact=opts.get_content_type_id(),
order_by=(("action_time", "ASC"),), select_related=True) order_by=("action_time",), select_related=True)
# If no history was found, see whether this object even exists. # If no history was found, see whether this object even exists.
try: try:
obj = mod.get_object(id__exact=object_id) obj = mod.get_object(id__exact=object_id)

View File

@ -27,7 +27,7 @@ def archive_index(request, app_label, module_name, date_field, num_latest=15, te
if num_latest: if num_latest:
lookup_kwargs.update({ lookup_kwargs.update({
'limit': num_latest, 'limit': num_latest,
'order_by': ((date_field, 'DESC'),), 'order_by': ('-' + date_field,),
}) })
latest = mod.get_list(**lookup_kwargs) latest = mod.get_list(**lookup_kwargs)
else: else: