codeburst

Bursts of code to power through your day. Web Development articles, tutorials, and news.

Follow publication

Django Admin Range-Based Date Hierarchy

Haki Benita
codeburst
Published in
6 min readDec 12, 2017

--

Identifying the problem

/admin/transactions/created__year=2017&created__month=11
WHERE created BETWEEN
'2017-01-01T00:00:00+00:00'::timestamptz AND
'2017-12-31T23:59:59.999999+00:00'::timestamptz
AND EXTRACT('month' FROM created AT TIME ZONE 'UTC') = 11)
Index Scan using transactions_transaction_created_8fd61a4b on transactions_transaction (cost=0.43..90663.65 rows=4561 width=471)Index Cond: ((created >= '2017-01-01 02:00:00+02'::timestamp with time zone) AND (created <= '2018-01-01 01:59:59.999999+02'::timestamp with time zone))Filter: (date_part('month'::text, timezone('UTC'::text, created)) = '11'::double precision)
analyze transaction_transaction (created);
WHERE created BETWEEN
'2017-01-01T00:00:00+00:00'::timestamptz AND
'2017-12-31T23:59:59.999999+00:00'::timestamptz
AND EXTRACT('month' FROM created AT TIME ZONE 'UTC') = 11)
WHERE created BETWEEN
'2017-11-01T00:00:00+00:00'::timestamptz AND
'2017-11-30T23:59:59.999999+00:00'::timestamptz
Index Scan using transactions_transaction_created_8fd61a4b on transactions_transaction (cost=0.43..1265.75 rows=13716 width=471)
Index Cond: ((created >= ‘2017–11–01 02:00:00+02’::timestamp with time zone) AND (created <= ‘2017–12–01 01:59:59.999999+02’::timestamp with time zone))
Photo by Braden Collum on Unsplash

The problem with the way date hierarchy is implemented

Function based index

create index transactions_transaction_created_month_brin on transactions_transaction using brin(extract(‘month’ from created at time zone ‘UTC’));

Simplify the condition used by Django date hierarchy

The implementation

class RangeBasedDateHierarchyListFilter(admin.ListFilter):
title = ''
def __init__(self, request, params, model, model_admin):
self.date_hierarchy_field = model_admin.date_hierarchy
self.date_hierarchy = {}
date_hierarchy_field_re = re.compile(
r'^{}__(day|month|year)$'.format(
self.date_hierarchy_field

))
for param in list(params.keys()):
match = date_hierarchy_field_re.match(param)
if match:
period = match.group(1)
self.date_hierarchy[period] = int(params.pop(param))
class RangeBasedDateHierarchyListFilter(admin.ListFilter):

def queryset(self, request, queryset):
tz = timezone.get_default_timezone()
from_date, to_date = get_date_range_for_hierarchy(self.date_hierarchy, tz)
return queryset.filter(**{
'{}__gte'.format(self.date_hierarchy_field): from_date,
'{}__lt'.format(self.date_hierarchy_field): to_date,
})
def get_date_range_for_hierarchy(date_hierarchy, tz):
"""Generate date range for date hierarchy.
date_hierarchy <dict>:
year (int)
month (int or None)
day (int or None)
tz <timezone or None>:
The timezone in which to generate the datetimes.
If None, the datetimes will be naive.
Returns (tuple):
from_date (datetime.datetime, aware if tz is set) inclusive
to_date (datetime.datetime, aware if tz is set) exclusive
"""
from_date = datetime.datetime(
date_hierarchy['year'],
date_hierarchy.get('month', 1),
date_hierarchy.get('day', 1),
)
if tz:
from_date = tz.localize(from_date)
if 'day' in date_hierarchy:
to_date = from_date + datetime.timedelta(days=1)
elif 'month' in date_hierarchy:
assert from_date.day == 1
to_date = (from_date + datetime.timedelta(days=32)).replace(day=1)
else:
to_date = from_date.replace(year=from_date.year + 1)
return from_date, to_date
class TransactionAdmin(admin.ModelAdmin):
...
date_hierarchy = 'created'
list_filter = (
...
RangeBasedDateHierarchyListFilter,
)
...

The result

Free

Distraction-free reading. No ads.

Organize your knowledge with lists and highlights.

Tell your story. Find your audience.

Membership

Read member-only stories

Support writers you read most

Earn money for your writing

Listen to audio narrations

Read offline with the Medium app

--

--

Published in codeburst

Bursts of code to power through your day. Web Development articles, tutorials, and news.

Written by Haki Benita

Full Stack Developer, Team Leader, Independent. More from me at https://hakibenita.com

Responses (1)

Write a response