aemdy
aemdy

Reputation: 3802

Global decimal rounding options in Django

Decimal numbers are by default rounded very unexpectedly, in order to make it work normally, it is needed to use ROUND_HALF_UP option.

>>> from decimal import *
>>> Decimal("2.5").quantize(Decimal(1))
Decimal('2')
>>> getcontext().rounding = ROUND_HALF_UP
>>> Decimal("2.5").quantize(Decimal(1))
Decimal('3')
>>> Decimal("2.4").quantize(Decimal(1))
Decimal('2')

My question is - where in the Django application I have to set rounding option, so that it would work globally in the project? By saying globally I mean templates (floatformat template tag), views, model decimal field and so on.

Upvotes: 1

Views: 2724

Answers (4)

Yahor Tsyplakou
Yahor Tsyplakou

Reputation: 88

For django project can work setting decimal.DefaultContext (py3, py2).

This context is most useful in multi-threaded environments.

This is my code from settings.py:

import decimal
# Set global decimal rounding to ROUND_HALF_UP (instead of ROUND_HALF_EVEN).
project_context = decimal.getcontext()
project_context.rounding = decimal.ROUND_HALF_UP
decimal.DefaultContext = project_context

Worked in 1.10. Based on my answer in this question.

Upvotes: 0

Yauhen Yakimovich
Yauhen Yakimovich

Reputation: 14211

Worked in 1.9.5 (based on comment from @ark):

In myapp/apps.py

from __future__ import unicode_literals
import decimal
from django.apps import AppConfig


class MyAppConfig(AppConfig):

    name = 'myapp'

    def ready(self):
        # Set precision
        decimal.getcontext().prec = 9
        decimal.getcontext().rounding = decimal.ROUND_HALF_DOWN

In settings.py

INSTALLED_APPS = list(INSTALLED_APPS)
INSTALLED_APPS.append('myapp.apps.MyAppConfig')

Upvotes: 3

Andrea Rabbaglietti
Andrea Rabbaglietti

Reputation: 948

Actually it doesn't work like Viktor suggested (although in django 1.5).

My solution is create and using a middleware like this:

# -*- coding: utf-8 -*-

import decimal
from django.conf import settings


class DecimalPrecisionMiddleware(object):
    def process_request(self, request):
        decimal_context = decimal.getcontext()
        decimal_context.prec = settings.DECIMAL_PRECISION # say: 4

and then in settings.py:

MIDDLEWARE_CLASSES = (
    'pathto.middleware.DecimalPrecisionMiddleware',
    # etc..
)

Upvotes: 0

Viktor Kerkez
Viktor Kerkez

Reputation: 46586

Decimal doesn't have anything to do with Django, they are part of the standard python library. The getcontext function returns the context of the current thread, so if you're not doing anything funky, every request will be executed in one thread. That basically mean that setting the option in the settings.py file should be enough.

Upvotes: 1

Related Questions