152 lines
4.3 KiB
Python
152 lines
4.3 KiB
Python
|
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)
|