Skip to content
Open
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
163 commits
Select commit Hold shift + click to select a range
bcc712e
Nanosecond support for datetime
SmartManoj Apr 30, 2022
db8a9a5
📜🤖 Added by blurb_it.
blurb-it[bot] Apr 30, 2022
b559573
typo
SmartManoj Apr 30, 2022
e0e0ee9
removed Decimal
SmartManoj May 6, 2022
c0af7e9
updated probe1
SmartManoj May 6, 2022
1678928
updated utcnow
SmartManoj May 6, 2022
7f6b8b8
Merge branch 'main' into patch-4
SmartManoj May 6, 2022
8492efa
Merge branch 'main' into patch-4
SmartManoj May 20, 2022
37e77d8
nit
SmartManoj Jan 28, 2025
1b500e6
don't divide
SmartManoj Jan 28, 2025
049a7dd
Merge branch 'main' into pr/92078
SmartManoj Jan 28, 2025
3de498e
add nanosecond
SmartManoj Jan 28, 2025
b0c9552
update docstring
SmartManoj Jan 28, 2025
69bfcd8
truncate decimal
SmartManoj Jan 28, 2025
ef62206
rename to nanosecond
SmartManoj Jan 29, 2025
78d4188
parse empty string
SmartManoj Jan 29, 2025
b56c892
Merge branch 'main' into patch-4
SmartManoj Apr 17, 2025
e7305a4
fix kwarg
SmartManoj Apr 17, 2025
9d2f818
update logic
SmartManoj Apr 17, 2025
0986e7b
lint
SmartManoj Apr 17, 2025
e437eaf
Merge branch 'main' into pr/92078
SmartManoj Apr 17, 2025
be25162
Merge branch 'main' into patch-4
SmartManoj Apr 23, 2025
38c00d6
cleanup
SmartManoj Apr 24, 2025
e23588b
revert changes in __reduce__
SmartManoj Apr 24, 2025
bf71681
lint
SmartManoj Apr 24, 2025
dac7876
update time class
SmartManoj Apr 24, 2025
2b78414
update whatsnew
SmartManoj Apr 24, 2025
efb694f
Change name to English
SmartManoj Apr 24, 2025
e4ae66d
Remove Emoji
SmartManoj Apr 24, 2025
f5c118d
Merge branch 'main' into pr/92078
SmartManoj Apr 24, 2025
0b7a81b
update news
SmartManoj Apr 24, 2025
8189e88
update time_isoformat
SmartManoj Apr 24, 2025
10b69f7
update check_time_args
SmartManoj Apr 24, 2025
b0ebfc5
add nanosecond arg
SmartManoj Apr 24, 2025
adb92a6
Merge branch 'main' into pr/92078
SmartManoj Apr 25, 2025
54127a2
update timedelta
SmartManoj Apr 25, 2025
633ff23
add complimentary changes
SmartManoj Apr 25, 2025
6687799
Merge branch 'main' into pr/92078
SmartManoj Apr 25, 2025
4ca6185
add ns
SmartManoj Apr 25, 2025
0c4ff5a
Merge branch 'main' into pr/92078
SmartManoj Apr 25, 2025
f100451
lint
SmartManoj Apr 25, 2025
4f5f428
add macro
SmartManoj Apr 25, 2025
6a43121
Merge branch 'main' into pr/92078
SmartManoj Apr 25, 2025
9c23406
update doc
SmartManoj Apr 25, 2025
b125d71
declare delta_ns
SmartManoj Apr 25, 2025
911ddb0
fix constructors
SmartManoj Apr 25, 2025
c775e96
Merge branch 'main' into pr/92078
SmartManoj Apr 25, 2025
3590c7c
fix signature
SmartManoj Apr 26, 2025
c323ed5
Merge branch 'main' into pr/92078
SmartManoj Apr 26, 2025
d577bc3
fix clinic input
SmartManoj Apr 26, 2025
29c8112
fix get_datetime_fromdateandtime
SmartManoj Apr 26, 2025
13dcc7b
fix macro
SmartManoj Apr 26, 2025
26c8800
fix get_time_fromtime
SmartManoj Apr 26, 2025
b30edbc
force regenerate clinic output
SmartManoj Apr 26, 2025
aed1afa
add append_keyword_nanosecond
SmartManoj Apr 27, 2025
75553a2
fix tests
SmartManoj Apr 26, 2025
dad595d
Merge branch 'main' into pr/92078
SmartManoj Apr 27, 2025
6f4db2b
Merge branch 'main' into pr/92078
SmartManoj Apr 28, 2025
ebaf468
remove test_backdoor_resistance
SmartManoj Apr 28, 2025
00e3b45
minor fixes
SmartManoj Apr 28, 2025
f9d468c
update tests
SmartManoj Apr 28, 2025
4959a6a
Merge branch 'main' into pr/92078
SmartManoj Apr 28, 2025
eabd9a5
fix test
SmartManoj Apr 28, 2025
7d8183c
Merge branch 'main' into pr/92078
SmartManoj Apr 28, 2025
70320ca
bug fix
SmartManoj Apr 29, 2025
10ac2d6
temp
SmartManoj Apr 29, 2025
bda6c29
Merge branch 'main' into pr/92078
SmartManoj Apr 29, 2025
0bcf405
bug fix
SmartManoj Apr 30, 2025
17f2943
update test
SmartManoj Apr 30, 2025
2173886
temp
SmartManoj Apr 30, 2025
387baf9
Merge branch 'main' into pr/92078
SmartManoj Apr 30, 2025
56bc655
regen
SmartManoj May 1, 2025
92c53b2
temp
SmartManoj May 1, 2025
629fa5b
Merge branch 'main' into pr/92078
SmartManoj May 1, 2025
9b73fd6
add missing arg
SmartManoj May 1, 2025
4fefead
use const
SmartManoj May 1, 2025
28647b0
update value
SmartManoj May 1, 2025
bdb8385
temp
SmartManoj May 1, 2025
861e403
Update datetimetester.py
SmartManoj May 1, 2025
7fddf11
Update datetime.h
SmartManoj May 1, 2025
0b0b214
fix arg
SmartManoj May 1, 2025
c2ed07c
temp
SmartManoj May 1, 2025
bae2e30
Merge branch 'main' into pr/92078
SmartManoj May 1, 2025
98297a1
lint
SmartManoj May 1, 2025
8d80782
temp
SmartManoj May 1, 2025
216b66b
Merge branch 'main' into pr/92078
SmartManoj May 2, 2025
c6c1084
doc
SmartManoj May 2, 2025
1fa2290
fix arg
SmartManoj May 2, 2025
3d44712
temp
SmartManoj May 2, 2025
eec6945
fix order
SmartManoj May 2, 2025
243fc01
set ns
SmartManoj May 3, 2025
30ca5fb
Update datetimetester.py
SmartManoj May 3, 2025
e359e90
lint
SmartManoj May 3, 2025
18f5344
Update datetimetester.py
SmartManoj May 3, 2025
f6b3da8
fix call_subclass_fold
SmartManoj May 3, 2025
90f7f74
fix pickle
SmartManoj May 3, 2025
bc241f0
Update datetime.rst
SmartManoj May 3, 2025
ecaf671
update call_subclass_fold
SmartManoj May 3, 2025
9853682
Update _pydatetime.py
SmartManoj May 3, 2025
f114afc
lint
SmartManoj May 3, 2025
797dcc4
fix parse_hh_mm_ss_ff
SmartManoj May 4, 2025
274aaee
upd ops
SmartManoj May 4, 2025
52dff86
account floating microseconds
SmartManoj May 4, 2025
02adcd8
test pure only now
SmartManoj May 4, 2025
b6a5155
Merge branch 'main' into pr/92078
SmartManoj May 4, 2025
e1b192a
fix nanoseconds_to_delta
SmartManoj May 4, 2025
a3c165b
temp
SmartManoj May 4, 2025
840f3a8
fix time_repr
SmartManoj May 4, 2025
0e6db63
temp
SmartManoj May 4, 2025
c141c4a
temp
SmartManoj May 4, 2025
120ed24
update delta_total_seconds
SmartManoj May 11, 2025
132f84b
update delta_new
SmartManoj May 11, 2025
e462ac8
temp
SmartManoj May 11, 2025
069f6ae
Merge branch 'main' into pr/92078
SmartManoj May 11, 2025
15b5f25
remove unused var
SmartManoj May 11, 2025
6ef2801
temp
SmartManoj May 11, 2025
f3a4344
fix delta_getstate
SmartManoj May 11, 2025
7ad5b67
update test_extreme_timedelta
SmartManoj May 11, 2025
ad22865
revert delta_new changes
SmartManoj May 12, 2025
49605ae
temp
SmartManoj May 12, 2025
e7feec3
temp
SmartManoj May 12, 2025
16f821b
Merge branch 'main' into pr/92078
SmartManoj May 12, 2025
6a5c4f0
remove microseconds float check
SmartManoj May 12, 2025
436775e
Clarify microseconds handling after divmod operation
SmartManoj May 13, 2025
f8d3402
separate long numbers
SmartManoj May 13, 2025
ed876f2
update delta_new
SmartManoj May 13, 2025
4d3573a
temp
SmartManoj May 13, 2025
f208cd4
Merge branch 'main' into pr/92078
SmartManoj May 13, 2025
77c4865
temp
SmartManoj May 13, 2025
9750570
fix multiply_int_timedelta
SmartManoj May 13, 2025
4062a59
temp
SmartManoj May 14, 2025
45108a0
Merge branch 'main' into pr/92078
SmartManoj May 14, 2025
01f273a
temp
SmartManoj May 17, 2025
d3857b2
Merge branch 'main' into pr/92078
SmartManoj May 17, 2025
469e077
lint
SmartManoj May 17, 2025
17decdc
move tmp functions
SmartManoj May 17, 2025
fd92889
remove partial duplicated logic
SmartManoj May 17, 2025
632eabd
update nanoseconds_to_delta for large number
SmartManoj May 17, 2025
33f382f
handle -ve leftover_ns
SmartManoj May 17, 2025
eb6e479
comment zero_delta due to weird bug
SmartManoj May 17, 2025
1c374af
temp-
SmartManoj May 17, 2025
7638a54
fix test_microsecond_rounding
SmartManoj May 17, 2025
8609df2
Merge branch 'main' into pr/92078
SmartManoj May 17, 2025
16becbe
normalize
SmartManoj May 17, 2025
236a840
fix segmentation fault
SmartManoj May 17, 2025
52bd399
temp-
SmartManoj May 17, 2025
5c29759
lint
SmartManoj May 17, 2025
1bfcc9e
Merge branch 'main' into pr/92078
SmartManoj May 17, 2025
960ae24
typo
SmartManoj May 18, 2025
7ab1368
add overflow test
SmartManoj May 18, 2025
fe8403c
fix rounding
SmartManoj May 18, 2025
dc92ee8
fix segfault
SmartManoj May 18, 2025
02f9e2b
temp-
SmartManoj May 18, 2025
10b28c1
Merge branch 'main' into pr/92078
SmartManoj May 18, 2025
28b661c
round if it has decimal points
SmartManoj May 18, 2025
1aeb27e
update datatype for windows
SmartManoj May 18, 2025
9abd6af
Merge branch 'main' into pr/92078
SmartManoj May 18, 2025
ef51bb3
fix order
SmartManoj May 18, 2025
28fd687
remove space
SmartManoj May 18, 2025
52a4436
remove extra dot
SmartManoj May 18, 2025
31307d2
add pickle compatibility
SmartManoj May 18, 2025
5e5711f
fix pickling
SmartManoj May 18, 2025
c4c5e76
lint
SmartManoj May 18, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
84 changes: 53 additions & 31 deletions Lib/datetime.py
Original file line number Diff line number Diff line change
Expand Up @@ -158,13 +158,14 @@ def _build_struct_time(y, m, d, hh, mm, ss, dstflag):
dnum = _days_before_month(y, m) + d
return _time.struct_time((y, m, d, hh, mm, ss, wday, dnum, dstflag))

def _format_time(hh, mm, ss, us, timespec='auto'):
def _format_time(hh, mm, ss, us, sus, timespec='auto'):
specs = {
'hours': '{:02d}',
'minutes': '{:02d}:{:02d}',
'seconds': '{:02d}:{:02d}:{:02d}',
'milliseconds': '{:02d}:{:02d}:{:02d}.{:03d}',
'microseconds': '{:02d}:{:02d}:{:02d}.{:06d}'
'microseconds': '{:02d}:{:02d}:{:02d}.{:06d}',
'submicroseconds': '{:02d}:{:02d}:{:02d}.{:06d}{:03d}',
}

if timespec == 'auto':
Expand All @@ -177,7 +178,7 @@ def _format_time(hh, mm, ss, us, timespec='auto'):
except KeyError:
raise ValueError('Unknown timespec value')
else:
return fmt.format(hh, mm, ss, us)
return fmt.format(hh, mm, ss, us, sus)

def _format_offset(off):
s = ''
Expand Down Expand Up @@ -525,11 +526,12 @@ def _check_date_fields(year, month, day):
raise ValueError('day must be in 1..%d' % dim, day)
return year, month, day

def _check_time_fields(hour, minute, second, microsecond, fold):
def _check_time_fields(hour, minute, second, microsecond, submicrosecond, fold):
hour = _index(hour)
minute = _index(minute)
second = _index(second)
microsecond = _index(microsecond)
submicrosecond = _index(submicrosecond)
if not 0 <= hour <= 23:
raise ValueError('hour must be in 0..23', hour)
if not 0 <= minute <= 59:
Expand All @@ -538,9 +540,11 @@ def _check_time_fields(hour, minute, second, microsecond, fold):
raise ValueError('second must be in 0..59', second)
if not 0 <= microsecond <= 999999:
raise ValueError('microsecond must be in 0..999999', microsecond)
if not 0 <= submicrosecond <= 999:
raise ValueError('Currently submicrosecond must be in 0..999', submicrosecond)
if fold not in (0, 1):
raise ValueError('fold must be either 0 or 1', fold)
return hour, minute, second, microsecond, fold
return hour, minute, second, microsecond, submicrosecond, fold

def _check_tzinfo_arg(tz):
if tz is not None and not isinstance(tz, tzinfo):
Expand Down Expand Up @@ -1274,7 +1278,15 @@ def __reduce__(self):
args = getinitargs()
else:
args = ()
return (self.__class__, args, self.__getstate__())
getstate = getattr(self, "__getstate__", None)
if getstate:
state = getstate()
else:
state = getattr(self, "__dict__", None) or None
if state is None:
return (self.__class__, args)
else:
return (self.__class__, args, state)


class IsoCalendarDate(tuple):
Expand Down Expand Up @@ -1331,9 +1343,9 @@ class time:
Properties (readonly):
hour, minute, second, microsecond, tzinfo, fold
"""
__slots__ = '_hour', '_minute', '_second', '_microsecond', '_tzinfo', '_hashcode', '_fold'
__slots__ = '_hour', '_minute', '_second', '_microsecond', '_submicrosecond', '_tzinfo', '_hashcode', '_fold'

def __new__(cls, hour=0, minute=0, second=0, microsecond=0, tzinfo=None, *, fold=0):
def __new__(cls, hour=0, minute=0, second=0, microsecond=0, tzinfo=None, *, fold=0, submicrosecond=0):
"""Constructor.

Arguments:
Expand All @@ -1359,14 +1371,15 @@ def __new__(cls, hour=0, minute=0, second=0, microsecond=0, tzinfo=None, *, fold
self.__setstate(hour, minute or None)
self._hashcode = -1
return self
hour, minute, second, microsecond, fold = _check_time_fields(
hour, minute, second, microsecond, fold)
hour, minute, second, microsecond, submicrosecond, fold = _check_time_fields(
hour, minute, second, microsecond, submicrosecond, fold)
_check_tzinfo_arg(tzinfo)
self = object.__new__(cls)
self._hour = hour
self._minute = minute
self._second = second
self._microsecond = microsecond
self._submicrosecond = submicrosecond
self._tzinfo = tzinfo
self._hashcode = -1
self._fold = fold
Expand All @@ -1393,6 +1406,11 @@ def microsecond(self):
"""microsecond (0-999999)"""
return self._microsecond

@property
def submicrosecond(self):
"""submicrosecond (0-999)"""
return self._submicrosecond

@property
def tzinfo(self):
"""timezone info object"""
Expand Down Expand Up @@ -1480,9 +1498,9 @@ def __hash__(self):
assert not m % timedelta(minutes=1), "whole minute"
m //= timedelta(minutes=1)
if 0 <= h < 24:
self._hashcode = hash(time(h, m, self.second, self.microsecond))
self._hashcode = hash(time(h, m, self.second, self.microsecond, self.submicrosecond))
else:
self._hashcode = hash((h, m, self.second, self.microsecond))
self._hashcode = hash((h, m, self.second, self.microsecond, self.submicrosecond))
return self._hashcode

# Conversion to string
Expand Down Expand Up @@ -1522,7 +1540,7 @@ def isoformat(self, timespec='auto'):
'minutes', 'seconds', 'milliseconds' and 'microseconds'.
"""
s = _format_time(self._hour, self._minute, self._second,
self._microsecond, timespec)
self._microsecond, self._submicrosecond, timespec)
tz = self._tzstr()
if tz:
s += tz
Expand Down Expand Up @@ -1671,7 +1689,7 @@ class datetime(date):
__slots__ = date.__slots__ + time.__slots__

def __new__(cls, year, month=None, day=None, hour=0, minute=0, second=0,
microsecond=0, tzinfo=None, *, fold=0):
microsecond=0, tzinfo=None, *, fold=0, submicrosecond=0):
if (isinstance(year, (bytes, str)) and len(year) == 10 and
1 <= ord(year[2:3])&0x7F <= 12):
# Pickle support
Expand All @@ -1689,8 +1707,8 @@ def __new__(cls, year, month=None, day=None, hour=0, minute=0, second=0,
self._hashcode = -1
return self
year, month, day = _check_date_fields(year, month, day)
hour, minute, second, microsecond, fold = _check_time_fields(
hour, minute, second, microsecond, fold)
hour, minute, second, microsecond, submicrosecond, fold = _check_time_fields(
hour, minute, second, microsecond, submicrosecond, fold)
_check_tzinfo_arg(tzinfo)
self = object.__new__(cls)
self._year = year
Expand All @@ -1700,6 +1718,7 @@ def __new__(cls, year, month=None, day=None, hour=0, minute=0, second=0,
self._minute = minute
self._second = second
self._microsecond = microsecond
self._submicrosecond = submicrosecond
self._tzinfo = tzinfo
self._hashcode = -1
self._fold = fold
Expand Down Expand Up @@ -1741,19 +1760,21 @@ def _fromtimestamp(cls, t, utc, tz):

A timezone info object may be passed in as well.
"""
frac, t = _math.modf(t)
us = round(frac * 1e6)
if us >= 1000000:
t += 1
us -= 1000000
elif us < 0:
t -= 1
us += 1000000

# f, a = math.modf(1651815892989527.7)
# print(f, a)
# 0.75 1651815892989527.0
# f = 0.7 -> 0.75 precision broken

# Decimal is a heavyweight module

t, dp = str(t).split('.')
us, sus = dp[:6], f'{dp[6:9]:0<3}'
t, us, sus = int(t), int(us), int(sus)
converter = _time.gmtime if utc else _time.localtime
y, m, d, hh, mm, ss, weekday, jday, dst = converter(t)
ss = min(ss, 59) # clamp out leap seconds if the platform has them
result = cls(y, m, d, hh, mm, ss, us, tz)
result = cls(y, m, d, hh, mm, ss, us, tz, submicrosecond=sus)
if tz is None and not utc:
# As of version 2015f max fold in IANA database is
# 23 hours at 1969-09-30 13:00:00 in Kwajalein.
Expand All @@ -1768,11 +1789,11 @@ def _fromtimestamp(cls, t, utc, tz):
return result

y, m, d, hh, mm, ss = converter(t - max_fold_seconds)[:6]
probe1 = cls(y, m, d, hh, mm, ss, us, tz)
probe1 = cls(y, m, d, hh, mm, ss, us, tz, submicrosecond=sus)
trans = result - probe1 - timedelta(0, max_fold_seconds)
if trans.days < 0:
y, m, d, hh, mm, ss = converter(t + trans // timedelta(0, 1))[:6]
probe2 = cls(y, m, d, hh, mm, ss, us, tz)
probe2 = cls(y, m, d, hh, mm, ss, us, tz, submicrosecond=sus)
if probe2 == result:
result._fold = 1
elif tz is not None:
Expand All @@ -1797,13 +1818,13 @@ def utcfromtimestamp(cls, t):
@classmethod
def now(cls, tz=None):
"Construct a datetime from time.time() and optional time zone info."
t = _time.time()
t = _time.time_ns() / 1e9
return cls.fromtimestamp(t, tz)

@classmethod
def utcnow(cls):
"Construct a UTC datetime from time.time()."
t = _time.time()
t = _time.time_ns() / 1e9
return cls.utcfromtimestamp(t)

@classmethod
Expand Down Expand Up @@ -2020,7 +2041,7 @@ def isoformat(self, sep='T', timespec='auto'):
"""
s = ("%04d-%02d-%02d%c" % (self._year, self._month, self._day, sep) +
_format_time(self._hour, self._minute, self._second,
self._microsecond, timespec))
self._microsecond, self._submicrosecond, timespec))

off = self.utcoffset()
tz = _format_offset(off)
Expand Down Expand Up @@ -2612,7 +2633,8 @@ def _name_from_offset(delta):
# pretty bizarre, and a tzinfo subclass can override fromutc() if it is.

try:
from _datetime import *
# TODO: temp comment for testing
from _datetime2 import *
except ImportError:
pass
else:
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Nanosecond support for datetime