2021-08-21 23:27:15 +08:00
|
|
|
import math
|
2016-03-22 09:06:54 +08:00
|
|
|
from decimal import Decimal
|
|
|
|
|
|
|
|
from django.core import validators
|
|
|
|
from django.core.exceptions import ValidationError
|
2020-12-11 01:00:57 +08:00
|
|
|
from django.db import models
|
2016-03-22 09:06:54 +08:00
|
|
|
from django.test import TestCase
|
|
|
|
|
|
|
|
from .models import BigD, Foo
|
|
|
|
|
|
|
|
|
|
|
|
class DecimalFieldTests(TestCase):
|
|
|
|
|
|
|
|
def test_to_python(self):
|
|
|
|
f = models.DecimalField(max_digits=4, decimal_places=2)
|
|
|
|
self.assertEqual(f.to_python(3), Decimal('3'))
|
|
|
|
self.assertEqual(f.to_python('3.14'), Decimal('3.14'))
|
2017-05-03 17:40:09 +08:00
|
|
|
# to_python() converts floats and honors max_digits.
|
|
|
|
self.assertEqual(f.to_python(3.1415926535897), Decimal('3.142'))
|
|
|
|
self.assertEqual(f.to_python(2.4), Decimal('2.400'))
|
|
|
|
# Uses default rounding of ROUND_HALF_EVEN.
|
|
|
|
self.assertEqual(f.to_python(2.0625), Decimal('2.062'))
|
|
|
|
self.assertEqual(f.to_python(2.1875), Decimal('2.188'))
|
2020-06-06 03:13:36 +08:00
|
|
|
|
|
|
|
def test_invalid_value(self):
|
|
|
|
field = models.DecimalField(max_digits=4, decimal_places=2)
|
|
|
|
msg = '“%s” value must be a decimal number.'
|
|
|
|
tests = [
|
|
|
|
(),
|
|
|
|
[],
|
|
|
|
{},
|
|
|
|
set(),
|
|
|
|
object(),
|
|
|
|
complex(),
|
|
|
|
'non-numeric string',
|
|
|
|
b'non-numeric byte-string',
|
|
|
|
]
|
|
|
|
for value in tests:
|
|
|
|
with self.subTest(value):
|
|
|
|
with self.assertRaisesMessage(ValidationError, msg % (value,)):
|
|
|
|
field.clean(value, None)
|
2016-03-22 09:06:54 +08:00
|
|
|
|
|
|
|
def test_default(self):
|
|
|
|
f = models.DecimalField(default=Decimal('0.00'))
|
|
|
|
self.assertEqual(f.get_default(), Decimal('0.00'))
|
|
|
|
|
2016-04-24 01:13:31 +08:00
|
|
|
def test_get_prep_value(self):
|
2016-03-22 09:06:54 +08:00
|
|
|
f = models.DecimalField(max_digits=5, decimal_places=1)
|
2016-06-17 02:19:18 +08:00
|
|
|
self.assertIsNone(f.get_prep_value(None))
|
2016-04-24 01:13:31 +08:00
|
|
|
self.assertEqual(f.get_prep_value('2.4'), Decimal('2.4'))
|
2016-03-22 09:06:54 +08:00
|
|
|
|
|
|
|
def test_filter_with_strings(self):
|
|
|
|
"""
|
|
|
|
Should be able to filter decimal fields using strings (#8023).
|
|
|
|
"""
|
|
|
|
foo = Foo.objects.create(a='abc', d=Decimal('12.34'))
|
|
|
|
self.assertEqual(list(Foo.objects.filter(d='12.34')), [foo])
|
|
|
|
|
|
|
|
def test_save_without_float_conversion(self):
|
|
|
|
"""
|
|
|
|
Ensure decimals don't go through a corrupting float conversion during
|
|
|
|
save (#5079).
|
|
|
|
"""
|
|
|
|
bd = BigD(d='12.9')
|
|
|
|
bd.save()
|
|
|
|
bd = BigD.objects.get(pk=bd.pk)
|
|
|
|
self.assertEqual(bd.d, Decimal('12.9'))
|
|
|
|
|
2021-08-21 23:27:15 +08:00
|
|
|
def test_save_nan_invalid(self):
|
|
|
|
msg = '“nan” value must be a decimal number.'
|
|
|
|
with self.assertRaisesMessage(ValidationError, msg):
|
|
|
|
BigD.objects.create(d=float('nan'))
|
|
|
|
with self.assertRaisesMessage(ValidationError, msg):
|
|
|
|
BigD.objects.create(d=math.nan)
|
|
|
|
|
2017-04-28 12:07:28 +08:00
|
|
|
def test_fetch_from_db_without_float_rounding(self):
|
|
|
|
big_decimal = BigD.objects.create(d=Decimal('.100000000000000000000000000005'))
|
|
|
|
big_decimal.refresh_from_db()
|
|
|
|
self.assertEqual(big_decimal.d, Decimal('.100000000000000000000000000005'))
|
|
|
|
|
2016-03-22 09:06:54 +08:00
|
|
|
def test_lookup_really_big_value(self):
|
|
|
|
"""
|
|
|
|
Really big values can be used in a filter statement.
|
|
|
|
"""
|
|
|
|
# This should not crash.
|
|
|
|
Foo.objects.filter(d__gte=100000000000)
|
|
|
|
|
|
|
|
def test_max_digits_validation(self):
|
|
|
|
field = models.DecimalField(max_digits=2)
|
|
|
|
expected_message = validators.DecimalValidator.messages['max_digits'] % {'max': 2}
|
|
|
|
with self.assertRaisesMessage(ValidationError, expected_message):
|
|
|
|
field.clean(100, None)
|
|
|
|
|
|
|
|
def test_max_decimal_places_validation(self):
|
|
|
|
field = models.DecimalField(decimal_places=1)
|
|
|
|
expected_message = validators.DecimalValidator.messages['max_decimal_places'] % {'max': 1}
|
|
|
|
with self.assertRaisesMessage(ValidationError, expected_message):
|
|
|
|
field.clean(Decimal('0.99'), None)
|
|
|
|
|
|
|
|
def test_max_whole_digits_validation(self):
|
|
|
|
field = models.DecimalField(max_digits=3, decimal_places=1)
|
|
|
|
expected_message = validators.DecimalValidator.messages['max_whole_digits'] % {'max': 2}
|
|
|
|
with self.assertRaisesMessage(ValidationError, expected_message):
|
|
|
|
field.clean(Decimal('999'), None)
|
2017-12-12 22:07:01 +08:00
|
|
|
|
|
|
|
def test_roundtrip_with_trailing_zeros(self):
|
|
|
|
"""Trailing zeros in the fractional part aren't truncated."""
|
|
|
|
obj = Foo.objects.create(a='bar', d=Decimal('8.320'))
|
|
|
|
obj.refresh_from_db()
|
|
|
|
self.assertEqual(obj.d.compare_total(Decimal('8.320')), Decimal('0'))
|