django-orchestra-test/orchestra/utils/humanize.py

152 lines
4.3 KiB
Python
Raw Permalink Normal View History

2023-07-09 07:51:51 +00:00
from datetime import datetime
from django.utils import timezone
from django.utils.translation import ngettext, gettext as _
def verbose_time(n, units, ago='ago'):
if n >= 5:
return _("{n} {units} {ago}").format(n=int(n), units=units, ago=ago)
return ngettext(
_("{n:.1f} {s_units} {ago}"),
_("{n:.1f} {units} {ago}"), n
).format(n=n, units=units, s_units=units[:-1], ago=ago)
OLDER_CHUNKS = (
(365.0, 'years'),
(30.0, 'months'),
(7.0, 'weeks'),
)
def _un(singular__plural, n=None):
singular, plural = singular__plural
return ngettext(singular, plural, n)
def naturaldatetime(date, show_seconds=False):
"""Convert datetime into a human natural date string."""
if not date:
return ''
right_now = timezone.now()
today = datetime(right_now.year, right_now.month,
right_now.day, tzinfo=right_now.tzinfo)
delta = right_now - date
delta_midnight = today - date
days = delta.days
hours = float(delta.seconds) / 3600
minutes = float(delta.seconds) / 60
seconds = delta.seconds
days = abs(days)
ago = ''
if right_now > date:
ago = 'ago'
if days == 0:
if int(hours) == 0:
if minutes >= 1 or not show_seconds:
return verbose_time(minutes, 'minutes', ago=ago)
else:
return verbose_time(seconds, 'seconds', ago=ago)
else:
return verbose_time(hours, 'hours', ago=ago)
if delta_midnight.days == 0:
date = timezone.localtime(date)
return _("yesterday at {time}").format(time=date.strftime('%H:%M'))
count = 0
for chunk, units in OLDER_CHUNKS:
if days < 7.0:
count = days + float(hours)/24
return verbose_time(count, 'days', ago=ago)
if days >= chunk:
count = (delta_midnight.days + 1) / chunk
count = abs(count)
return verbose_time(count, units, ago=ago)
def naturaldate(date):
if not date:
return ''
today = timezone.now().date()
delta = today - date
days = delta.days
if days == 0:
return _('today')
elif days == 1:
return _('yesterday')
ago = ' ago'
if days < 0 or today < date:
ago = ''
days = abs(days)
delta_midnight = today - date
count = 0
for chunk, units in OLDER_CHUNKS:
if days < 7.0:
count = days
fmt = verbose_time(count, 'days', ago=ago)
return fmt.format(num=count, ago=ago)
if days >= chunk:
count = (delta_midnight.days + 1) / chunk
count = abs(count)
fmt = verbose_time(count, units, ago=ago)
return fmt.format(num=count, ago=ago)
def text2int(textnum, numwords={}):
if not numwords:
units = (
'zero', 'one', 'two', 'three', 'four', 'five', 'six', 'seven', 'eight',
'nine', 'ten', 'eleven', 'twelve', 'thirteen', 'fourteen', 'fifteen',
'sixteen', 'seventeen', 'eighteen', 'nineteen',
)
tens = ('', '', 'twenty', 'thirty', 'forty', 'fifty', 'sixty', 'seventy', 'eighty', 'ninety')
scales = ['hundred', 'thousand', 'million', 'billion', 'trillion']
numwords['and'] = (1, 0)
for idx, word in enumerate(units):
numwords[word] = (1, idx)
for idx, word in enumerate(tens):
numwords[word] = (1, idx * 10)
for idx, word in enumerate(scales):
numwords[word] = (10 ** (idx * 3 or 2), 0)
current = result = 0
for word in textnum.split():
word = word.lower()
if word not in numwords:
raise Exception("Illegal word: " + word)
scale, increment = numwords[word]
current = current * scale + increment
if scale > 100:
result += current
current = 0
return result + current
UNITS_CONVERSIONS = {
1024**4: ['TB', 'TiB', 'TERABYTES'],
1024**3: ['GB', 'GiB', 'GYGABYTES'],
1024**2: ['MB', 'MiB', 'MEGABYTES'],
1024: ['KB', 'KiB', 'KYLOBYTES'],
1: ['B', 'BYTES'],
}
def unit_to_bytes(unit):
for bytes, units in UNITS_CONVERSIONS.items():
if unit in units:
return bytes
raise KeyError("%s is not a valid unit." % unit)