Source code for pywikibot.time

"""Time handling module.

.. versionadded:: 7.5
"""
#
# (C) Pywikibot team, 2007-2022
#
# Distributed under the terms of the MIT license.
#
import datetime
import math
import re
import types
from typing import Type, Union

import pywikibot
from pywikibot.backports import Tuple
from pywikibot.tools import classproperty


__all__ = (
    'parse_duration',
    'str2timedelta',
    'MW_KEYS',
    'Timestamp',
    'TZoneFixedOffset'
)

#: .. versionadded:: 7.5
MW_KEYS = types.MappingProxyType({
    's': 'seconds',
    'h': 'hours',
    'd': 'days',
    'w': 'weeks',
    'y': 'years',
    # 'months' and 'minutes' were removed because confusion outweighs merit
})


[docs]class Timestamp(datetime.datetime): """Class for handling MediaWiki timestamps. This inherits from datetime.datetime, so it can use all of the methods and operations of a datetime object. To ensure that the results of any operation are also a Timestamp object, be sure to use only Timestamp objects (and datetime.timedeltas) in any operation. Use Timestamp.fromISOformat() and Timestamp.fromtimestampformat() to create Timestamp objects from MediaWiki string formats. As these constructors are typically used to create objects using data passed provided by site and page methods, some of which return a Timestamp when previously they returned a MediaWiki string representation, these methods also accept a Timestamp object, in which case they return a clone. Alternatively, Timestamp.set_timestamp() can create Timestamp objects from Timestamp, datetime.datetime object, or strings compliant with ISO8601, MW, or POSIX formats. Use Site.server_time() for the current time; this is more reliable than using Timestamp.utcnow(). .. versionchanged:: 7.5 moved to :mod:`time` module """ mediawikiTSFormat = '%Y%m%d%H%M%S' # noqa: N815 _ISO8601Format_new = '{0:+05d}-{1:02d}-{2:02d}T{3:02d}:{4:02d}:{5:02d}Z'
[docs] @classmethod def set_timestamp(cls: Type['Timestamp'], ts: Union[str, datetime.datetime, 'Timestamp'] ) -> 'Timestamp': """Set Timestamp from input object. ts is converted to a datetime naive object representing UTC time. String shall be compliant with: - Mediwiki timestamp format: YYYYMMDDHHMMSS - ISO8601 format: YYYY-MM-DD[T ]HH:MM:SS[Z|±HH[MM[SS[.ffffff]]]] - POSIX format: seconds from Unix epoch S{1,13}[.ffffff]] .. versionadded:: 7.5 :param ts: Timestamp, datetime.datetime or str :return: Timestamp object :raises ValueError: conversion failed """ if isinstance(ts, cls): return ts if isinstance(ts, datetime.datetime): return cls._from_datetime(ts) if isinstance(ts, str): return cls._from_string(ts) raise ValueError('Unsupported "ts" type, got "{}" ({})' .format(ts, type(ts).__name__))
@staticmethod def _from_datetime(dt: datetime.datetime) -> 'Timestamp': """Convert a datetime.datetime timestamp to a Timestamp object. .. versionadded:: 7.5 """ return Timestamp(dt.year, dt.month, dt.day, dt.hour, dt.minute, dt.second, dt.microsecond, dt.tzinfo) @classmethod def _from_mw(cls: Type['Timestamp'], timestr: str) -> 'Timestamp': """Convert a string in MW format to a Timestamp object. Mediwiki timestamp format: YYYYMMDDHHMMSS .. versionadded:: 7.5 """ RE_MW = r'\d{14}$' # noqa: N806 m = re.match(RE_MW, timestr) if not m: msg = "time data '{timestr}' does not match MW format." raise ValueError(msg.format(timestr=timestr)) return cls.strptime(timestr, cls.mediawikiTSFormat) @classmethod def _from_iso8601(cls: Type['Timestamp'], timestr: str) -> 'Timestamp': """Convert a string in ISO8601 format to a Timestamp object. ISO8601 format: - YYYY-MM-DD[T ]HH:MM:SS[[.,]ffffff][Z|±HH[MM[SS[.ffffff]]]] .. versionadded:: 7.5 """ RE_ISO8601 = (r'(?:\d{4}-\d{2}-\d{2})(?P<sep>[T ])' # noqa: N806 r'(?:\d{2}:\d{2}:\d{2})(?P<u>[.,]\d{1,6})?' r'(?P<tz>Z|[+\-]\d{2}:?\d{,2})?$' ) m = re.match(RE_ISO8601, timestr) if not m: msg = "time data '{timestr}' does not match ISO8601 format." raise ValueError(msg.format(timestr=timestr)) strpfmt = '%Y-%m-%d{sep}%H:%M:%S'.format(sep=m.group('sep')) strpstr = timestr[:19] if m['u']: strpfmt += '.%f' strpstr += m['u'].replace(',', '.') # .ljust(7, '0') if m['tz']: if m['tz'] == 'Z': strpfmt += 'Z' strpstr += 'Z' else: strpfmt += '%z' # strptime wants HHMM, without ':' strpstr += (m['tz'].replace(':', '')).ljust(5, '0') ts = cls.strptime(strpstr, strpfmt) if ts.tzinfo is not None: ts = ts.astimezone(datetime.timezone.utc).replace(tzinfo=None) # why pytest in py35/py37 fails without this? ts = cls._from_datetime(ts) return ts @classmethod def _from_posix(cls: Type['Timestamp'], timestr: str) -> 'Timestamp': """Convert a string in POSIX format to a Timestamp object. POSIX format: SECONDS[.ffffff]] .. versionadded:: 7.5 """ RE_POSIX = r'(?P<S>-?\d{1,13})(?:\.(?P<u>\d{1,6}))?$' # noqa: N806 m = re.match(RE_POSIX, timestr) if not m: msg = "time data '{timestr}' does not match POSIX format." raise ValueError(msg.format(timestr=timestr)) sec = int(m['S']) usec = m['u'] usec = int(usec.ljust(6, '0')) if usec else 0 if sec < 0 < usec: sec -= 1 usec = 1_000_000 - usec ts = cls(1970, 1, 1) + datetime.timedelta(seconds=sec, microseconds=usec) return ts @classmethod def _from_string(cls: Type['Timestamp'], timestr: str) -> 'Timestamp': """Convert a string to a Timestamp object. .. versionadded:: 7.5 """ handlers = [ cls._from_mw, cls._from_iso8601, cls._from_posix, ] for handler in handlers: try: return handler(timestr) except ValueError: continue msg = "time data '{timestr}' does not match any format." raise ValueError(msg.format(timestr=timestr))
[docs] def clone(self) -> 'Timestamp': """Clone this instance.""" return self.replace(microsecond=self.microsecond)
@classproperty def ISO8601Format(cls: Type['Timestamp']) -> str: # noqa: N802 """ISO8601 format string class property for compatibility purpose.""" return cls._ISO8601Format() @classmethod def _ISO8601Format(cls: Type['Timestamp'], # noqa: N802 sep: str = 'T') -> str: """ISO8601 format string. :param sep: one-character separator, placed between the date and time :return: ISO8601 format string """ assert len(sep) == 1 return f'%Y-%m-%d{sep}%H:%M:%SZ'
[docs] @classmethod def fromISOformat(cls: Type['Timestamp'], # noqa: N802 ts: Union[str, 'Timestamp'], sep: str = 'T') -> 'Timestamp': """Convert an ISO 8601 timestamp to a Timestamp object. :param ts: ISO 8601 timestamp or a Timestamp object already :param sep: one-character separator, placed between the date and time :return: Timestamp object """ # If inadvertently passed a Timestamp object, use replace() # to create a clone. if isinstance(ts, cls): return ts.clone() _ts = f'{ts[:10]}{sep}{ts[11:]}' return cls._from_iso8601(_ts)
[docs] @classmethod def fromtimestampformat(cls: Type['Timestamp'], ts: Union[str, 'Timestamp'] ) -> 'Timestamp': """Convert a MediaWiki internal timestamp to a Timestamp object.""" # If inadvertently passed a Timestamp object, use replace() # to create a clone. if isinstance(ts, cls): return ts.clone() if len(ts) == 8: # year, month and day are given only ts += '000000' return cls._from_mw(ts)
[docs] def isoformat(self, sep: str = 'T') -> str: # type: ignore[override] """ Convert object to an ISO 8601 timestamp accepted by MediaWiki. datetime.datetime.isoformat does not postfix the ISO formatted date with a 'Z' unless a timezone is included, which causes MediaWiki ~1.19 and earlier to fail. """ return self.strftime(self._ISO8601Format(sep))
[docs] def totimestampformat(self) -> str: """Convert object to a MediaWiki internal timestamp.""" return self.strftime(self.mediawikiTSFormat)
[docs] def posix_timestamp(self) -> float: """ Convert object to a POSIX timestamp. See Note in datetime.timestamp(). .. versionadded:: 7.5 """ return self.replace(tzinfo=datetime.timezone.utc).timestamp()
[docs] def posix_timestamp_format(self) -> str: """Convert object to a POSIX timestamp format. .. versionadded:: 7.5 """ return f'{self.posix_timestamp():.6f}'
def __str__(self) -> str: """Return a string format recognized by the API.""" return self.isoformat() def __add__(self, other: datetime.timedelta) -> 'Timestamp': """Perform addition, returning a Timestamp instead of datetime.""" newdt = super().__add__(other) if isinstance(newdt, datetime.datetime): return self._from_datetime(newdt) return newdt def __sub__(self, other: datetime.timedelta # type: ignore[override] ) -> 'Timestamp': """Perform subtraction, returning a Timestamp instead of datetime.""" newdt = super().__sub__(other) if isinstance(newdt, datetime.datetime): return self._from_datetime(newdt) return newdt
[docs]class TZoneFixedOffset(datetime.tzinfo): """ Class building tzinfo objects for fixed-offset time zones. :param offset: a number indicating fixed offset in minutes east from UTC :param name: a string with name of the timezone """ def __init__(self, offset: int, name: str) -> None: """Initializer.""" self._offset = datetime.timedelta(minutes=offset) self._name = name
[docs] def utcoffset(self, dt): """Return the offset to UTC.""" return self._offset
[docs] def tzname(self, dt): """Return the name of the timezone.""" return self._name
[docs] def dst(self, dt): """Return no daylight savings time.""" return datetime.timedelta(0)
def __repr__(self) -> str: """Return the internal representation of the timezone.""" return '{}({}, {})'.format( self.__class__.__name__, self._offset.days * 86400 + self._offset.seconds, self._name )
[docs]def str2timedelta(string: str, timestamp=None) -> datetime.timedelta: """ Return a timedelta for a shorthand duration. :param string: a string defining a time period: Examples:: 300s - 300 seconds 36h - 36 hours 7d - 7 days 2w - 2 weeks (14 days) 1y - 1 year :param timestamp: a timestamp to calculate a more accurate duration offset used by years :type timestamp: datetime.datetime :return: the corresponding timedelta object """ key, duration = parse_duration(string) if key == 'y': # years days = math.ceil(duration * 365.25) duration *= 12 if timestamp: return pywikibot.date.apply_month_delta( timestamp.date(), month_delta=duration) - timestamp.date() return datetime.timedelta(days=days) # days, seconds, hours, weeks return datetime.timedelta(**{MW_KEYS[key]: duration})
[docs]def parse_duration(string: str) -> Tuple[str, int]: """ Return the key and duration extracted from the string. :param string: a string defining a time period Examples:: 300s - 300 seconds 36h - 36 hours 7d - 7 days 2w - 2 weeks (14 days) 1y - 1 year :return: key and duration extracted form the string """ if len(string) < 2: raise ValueError('Time period should be a numeric value followed by ' 'its qualifier') key, duration = string[-1], string[:-1] if key not in MW_KEYS: raise ValueError('Time period qualifier is unrecognized: {}' .format(string)) if not duration.isdigit(): raise ValueError("Time period's duration should be numeric: {}" .format(string)) return key, int(duration)