Closes issue #20858: Enhancements/fixes to pure-python datetime module

This patch brings the pure-python datetime more in-line with the C
module.  Patch contributed by Brian Kearns, a PyPy developer.  PyPy
project has been running these modifications in PyPy2 stdlib.

This commit includes:

- General PEP8/cleanups;
- Better testing of argument types passed to constructors;
- Removal of duplicate operations;
- Optimization of timedelta creation;
- Caching the result of __hash__ like the C accelerator;
- Enhancements/bug fixes in tests.
This commit is contained in:
Alexander Belopolsky 2014-09-28 19:11:56 -04:00
parent a2f93885b0
commit 6c7a4182f5
3 changed files with 239 additions and 141 deletions

View File

@ -12,7 +12,7 @@ def _cmp(x, y):
MINYEAR = 1 MINYEAR = 1
MAXYEAR = 9999 MAXYEAR = 9999
_MAXORDINAL = 3652059 # date.max.toordinal() _MAXORDINAL = 3652059 # date.max.toordinal()
# Utility functions, adapted from Python's Demo/classes/Dates.py, which # Utility functions, adapted from Python's Demo/classes/Dates.py, which
# also assumes the current Gregorian calendar indefinitely extended in # also assumes the current Gregorian calendar indefinitely extended in
@ -26,7 +26,7 @@ _MAXORDINAL = 3652059 # date.max.toordinal()
# -1 is a placeholder for indexing purposes. # -1 is a placeholder for indexing purposes.
_DAYS_IN_MONTH = [-1, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31] _DAYS_IN_MONTH = [-1, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31]
_DAYS_BEFORE_MONTH = [-1] # -1 is a placeholder for indexing purposes. _DAYS_BEFORE_MONTH = [-1] # -1 is a placeholder for indexing purposes.
dbm = 0 dbm = 0
for dim in _DAYS_IN_MONTH[1:]: for dim in _DAYS_IN_MONTH[1:]:
_DAYS_BEFORE_MONTH.append(dbm) _DAYS_BEFORE_MONTH.append(dbm)
@ -162,9 +162,9 @@ def _format_time(hh, mm, ss, us):
# Correctly substitute for %z and %Z escapes in strftime formats. # Correctly substitute for %z and %Z escapes in strftime formats.
def _wrap_strftime(object, format, timetuple): def _wrap_strftime(object, format, timetuple):
# Don't call utcoffset() or tzname() unless actually needed. # Don't call utcoffset() or tzname() unless actually needed.
freplace = None # the string to use for %f freplace = None # the string to use for %f
zreplace = None # the string to use for %z zreplace = None # the string to use for %z
Zreplace = None # the string to use for %Z Zreplace = None # the string to use for %Z
# Scan format for %z and %Z escapes, replacing as needed. # Scan format for %z and %Z escapes, replacing as needed.
newformat = [] newformat = []
@ -217,11 +217,6 @@ def _wrap_strftime(object, format, timetuple):
newformat = "".join(newformat) newformat = "".join(newformat)
return _time.strftime(newformat, timetuple) return _time.strftime(newformat, timetuple)
def _call_tzinfo_method(tzinfo, methname, tzinfoarg):
if tzinfo is None:
return None
return getattr(tzinfo, methname)(tzinfoarg)
# Just raise TypeError if the arg isn't None or a string. # Just raise TypeError if the arg isn't None or a string.
def _check_tzname(name): def _check_tzname(name):
if name is not None and not isinstance(name, str): if name is not None and not isinstance(name, str):
@ -245,13 +240,31 @@ def _check_utc_offset(name, offset):
raise ValueError("tzinfo.%s() must return a whole number " raise ValueError("tzinfo.%s() must return a whole number "
"of minutes, got %s" % (name, offset)) "of minutes, got %s" % (name, offset))
if not -timedelta(1) < offset < timedelta(1): if not -timedelta(1) < offset < timedelta(1):
raise ValueError("%s()=%s, must be must be strictly between" raise ValueError("%s()=%s, must be must be strictly between "
" -timedelta(hours=24) and timedelta(hours=24)" "-timedelta(hours=24) and timedelta(hours=24)" %
% (name, offset)) (name, offset))
def _check_int_field(value):
if isinstance(value, int):
return value
if not isinstance(value, float):
try:
value = value.__int__()
except AttributeError:
pass
else:
if isinstance(value, int):
return value
raise TypeError('__int__ returned non-int (type %s)' %
type(value).__name__)
raise TypeError('an integer is required (got type %s)' %
type(value).__name__)
raise TypeError('integer argument expected, got float')
def _check_date_fields(year, month, day): def _check_date_fields(year, month, day):
if not isinstance(year, int): year = _check_int_field(year)
raise TypeError('int expected') month = _check_int_field(month)
day = _check_int_field(day)
if not MINYEAR <= year <= MAXYEAR: if not MINYEAR <= year <= MAXYEAR:
raise ValueError('year must be in %d..%d' % (MINYEAR, MAXYEAR), year) raise ValueError('year must be in %d..%d' % (MINYEAR, MAXYEAR), year)
if not 1 <= month <= 12: if not 1 <= month <= 12:
@ -259,10 +272,13 @@ def _check_date_fields(year, month, day):
dim = _days_in_month(year, month) dim = _days_in_month(year, month)
if not 1 <= day <= dim: if not 1 <= day <= dim:
raise ValueError('day must be in 1..%d' % dim, day) raise ValueError('day must be in 1..%d' % dim, day)
return year, month, day
def _check_time_fields(hour, minute, second, microsecond): def _check_time_fields(hour, minute, second, microsecond):
if not isinstance(hour, int): hour = _check_int_field(hour)
raise TypeError('int expected') minute = _check_int_field(minute)
second = _check_int_field(second)
microsecond = _check_int_field(microsecond)
if not 0 <= hour <= 23: if not 0 <= hour <= 23:
raise ValueError('hour must be in 0..23', hour) raise ValueError('hour must be in 0..23', hour)
if not 0 <= minute <= 59: if not 0 <= minute <= 59:
@ -271,6 +287,7 @@ def _check_time_fields(hour, minute, second, microsecond):
raise ValueError('second must be in 0..59', second) raise ValueError('second must be in 0..59', second)
if not 0 <= microsecond <= 999999: if not 0 <= microsecond <= 999999:
raise ValueError('microsecond must be in 0..999999', microsecond) raise ValueError('microsecond must be in 0..999999', microsecond)
return hour, minute, second, microsecond
def _check_tzinfo_arg(tz): def _check_tzinfo_arg(tz):
if tz is not None and not isinstance(tz, tzinfo): if tz is not None and not isinstance(tz, tzinfo):
@ -297,7 +314,7 @@ class timedelta:
Representation: (days, seconds, microseconds). Why? Because I Representation: (days, seconds, microseconds). Why? Because I
felt like it. felt like it.
""" """
__slots__ = '_days', '_seconds', '_microseconds' __slots__ = '_days', '_seconds', '_microseconds', '_hashcode'
def __new__(cls, days=0, seconds=0, microseconds=0, def __new__(cls, days=0, seconds=0, microseconds=0,
milliseconds=0, minutes=0, hours=0, weeks=0): milliseconds=0, minutes=0, hours=0, weeks=0):
@ -363,38 +380,26 @@ class timedelta:
# secondsfrac isn't referenced again # secondsfrac isn't referenced again
if isinstance(microseconds, float): if isinstance(microseconds, float):
microseconds += usdouble microseconds = round(microseconds + usdouble)
microseconds = round(microseconds, 0)
seconds, microseconds = divmod(microseconds, 1e6)
assert microseconds == int(microseconds)
assert seconds == int(seconds)
days, seconds = divmod(seconds, 24.*3600.)
assert days == int(days)
assert seconds == int(seconds)
d += int(days)
s += int(seconds) # can't overflow
assert isinstance(s, int)
assert abs(s) <= 3 * 24 * 3600
else:
seconds, microseconds = divmod(microseconds, 1000000) seconds, microseconds = divmod(microseconds, 1000000)
days, seconds = divmod(seconds, 24*3600) days, seconds = divmod(seconds, 24*3600)
d += days d += days
s += int(seconds) # can't overflow s += seconds
assert isinstance(s, int) else:
assert abs(s) <= 3 * 24 * 3600 microseconds = int(microseconds)
microseconds = float(microseconds) seconds, microseconds = divmod(microseconds, 1000000)
microseconds += usdouble days, seconds = divmod(seconds, 24*3600)
microseconds = round(microseconds, 0) d += days
s += seconds
microseconds = round(microseconds + usdouble)
assert isinstance(s, int)
assert isinstance(microseconds, int)
assert abs(s) <= 3 * 24 * 3600 assert abs(s) <= 3 * 24 * 3600
assert abs(microseconds) < 3.1e6 assert abs(microseconds) < 3.1e6
# Just a little bit of carrying possible for microseconds and seconds. # Just a little bit of carrying possible for microseconds and seconds.
assert isinstance(microseconds, float) seconds, us = divmod(microseconds, 1000000)
assert int(microseconds) == microseconds s += seconds
us = int(microseconds)
seconds, us = divmod(us, 1000000)
s += seconds # cant't overflow
assert isinstance(s, int)
days, s = divmod(s, 24*3600) days, s = divmod(s, 24*3600)
d += days d += days
@ -402,14 +407,14 @@ class timedelta:
assert isinstance(s, int) and 0 <= s < 24*3600 assert isinstance(s, int) and 0 <= s < 24*3600
assert isinstance(us, int) and 0 <= us < 1000000 assert isinstance(us, int) and 0 <= us < 1000000
self = object.__new__(cls)
self._days = d
self._seconds = s
self._microseconds = us
if abs(d) > 999999999: if abs(d) > 999999999:
raise OverflowError("timedelta # of days is too large: %d" % d) raise OverflowError("timedelta # of days is too large: %d" % d)
self = object.__new__(cls)
self._days = d
self._seconds = s
self._microseconds = us
self._hashcode = -1
return self return self
def __repr__(self): def __repr__(self):
@ -442,7 +447,7 @@ class timedelta:
def total_seconds(self): def total_seconds(self):
"""Total seconds in the duration.""" """Total seconds in the duration."""
return ((self.days * 86400 + self.seconds)*10**6 + return ((self.days * 86400 + self.seconds) * 10**6 +
self.microseconds) / 10**6 self.microseconds) / 10**6
# Read-only field accessors # Read-only field accessors
@ -597,7 +602,9 @@ class timedelta:
return _cmp(self._getstate(), other._getstate()) return _cmp(self._getstate(), other._getstate())
def __hash__(self): def __hash__(self):
return hash(self._getstate()) if self._hashcode == -1:
self._hashcode = hash(self._getstate())
return self._hashcode
def __bool__(self): def __bool__(self):
return (self._days != 0 or return (self._days != 0 or
@ -645,7 +652,7 @@ class date:
Properties (readonly): Properties (readonly):
year, month, day year, month, day
""" """
__slots__ = '_year', '_month', '_day' __slots__ = '_year', '_month', '_day', '_hashcode'
def __new__(cls, year, month=None, day=None): def __new__(cls, year, month=None, day=None):
"""Constructor. """Constructor.
@ -654,17 +661,19 @@ class date:
year, month, day (required, base 1) year, month, day (required, base 1)
""" """
if (isinstance(year, bytes) and len(year) == 4 and if month is None and isinstance(year, bytes) and len(year) == 4 and \
1 <= year[2] <= 12 and month is None): # Month is sane 1 <= year[2] <= 12:
# Pickle support # Pickle support
self = object.__new__(cls) self = object.__new__(cls)
self.__setstate(year) self.__setstate(year)
self._hashcode = -1
return self return self
_check_date_fields(year, month, day) year, month, day = _check_date_fields(year, month, day)
self = object.__new__(cls) self = object.__new__(cls)
self._year = year self._year = year
self._month = month self._month = month
self._day = day self._day = day
self._hashcode = -1
return self return self
# Additional constructors # Additional constructors
@ -728,6 +737,8 @@ class date:
return _wrap_strftime(self, fmt, self.timetuple()) return _wrap_strftime(self, fmt, self.timetuple())
def __format__(self, fmt): def __format__(self, fmt):
if not isinstance(fmt, str):
raise TypeError("must be str, not %s" % type(fmt).__name__)
if len(fmt) != 0: if len(fmt) != 0:
return self.strftime(fmt) return self.strftime(fmt)
return str(self) return str(self)
@ -784,7 +795,6 @@ class date:
month = self._month month = self._month
if day is None: if day is None:
day = self._day day = self._day
_check_date_fields(year, month, day)
return date(year, month, day) return date(year, month, day)
# Comparisons of date objects with other. # Comparisons of date objects with other.
@ -827,7 +837,9 @@ class date:
def __hash__(self): def __hash__(self):
"Hash." "Hash."
return hash(self._getstate()) if self._hashcode == -1:
self._hashcode = hash(self._getstate())
return self._hashcode
# Computations # Computations
@ -897,8 +909,6 @@ class date:
return bytes([yhi, ylo, self._month, self._day]), return bytes([yhi, ylo, self._month, self._day]),
def __setstate(self, string): def __setstate(self, string):
if len(string) != 4 or not (1 <= string[2] <= 12):
raise TypeError("not enough arguments")
yhi, ylo, self._month, self._day = string yhi, ylo, self._month, self._day = string
self._year = yhi * 256 + ylo self._year = yhi * 256 + ylo
@ -917,6 +927,7 @@ class tzinfo:
Subclasses must override the name(), utcoffset() and dst() methods. Subclasses must override the name(), utcoffset() and dst() methods.
""" """
__slots__ = () __slots__ = ()
def tzname(self, dt): def tzname(self, dt):
"datetime -> string name of time zone." "datetime -> string name of time zone."
raise NotImplementedError("tzinfo subclass must override tzname()") raise NotImplementedError("tzinfo subclass must override tzname()")
@ -1003,6 +1014,7 @@ class time:
Properties (readonly): Properties (readonly):
hour, minute, second, microsecond, tzinfo hour, minute, second, microsecond, tzinfo
""" """
__slots__ = '_hour', '_minute', '_second', '_microsecond', '_tzinfo', '_hashcode'
def __new__(cls, hour=0, minute=0, second=0, microsecond=0, tzinfo=None): def __new__(cls, hour=0, minute=0, second=0, microsecond=0, tzinfo=None):
"""Constructor. """Constructor.
@ -1013,18 +1025,22 @@ class time:
second, microsecond (default to zero) second, microsecond (default to zero)
tzinfo (default to None) tzinfo (default to None)
""" """
self = object.__new__(cls) if isinstance(hour, bytes) and len(hour) == 6 and hour[0] < 24:
if isinstance(hour, bytes) and len(hour) == 6:
# Pickle support # Pickle support
self = object.__new__(cls)
self.__setstate(hour, minute or None) self.__setstate(hour, minute or None)
self._hashcode = -1
return self return self
hour, minute, second, microsecond = _check_time_fields(
hour, minute, second, microsecond)
_check_tzinfo_arg(tzinfo) _check_tzinfo_arg(tzinfo)
_check_time_fields(hour, minute, second, microsecond) self = object.__new__(cls)
self._hour = hour self._hour = hour
self._minute = minute self._minute = minute
self._second = second self._second = second
self._microsecond = microsecond self._microsecond = microsecond
self._tzinfo = tzinfo self._tzinfo = tzinfo
self._hashcode = -1
return self return self
# Read-only field accessors # Read-only field accessors
@ -1109,8 +1125,8 @@ class time:
if base_compare: if base_compare:
return _cmp((self._hour, self._minute, self._second, return _cmp((self._hour, self._minute, self._second,
self._microsecond), self._microsecond),
(other._hour, other._minute, other._second, (other._hour, other._minute, other._second,
other._microsecond)) other._microsecond))
if myoff is None or otoff is None: if myoff is None or otoff is None:
if allow_mixed: if allow_mixed:
return 2 # arbitrary non-zero value return 2 # arbitrary non-zero value
@ -1123,16 +1139,20 @@ class time:
def __hash__(self): def __hash__(self):
"""Hash.""" """Hash."""
tzoff = self.utcoffset() if self._hashcode == -1:
if not tzoff: # zero or None tzoff = self.utcoffset()
return hash(self._getstate()[0]) if not tzoff: # zero or None
h, m = divmod(timedelta(hours=self.hour, minutes=self.minute) - tzoff, self._hashcode = hash(self._getstate()[0])
timedelta(hours=1)) else:
assert not m % timedelta(minutes=1), "whole minute" h, m = divmod(timedelta(hours=self.hour, minutes=self.minute) - tzoff,
m //= timedelta(minutes=1) timedelta(hours=1))
if 0 <= h < 24: assert not m % timedelta(minutes=1), "whole minute"
return hash(time(h, m, self.second, self.microsecond)) m //= timedelta(minutes=1)
return hash((h, m, self.second, self.microsecond)) if 0 <= h < 24:
self._hashcode = hash(time(h, m, self.second, self.microsecond))
else:
self._hashcode = hash((h, m, self.second, self.microsecond))
return self._hashcode
# Conversion to string # Conversion to string
@ -1195,6 +1215,8 @@ class time:
return _wrap_strftime(self, fmt, timetuple) return _wrap_strftime(self, fmt, timetuple)
def __format__(self, fmt): def __format__(self, fmt):
if not isinstance(fmt, str):
raise TypeError("must be str, not %s" % type(fmt).__name__)
if len(fmt) != 0: if len(fmt) != 0:
return self.strftime(fmt) return self.strftime(fmt)
return str(self) return str(self)
@ -1251,8 +1273,6 @@ class time:
microsecond = self.microsecond microsecond = self.microsecond
if tzinfo is True: if tzinfo is True:
tzinfo = self.tzinfo tzinfo = self.tzinfo
_check_time_fields(hour, minute, second, microsecond)
_check_tzinfo_arg(tzinfo)
return time(hour, minute, second, microsecond, tzinfo) return time(hour, minute, second, microsecond, tzinfo)
# Pickle support. # Pickle support.
@ -1268,15 +1288,11 @@ class time:
return (basestate, self._tzinfo) return (basestate, self._tzinfo)
def __setstate(self, string, tzinfo): def __setstate(self, string, tzinfo):
if len(string) != 6 or string[0] >= 24: if tzinfo is not None and not isinstance(tzinfo, _tzinfo_class):
raise TypeError("an integer is required") raise TypeError("bad tzinfo state arg")
(self._hour, self._minute, self._second, self._hour, self._minute, self._second, us1, us2, us3 = string
us1, us2, us3) = string
self._microsecond = (((us1 << 8) | us2) << 8) | us3 self._microsecond = (((us1 << 8) | us2) << 8) | us3
if tzinfo is None or isinstance(tzinfo, _tzinfo_class): self._tzinfo = tzinfo
self._tzinfo = tzinfo
else:
raise TypeError("bad tzinfo state arg %r" % tzinfo)
def __reduce__(self): def __reduce__(self):
return (time, self._getstate()) return (time, self._getstate())
@ -1293,25 +1309,30 @@ class datetime(date):
The year, month and day arguments are required. tzinfo may be None, or an The year, month and day arguments are required. tzinfo may be None, or an
instance of a tzinfo subclass. The remaining arguments may be ints. instance of a tzinfo subclass. The remaining arguments may be ints.
""" """
__slots__ = date.__slots__ + time.__slots__
__slots__ = date.__slots__ + (
'_hour', '_minute', '_second',
'_microsecond', '_tzinfo')
def __new__(cls, year, month=None, day=None, hour=0, minute=0, second=0, def __new__(cls, year, month=None, day=None, hour=0, minute=0, second=0,
microsecond=0, tzinfo=None): microsecond=0, tzinfo=None):
if isinstance(year, bytes) and len(year) == 10: if isinstance(year, bytes) and len(year) == 10 and 1 <= year[2] <= 12:
# Pickle support # Pickle support
self = date.__new__(cls, year[:4]) self = object.__new__(cls)
self.__setstate(year, month) self.__setstate(year, month)
self._hashcode = -1
return self return self
year, month, day = _check_date_fields(year, month, day)
hour, minute, second, microsecond = _check_time_fields(
hour, minute, second, microsecond)
_check_tzinfo_arg(tzinfo) _check_tzinfo_arg(tzinfo)
_check_time_fields(hour, minute, second, microsecond) self = object.__new__(cls)
self = date.__new__(cls, year, month, day) self._year = year
self._month = month
self._day = day
self._hour = hour self._hour = hour
self._minute = minute self._minute = minute
self._second = second self._second = second
self._microsecond = microsecond self._microsecond = microsecond
self._tzinfo = tzinfo self._tzinfo = tzinfo
self._hashcode = -1
return self return self
# Read-only field accessors # Read-only field accessors
@ -1346,7 +1367,6 @@ class datetime(date):
A timezone info object may be passed in as well. A timezone info object may be passed in as well.
""" """
_check_tzinfo_arg(tz) _check_tzinfo_arg(tz)
converter = _time.localtime if tz is None else _time.gmtime converter = _time.localtime if tz is None else _time.gmtime
@ -1385,11 +1405,6 @@ class datetime(date):
ss = min(ss, 59) # clamp out leap seconds if the platform has them ss = min(ss, 59) # clamp out leap seconds if the platform has them
return cls(y, m, d, hh, mm, ss, us) return cls(y, m, d, hh, mm, ss, us)
# XXX This is supposed to do better than we *can* do by using time.time(),
# XXX if the platform supports a more accurate way. The C implementation
# XXX uses gettimeofday on platforms that have it, but that isn't
# XXX available from Python. So now() may return different results
# XXX across the implementations.
@classmethod @classmethod
def now(cls, tz=None): def now(cls, tz=None):
"Construct a datetime from time.time() and optional time zone info." "Construct a datetime from time.time() and optional time zone info."
@ -1476,11 +1491,8 @@ class datetime(date):
microsecond = self.microsecond microsecond = self.microsecond
if tzinfo is True: if tzinfo is True:
tzinfo = self.tzinfo tzinfo = self.tzinfo
_check_date_fields(year, month, day) return datetime(year, month, day, hour, minute, second, microsecond,
_check_time_fields(hour, minute, second, microsecond) tzinfo)
_check_tzinfo_arg(tzinfo)
return datetime(year, month, day, hour, minute, second,
microsecond, tzinfo)
def astimezone(self, tz=None): def astimezone(self, tz=None):
if tz is None: if tz is None:
@ -1550,10 +1562,9 @@ class datetime(date):
Optional argument sep specifies the separator between date and Optional argument sep specifies the separator between date and
time, default 'T'. time, default 'T'.
""" """
s = ("%04d-%02d-%02d%c" % (self._year, self._month, self._day, s = ("%04d-%02d-%02d%c" % (self._year, self._month, self._day, sep) +
sep) + _format_time(self._hour, self._minute, self._second,
_format_time(self._hour, self._minute, self._second, self._microsecond))
self._microsecond))
off = self.utcoffset() off = self.utcoffset()
if off is not None: if off is not None:
if off.days < 0: if off.days < 0:
@ -1569,7 +1580,7 @@ class datetime(date):
def __repr__(self): def __repr__(self):
"""Convert to formal string, for repr().""" """Convert to formal string, for repr()."""
L = [self._year, self._month, self._day, # These are never zero L = [self._year, self._month, self._day, # These are never zero
self._hour, self._minute, self._second, self._microsecond] self._hour, self._minute, self._second, self._microsecond]
if L[-1] == 0: if L[-1] == 0:
del L[-1] del L[-1]
@ -1609,7 +1620,9 @@ class datetime(date):
it mean anything in particular. For example, "GMT", "UTC", "-500", it mean anything in particular. For example, "GMT", "UTC", "-500",
"-5:00", "EDT", "US/Eastern", "America/New York" are all valid replies. "-5:00", "EDT", "US/Eastern", "America/New York" are all valid replies.
""" """
name = _call_tzinfo_method(self._tzinfo, "tzname", self) if self._tzinfo is None:
return None
name = self._tzinfo.tzname(self)
_check_tzname(name) _check_tzname(name)
return name return name
@ -1695,9 +1708,9 @@ class datetime(date):
return _cmp((self._year, self._month, self._day, return _cmp((self._year, self._month, self._day,
self._hour, self._minute, self._second, self._hour, self._minute, self._second,
self._microsecond), self._microsecond),
(other._year, other._month, other._day, (other._year, other._month, other._day,
other._hour, other._minute, other._second, other._hour, other._minute, other._second,
other._microsecond)) other._microsecond))
if myoff is None or otoff is None: if myoff is None or otoff is None:
if allow_mixed: if allow_mixed:
return 2 # arbitrary non-zero value return 2 # arbitrary non-zero value
@ -1755,12 +1768,15 @@ class datetime(date):
return base + otoff - myoff return base + otoff - myoff
def __hash__(self): def __hash__(self):
tzoff = self.utcoffset() if self._hashcode == -1:
if tzoff is None: tzoff = self.utcoffset()
return hash(self._getstate()[0]) if tzoff is None:
days = _ymd2ord(self.year, self.month, self.day) self._hashcode = hash(self._getstate()[0])
seconds = self.hour * 3600 + self.minute * 60 + self.second else:
return hash(timedelta(days, seconds, self.microsecond) - tzoff) days = _ymd2ord(self.year, self.month, self.day)
seconds = self.hour * 3600 + self.minute * 60 + self.second
self._hashcode = hash(timedelta(days, seconds, self.microsecond) - tzoff)
return self._hashcode
# Pickle support. # Pickle support.
@ -1777,14 +1793,13 @@ class datetime(date):
return (basestate, self._tzinfo) return (basestate, self._tzinfo)
def __setstate(self, string, tzinfo): def __setstate(self, string, tzinfo):
if tzinfo is not None and not isinstance(tzinfo, _tzinfo_class):
raise TypeError("bad tzinfo state arg")
(yhi, ylo, self._month, self._day, self._hour, (yhi, ylo, self._month, self._day, self._hour,
self._minute, self._second, us1, us2, us3) = string self._minute, self._second, us1, us2, us3) = string
self._year = yhi * 256 + ylo self._year = yhi * 256 + ylo
self._microsecond = (((us1 << 8) | us2) << 8) | us3 self._microsecond = (((us1 << 8) | us2) << 8) | us3
if tzinfo is None or isinstance(tzinfo, _tzinfo_class): self._tzinfo = tzinfo
self._tzinfo = tzinfo
else:
raise TypeError("bad tzinfo state arg %r" % tzinfo)
def __reduce__(self): def __reduce__(self):
return (self.__class__, self._getstate()) return (self.__class__, self._getstate())
@ -1800,7 +1815,7 @@ def _isoweek1monday(year):
# XXX This could be done more efficiently # XXX This could be done more efficiently
THURSDAY = 3 THURSDAY = 3
firstday = _ymd2ord(year, 1, 1) firstday = _ymd2ord(year, 1, 1)
firstweekday = (firstday + 6) % 7 # See weekday() above firstweekday = (firstday + 6) % 7 # See weekday() above
week1monday = firstday - firstweekday week1monday = firstday - firstweekday
if firstweekday > THURSDAY: if firstweekday > THURSDAY:
week1monday += 7 week1monday += 7
@ -1821,13 +1836,12 @@ class timezone(tzinfo):
elif not isinstance(name, str): elif not isinstance(name, str):
raise TypeError("name must be a string") raise TypeError("name must be a string")
if not cls._minoffset <= offset <= cls._maxoffset: if not cls._minoffset <= offset <= cls._maxoffset:
raise ValueError("offset must be a timedelta" raise ValueError("offset must be a timedelta "
" strictly between -timedelta(hours=24) and" "strictly between -timedelta(hours=24) and "
" timedelta(hours=24).") "timedelta(hours=24).")
if (offset.microseconds != 0 or if (offset.microseconds != 0 or offset.seconds % 60 != 0):
offset.seconds % 60 != 0): raise ValueError("offset must be a timedelta "
raise ValueError("offset must be a timedelta" "representing a whole number of minutes")
" representing a whole number of minutes")
return cls._create(offset, name) return cls._create(offset, name)
@classmethod @classmethod
@ -2124,14 +2138,13 @@ except ImportError:
pass pass
else: else:
# Clean up unused names # Clean up unused names
del (_DAYNAMES, _DAYS_BEFORE_MONTH, _DAYS_IN_MONTH, del (_DAYNAMES, _DAYS_BEFORE_MONTH, _DAYS_IN_MONTH, _DI100Y, _DI400Y,
_DI100Y, _DI400Y, _DI4Y, _MAXORDINAL, _MONTHNAMES, _DI4Y, _EPOCH, _MAXORDINAL, _MONTHNAMES, _build_struct_time,
_build_struct_time, _call_tzinfo_method, _check_date_fields, _check_date_fields, _check_int_field, _check_time_fields,
_check_time_fields, _check_tzinfo_arg, _check_tzname, _check_tzinfo_arg, _check_tzname, _check_utc_offset, _cmp, _cmperror,
_check_utc_offset, _cmp, _cmperror, _date_class, _days_before_month, _date_class, _days_before_month, _days_before_year, _days_in_month,
_days_before_year, _days_in_month, _format_time, _is_leap, _format_time, _is_leap, _isoweek1monday, _math, _ord2ymd,
_isoweek1monday, _math, _ord2ymd, _time, _time_class, _tzinfo_class, _time, _time_class, _tzinfo_class, _wrap_strftime, _ymd2ord)
_wrap_strftime, _ymd2ord)
# XXX Since import * above excludes names that start with _, # XXX Since import * above excludes names that start with _,
# docstring does not get overwritten. In the future, it may be # docstring does not get overwritten. In the future, it may be
# appropriate to maintain a single module level docstring and # appropriate to maintain a single module level docstring and

View File

@ -50,6 +50,17 @@ class TestModule(unittest.TestCase):
self.assertEqual(datetime.MINYEAR, 1) self.assertEqual(datetime.MINYEAR, 1)
self.assertEqual(datetime.MAXYEAR, 9999) self.assertEqual(datetime.MAXYEAR, 9999)
def test_name_cleanup(self):
if '_Fast' not in str(self):
return
datetime = datetime_module
names = set(name for name in dir(datetime)
if not name.startswith('__') and not name.endswith('__'))
allowed = set(['MAXYEAR', 'MINYEAR', 'date', 'datetime',
'datetime_CAPI', 'time', 'timedelta', 'timezone',
'tzinfo'])
self.assertEqual(names - allowed, set([]))
############################################################################# #############################################################################
# tzinfo tests # tzinfo tests
@ -616,8 +627,12 @@ class TestTimeDelta(HarmlessMixedComparison, unittest.TestCase):
# Single-field rounding. # Single-field rounding.
eq(td(milliseconds=0.4/1000), td(0)) # rounds to 0 eq(td(milliseconds=0.4/1000), td(0)) # rounds to 0
eq(td(milliseconds=-0.4/1000), td(0)) # rounds to 0 eq(td(milliseconds=-0.4/1000), td(0)) # rounds to 0
eq(td(milliseconds=0.5/1000), td(microseconds=0))
eq(td(milliseconds=-0.5/1000), td(microseconds=0))
eq(td(milliseconds=0.6/1000), td(microseconds=1)) eq(td(milliseconds=0.6/1000), td(microseconds=1))
eq(td(milliseconds=-0.6/1000), td(microseconds=-1)) eq(td(milliseconds=-0.6/1000), td(microseconds=-1))
eq(td(seconds=0.5/10**6), td(microseconds=0))
eq(td(seconds=-0.5/10**6), td(microseconds=0))
# Rounding due to contributions from more than one field. # Rounding due to contributions from more than one field.
us_per_hour = 3600e6 us_per_hour = 3600e6
@ -1131,11 +1146,13 @@ class TestDate(HarmlessMixedComparison, unittest.TestCase):
#check that this standard extension works #check that this standard extension works
t.strftime("%f") t.strftime("%f")
def test_format(self): def test_format(self):
dt = self.theclass(2007, 9, 10) dt = self.theclass(2007, 9, 10)
self.assertEqual(dt.__format__(''), str(dt)) self.assertEqual(dt.__format__(''), str(dt))
with self.assertRaisesRegex(TypeError, '^must be str, not int$'):
dt.__format__(123)
# check that a derived class's __str__() gets called # check that a derived class's __str__() gets called
class A(self.theclass): class A(self.theclass):
def __str__(self): def __str__(self):
@ -1391,9 +1408,10 @@ class TestDate(HarmlessMixedComparison, unittest.TestCase):
for month_byte in b'9', b'\0', b'\r', b'\xff': for month_byte in b'9', b'\0', b'\r', b'\xff':
self.assertRaises(TypeError, self.theclass, self.assertRaises(TypeError, self.theclass,
base[:2] + month_byte + base[3:]) base[:2] + month_byte + base[3:])
# Good bytes, but bad tzinfo: if issubclass(self.theclass, datetime):
self.assertRaises(TypeError, self.theclass, # Good bytes, but bad tzinfo:
bytes([1] * len(base)), 'EST') with self.assertRaisesRegex(TypeError, '^bad tzinfo state arg$'):
self.theclass(bytes([1] * len(base)), 'EST')
for ord_byte in range(1, 13): for ord_byte in range(1, 13):
# This shouldn't blow up because of the month byte alone. If # This shouldn't blow up because of the month byte alone. If
@ -1469,6 +1487,9 @@ class TestDateTime(TestDate):
dt = self.theclass(2007, 9, 10, 4, 5, 1, 123) dt = self.theclass(2007, 9, 10, 4, 5, 1, 123)
self.assertEqual(dt.__format__(''), str(dt)) self.assertEqual(dt.__format__(''), str(dt))
with self.assertRaisesRegex(TypeError, '^must be str, not int$'):
dt.__format__(123)
# check that a derived class's __str__() gets called # check that a derived class's __str__() gets called
class A(self.theclass): class A(self.theclass):
def __str__(self): def __str__(self):
@ -1789,6 +1810,7 @@ class TestDateTime(TestDate):
tzinfo=timezone(timedelta(hours=-5), 'EST')) tzinfo=timezone(timedelta(hours=-5), 'EST'))
self.assertEqual(t.timestamp(), self.assertEqual(t.timestamp(),
18000 + 3600 + 2*60 + 3 + 4*1e-6) 18000 + 3600 + 2*60 + 3 + 4*1e-6)
def test_microsecond_rounding(self): def test_microsecond_rounding(self):
for fts in [self.theclass.fromtimestamp, for fts in [self.theclass.fromtimestamp,
self.theclass.utcfromtimestamp]: self.theclass.utcfromtimestamp]:
@ -1839,6 +1861,7 @@ class TestDateTime(TestDate):
for insane in -1e200, 1e200: for insane in -1e200, 1e200:
self.assertRaises(OverflowError, self.theclass.utcfromtimestamp, self.assertRaises(OverflowError, self.theclass.utcfromtimestamp,
insane) insane)
@unittest.skipIf(sys.platform == "win32", "Windows doesn't accept negative timestamps") @unittest.skipIf(sys.platform == "win32", "Windows doesn't accept negative timestamps")
def test_negative_float_fromtimestamp(self): def test_negative_float_fromtimestamp(self):
# The result is tz-dependent; at least test that this doesn't # The result is tz-dependent; at least test that this doesn't
@ -2218,6 +2241,9 @@ class TestTime(HarmlessMixedComparison, unittest.TestCase):
t = self.theclass(1, 2, 3, 4) t = self.theclass(1, 2, 3, 4)
self.assertEqual(t.__format__(''), str(t)) self.assertEqual(t.__format__(''), str(t))
with self.assertRaisesRegex(TypeError, '^must be str, not int$'):
t.__format__(123)
# check that a derived class's __str__() gets called # check that a derived class's __str__() gets called
class A(self.theclass): class A(self.theclass):
def __str__(self): def __str__(self):
@ -2347,6 +2373,9 @@ class TestTime(HarmlessMixedComparison, unittest.TestCase):
for hour_byte in ' ', '9', chr(24), '\xff': for hour_byte in ' ', '9', chr(24), '\xff':
self.assertRaises(TypeError, self.theclass, self.assertRaises(TypeError, self.theclass,
hour_byte + base[1:]) hour_byte + base[1:])
# Good bytes, but bad tzinfo:
with self.assertRaisesRegex(TypeError, '^bad tzinfo state arg$'):
self.theclass(bytes([1] * len(base)), 'EST')
# A mixin for classes with a tzinfo= argument. Subclasses must define # A mixin for classes with a tzinfo= argument. Subclasses must define
# theclass as a class atribute, and theclass(1, 1, 1, tzinfo=whatever) # theclass as a class atribute, and theclass(1, 1, 1, tzinfo=whatever)
@ -2606,7 +2635,7 @@ class TestTimeTZ(TestTime, TZInfoBase, unittest.TestCase):
self.assertRaises(TypeError, t.strftime, "%Z") self.assertRaises(TypeError, t.strftime, "%Z")
# Issue #6697: # Issue #6697:
if '_Fast' in str(type(self)): if '_Fast' in str(self):
Badtzname.tz = '\ud800' Badtzname.tz = '\ud800'
self.assertRaises(ValueError, t.strftime, "%Z") self.assertRaises(ValueError, t.strftime, "%Z")
@ -3768,6 +3797,61 @@ class Oddballs(unittest.TestCase):
self.assertEqual(as_datetime, datetime_sc) self.assertEqual(as_datetime, datetime_sc)
self.assertEqual(datetime_sc, as_datetime) self.assertEqual(datetime_sc, as_datetime)
def test_extra_attributes(self):
for x in [date.today(),
time(),
datetime.utcnow(),
timedelta(),
tzinfo(),
timezone(timedelta())]:
with self.assertRaises(AttributeError):
x.abc = 1
def test_check_arg_types(self):
import decimal
class Number:
def __init__(self, value):
self.value = value
def __int__(self):
return self.value
for xx in [decimal.Decimal(10),
decimal.Decimal('10.9'),
Number(10)]:
self.assertEqual(datetime(10, 10, 10, 10, 10, 10, 10),
datetime(xx, xx, xx, xx, xx, xx, xx))
with self.assertRaisesRegex(TypeError, '^an integer is required '
'\(got type str\)$'):
datetime(10, 10, '10')
f10 = Number(10.9)
with self.assertRaisesRegex(TypeError, '^__int__ returned non-int '
'\(type float\)$'):
datetime(10, 10, f10)
class Float(float):
pass
s10 = Float(10.9)
with self.assertRaisesRegex(TypeError, '^integer argument expected, '
'got float$'):
datetime(10, 10, s10)
with self.assertRaises(TypeError):
datetime(10., 10, 10)
with self.assertRaises(TypeError):
datetime(10, 10., 10)
with self.assertRaises(TypeError):
datetime(10, 10, 10.)
with self.assertRaises(TypeError):
datetime(10, 10, 10, 10.)
with self.assertRaises(TypeError):
datetime(10, 10, 10, 10, 10.)
with self.assertRaises(TypeError):
datetime(10, 10, 10, 10, 10, 10.)
with self.assertRaises(TypeError):
datetime(10, 10, 10, 10, 10, 10, 10.)
def test_main(): def test_main():
support.run_unittest(__name__) support.run_unittest(__name__)

View File

@ -689,6 +689,7 @@ Per Øyvind Karlsen
Anton Kasyanov Anton Kasyanov
Lou Kates Lou Kates
Hiroaki Kawai Hiroaki Kawai
Brian Kearns
Sebastien Keim Sebastien Keim
Ryan Kelly Ryan Kelly
Dan Kenigsberg Dan Kenigsberg