Fixed #27486 -- Fixed Python 3.7 DeprecationWarning in intword and filesizeformat filters.
intword and filesizeformat passed floats to ngettext() which is deprecated in Python 3.7. The rationale for this warning is documented in BPO-28692: https://bugs.python.org/issue28692. For filesizeformat, the filesize value is expected to be an int -- it fills %d string formatting placeholders. It was likely coerced to a float to ensure floating point division on Python 2. Python 3 always does floating point division, so coerce to an int instead of a float to fix the warning. For intword, the number may contain a decimal component. In English, a decimal component makes the noun plural. A helper function, round_away_from_one(), was added to convert the float to an integer that is appropriate for ngettext().
This commit is contained in:
parent
175656e166
commit
9e38ed0536
|
@ -10,7 +10,7 @@ from django.utils.safestring import mark_safe
|
||||||
from django.utils.timezone import is_aware, utc
|
from django.utils.timezone import is_aware, utc
|
||||||
from django.utils.translation import (
|
from django.utils.translation import (
|
||||||
gettext as _, gettext_lazy, ngettext, ngettext_lazy, npgettext_lazy,
|
gettext as _, gettext_lazy, ngettext, ngettext_lazy, npgettext_lazy,
|
||||||
pgettext,
|
pgettext, round_away_from_one,
|
||||||
)
|
)
|
||||||
|
|
||||||
register = template.Library()
|
register = template.Library()
|
||||||
|
@ -158,7 +158,8 @@ def intword(value):
|
||||||
large_number = 10 ** exponent
|
large_number = 10 ** exponent
|
||||||
if value < large_number * 1000:
|
if value < large_number * 1000:
|
||||||
new_value = value / large_number
|
new_value = value / large_number
|
||||||
return _check_for_i18n(new_value, *converters(new_value))
|
rounded_value = round_away_from_one(new_value)
|
||||||
|
return _check_for_i18n(new_value, *converters(rounded_value))
|
||||||
return value
|
return value
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -812,7 +812,7 @@ def filesizeformat(bytes_):
|
||||||
102 bytes, etc.).
|
102 bytes, etc.).
|
||||||
"""
|
"""
|
||||||
try:
|
try:
|
||||||
bytes_ = float(bytes_)
|
bytes_ = int(bytes_)
|
||||||
except (TypeError, ValueError, UnicodeDecodeError):
|
except (TypeError, ValueError, UnicodeDecodeError):
|
||||||
value = ngettext("%(size)d byte", "%(size)d bytes", 0) % {'size': 0}
|
value = ngettext("%(size)d byte", "%(size)d bytes", 0) % {'size': 0}
|
||||||
return avoid_wrapping(value)
|
return avoid_wrapping(value)
|
||||||
|
|
|
@ -4,6 +4,7 @@ Internationalization support.
|
||||||
import re
|
import re
|
||||||
import warnings
|
import warnings
|
||||||
from contextlib import ContextDecorator
|
from contextlib import ContextDecorator
|
||||||
|
from decimal import ROUND_UP, Decimal
|
||||||
|
|
||||||
from django.utils.autoreload import autoreload_started, file_changed
|
from django.utils.autoreload import autoreload_started, file_changed
|
||||||
from django.utils.deprecation import RemovedInDjango40Warning
|
from django.utils.deprecation import RemovedInDjango40Warning
|
||||||
|
@ -332,3 +333,7 @@ trim_whitespace_re = re.compile(r'\s*\n\s*')
|
||||||
|
|
||||||
def trim_whitespace(s):
|
def trim_whitespace(s):
|
||||||
return trim_whitespace_re.sub(' ', s.strip())
|
return trim_whitespace_re.sub(' ', s.strip())
|
||||||
|
|
||||||
|
|
||||||
|
def round_away_from_one(value):
|
||||||
|
return int(Decimal(value - 1).quantize(Decimal('0'), rounding=ROUND_UP)) + 1
|
||||||
|
|
|
@ -57,7 +57,9 @@ e.g. with the ``'de'`` language:
|
||||||
===========
|
===========
|
||||||
|
|
||||||
Converts a large integer (or a string representation of an integer) to a
|
Converts a large integer (or a string representation of an integer) to a
|
||||||
friendly text representation. Works best for numbers over 1 million.
|
friendly text representation. Translates ``1.0`` as a singular phrase and all
|
||||||
|
other numeric values as plural, this may be incorrect for some languages. Works
|
||||||
|
best for numbers over 1 million.
|
||||||
|
|
||||||
Examples:
|
Examples:
|
||||||
|
|
||||||
|
@ -74,6 +76,11 @@ e.g. with the ``'de'`` language:
|
||||||
* ``1200000`` becomes ``'1,2 Millionen'``.
|
* ``1200000`` becomes ``'1,2 Millionen'``.
|
||||||
* ``1200000000`` becomes ``'1,2 Milliarden'``.
|
* ``1200000000`` becomes ``'1,2 Milliarden'``.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.0
|
||||||
|
|
||||||
|
All numeric values are now translated as plural, except ``1.0`` which is
|
||||||
|
translated as a singular phrase. This may be incorrect for some languages.
|
||||||
|
|
||||||
.. templatefilter:: naturalday
|
.. templatefilter:: naturalday
|
||||||
|
|
||||||
``naturalday``
|
``naturalday``
|
||||||
|
|
|
@ -416,6 +416,9 @@ Miscellaneous
|
||||||
when ``doseq=False``, rather than iterating them, bringing it into line with
|
when ``doseq=False``, rather than iterating them, bringing it into line with
|
||||||
the standard library :func:`urllib.parse.urlencode` function.
|
the standard library :func:`urllib.parse.urlencode` function.
|
||||||
|
|
||||||
|
* ``intword`` template filter now translates ``1.0`` as a singular phrase and
|
||||||
|
all other numeric values as plural. This may be incorrect for some languages.
|
||||||
|
|
||||||
.. _deprecated-features-3.0:
|
.. _deprecated-features-3.0:
|
||||||
|
|
||||||
Features deprecated in 3.0
|
Features deprecated in 3.0
|
||||||
|
|
|
@ -34,9 +34,9 @@ from django.utils.translation import (
|
||||||
LANGUAGE_SESSION_KEY, activate, check_for_language, deactivate,
|
LANGUAGE_SESSION_KEY, activate, check_for_language, deactivate,
|
||||||
get_language, get_language_bidi, get_language_from_request,
|
get_language, get_language_bidi, get_language_from_request,
|
||||||
get_language_info, gettext, gettext_lazy, ngettext, ngettext_lazy,
|
get_language_info, gettext, gettext_lazy, ngettext, ngettext_lazy,
|
||||||
npgettext, npgettext_lazy, pgettext, to_language, to_locale, trans_null,
|
npgettext, npgettext_lazy, pgettext, round_away_from_one, to_language,
|
||||||
trans_real, ugettext, ugettext_lazy, ugettext_noop, ungettext,
|
to_locale, trans_null, trans_real, ugettext, ugettext_lazy, ugettext_noop,
|
||||||
ungettext_lazy,
|
ungettext, ungettext_lazy,
|
||||||
)
|
)
|
||||||
from django.utils.translation.reloader import (
|
from django.utils.translation.reloader import (
|
||||||
translation_file_changed, watch_for_translation_changes,
|
translation_file_changed, watch_for_translation_changes,
|
||||||
|
@ -1883,3 +1883,31 @@ class TranslationFileChangedTests(SimpleTestCase):
|
||||||
self.assertEqual(trans_real._translations, {})
|
self.assertEqual(trans_real._translations, {})
|
||||||
self.assertIsNone(trans_real._default)
|
self.assertIsNone(trans_real._default)
|
||||||
self.assertIsInstance(trans_real._active, _thread._local)
|
self.assertIsInstance(trans_real._active, _thread._local)
|
||||||
|
|
||||||
|
|
||||||
|
class UtilsTests(SimpleTestCase):
|
||||||
|
def test_round_away_from_one(self):
|
||||||
|
tests = [
|
||||||
|
(0, 0),
|
||||||
|
(0., 0),
|
||||||
|
(0.25, 0),
|
||||||
|
(0.5, 0),
|
||||||
|
(0.75, 0),
|
||||||
|
(1, 1),
|
||||||
|
(1., 1),
|
||||||
|
(1.25, 2),
|
||||||
|
(1.5, 2),
|
||||||
|
(1.75, 2),
|
||||||
|
(-0., 0),
|
||||||
|
(-0.25, -1),
|
||||||
|
(-0.5, -1),
|
||||||
|
(-0.75, -1),
|
||||||
|
(-1, -1),
|
||||||
|
(-1., -1),
|
||||||
|
(-1.25, -2),
|
||||||
|
(-1.5, -2),
|
||||||
|
(-1.75, -2),
|
||||||
|
]
|
||||||
|
for value, expected in tests:
|
||||||
|
with self.subTest(value=value):
|
||||||
|
self.assertEqual(round_away_from_one(value), expected)
|
||||||
|
|
Loading…
Reference in New Issue