| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- import re
- from calendar import day_abbr, day_name, month_abbr, month_name
- from datetime import datetime as datetime_
- from datetime import timedelta, timezone
- from functools import lru_cache, partial
- from time import localtime, strftime
- tokens = r"H{1,2}|h{1,2}|m{1,2}|s{1,2}|S+|YYYY|YY|M{1,4}|D{1,4}|Z{1,2}|zz|A|X|x|E|Q|dddd|ddd|d"
- pattern = re.compile(r"(?:{0})|\[(?:{0}|!UTC|)\]".format(tokens))
- def _builtin_datetime_formatter(is_utc, format_string, dt):
- if is_utc:
- dt = dt.astimezone(timezone.utc)
- return dt.strftime(format_string)
- def _loguru_datetime_formatter(is_utc, format_string, formatters, dt):
- if is_utc:
- dt = dt.astimezone(timezone.utc)
- t = dt.timetuple()
- args = tuple(f(t, dt) for f in formatters)
- return format_string % args
- def _default_datetime_formatter(dt):
- return "%04d-%02d-%02d %02d:%02d:%02d.%03d" % (
- dt.year,
- dt.month,
- dt.day,
- dt.hour,
- dt.minute,
- dt.second,
- dt.microsecond // 1000,
- )
- def _format_timezone(tzinfo, *, sep):
- offset = tzinfo.utcoffset(None).total_seconds()
- sign = "+" if offset >= 0 else "-"
- (h, m), s = divmod(abs(offset // 60), 60), abs(offset) % 60
- z = "%s%02d%s%02d" % (sign, h, sep, m)
- if s > 0:
- if s.is_integer():
- z += "%s%02d" % (sep, s)
- else:
- z += "%s%09.06f" % (sep, s)
- return z
- @lru_cache(maxsize=32)
- def _compile_format(spec):
- if spec == "YYYY-MM-DD HH:mm:ss.SSS":
- return _default_datetime_formatter
- is_utc = spec.endswith("!UTC")
- if is_utc:
- spec = spec[:-4]
- if not spec:
- spec = "%Y-%m-%dT%H:%M:%S.%f%z"
- if "%" in spec:
- return partial(_builtin_datetime_formatter, is_utc, spec)
- if "SSSSSSS" in spec:
- raise ValueError(
- "Invalid time format: the provided format string contains more than six successive "
- "'S' characters. This may be due to an attempt to use nanosecond precision, which "
- "is not supported."
- )
- rep = {
- "YYYY": ("%04d", lambda t, dt: t.tm_year),
- "YY": ("%02d", lambda t, dt: t.tm_year % 100),
- "Q": ("%d", lambda t, dt: (t.tm_mon - 1) // 3 + 1),
- "MMMM": ("%s", lambda t, dt: month_name[t.tm_mon]),
- "MMM": ("%s", lambda t, dt: month_abbr[t.tm_mon]),
- "MM": ("%02d", lambda t, dt: t.tm_mon),
- "M": ("%d", lambda t, dt: t.tm_mon),
- "DDDD": ("%03d", lambda t, dt: t.tm_yday),
- "DDD": ("%d", lambda t, dt: t.tm_yday),
- "DD": ("%02d", lambda t, dt: t.tm_mday),
- "D": ("%d", lambda t, dt: t.tm_mday),
- "dddd": ("%s", lambda t, dt: day_name[t.tm_wday]),
- "ddd": ("%s", lambda t, dt: day_abbr[t.tm_wday]),
- "d": ("%d", lambda t, dt: t.tm_wday),
- "E": ("%d", lambda t, dt: t.tm_wday + 1),
- "HH": ("%02d", lambda t, dt: t.tm_hour),
- "H": ("%d", lambda t, dt: t.tm_hour),
- "hh": ("%02d", lambda t, dt: (t.tm_hour - 1) % 12 + 1),
- "h": ("%d", lambda t, dt: (t.tm_hour - 1) % 12 + 1),
- "mm": ("%02d", lambda t, dt: t.tm_min),
- "m": ("%d", lambda t, dt: t.tm_min),
- "ss": ("%02d", lambda t, dt: t.tm_sec),
- "s": ("%d", lambda t, dt: t.tm_sec),
- "S": ("%d", lambda t, dt: dt.microsecond // 100000),
- "SS": ("%02d", lambda t, dt: dt.microsecond // 10000),
- "SSS": ("%03d", lambda t, dt: dt.microsecond // 1000),
- "SSSS": ("%04d", lambda t, dt: dt.microsecond // 100),
- "SSSSS": ("%05d", lambda t, dt: dt.microsecond // 10),
- "SSSSSS": ("%06d", lambda t, dt: dt.microsecond),
- "A": ("%s", lambda t, dt: "AM" if t.tm_hour < 12 else "PM"),
- "Z": ("%s", lambda t, dt: _format_timezone(dt.tzinfo or timezone.utc, sep=":")),
- "ZZ": ("%s", lambda t, dt: _format_timezone(dt.tzinfo or timezone.utc, sep="")),
- "zz": ("%s", lambda t, dt: (dt.tzinfo or timezone.utc).tzname(dt) or ""),
- "X": ("%d", lambda t, dt: dt.timestamp()),
- "x": ("%d", lambda t, dt: int(dt.timestamp() * 1000000 + dt.microsecond)),
- }
- format_string = ""
- formatters = []
- pos = 0
- for match in pattern.finditer(spec):
- start, end = match.span()
- format_string += spec[pos:start]
- pos = end
- token = match.group(0)
- try:
- specifier, formatter = rep[token]
- except KeyError:
- format_string += token[1:-1]
- else:
- format_string += specifier
- formatters.append(formatter)
- format_string += spec[pos:]
- return partial(_loguru_datetime_formatter, is_utc, format_string, formatters)
- class datetime(datetime_): # noqa: N801
- def __format__(self, fmt):
- return _compile_format(fmt)(self)
- def aware_now():
- now = datetime_.now()
- timestamp = now.timestamp()
- local = localtime(timestamp)
- try:
- seconds = local.tm_gmtoff
- zone = local.tm_zone
- except AttributeError:
- # Workaround for Python 3.5.
- utc_naive = datetime_.fromtimestamp(timestamp, tz=timezone.utc).replace(tzinfo=None)
- offset = datetime_.fromtimestamp(timestamp) - utc_naive
- seconds = offset.total_seconds()
- zone = strftime("%Z")
- tzinfo = timezone(timedelta(seconds=seconds), zone)
- return datetime.combine(now.date(), now.time().replace(tzinfo=tzinfo))
|