Skip to content

Commit 126910e

Browse files
authored
pythongh-122272: Guarantee specifiers %F and %C for datetime.strftime to be 0-padded (pythonGH-122436)
1 parent 7cd3aa4 commit 126910e

7 files changed

+145
-14
lines changed

Lib/_pydatetime.py

+21-4
Original file line numberDiff line numberDiff line change
@@ -215,6 +215,17 @@ def _need_normalize_century():
215215
_normalize_century = True
216216
return _normalize_century
217217

218+
_supports_c99 = None
219+
def _can_support_c99():
220+
global _supports_c99
221+
if _supports_c99 is None:
222+
try:
223+
_supports_c99 = (
224+
_time.strftime("%F", (1900, 1, 1, 0, 0, 0, 0, 1, 0)) == "1900-01-01")
225+
except ValueError:
226+
_supports_c99 = False
227+
return _supports_c99
228+
218229
# Correctly substitute for %z and %Z escapes in strftime formats.
219230
def _wrap_strftime(object, format, timetuple):
220231
# Don't call utcoffset() or tzname() unless actually needed.
@@ -272,14 +283,20 @@ def _wrap_strftime(object, format, timetuple):
272283
# strftime is going to have at this: escape %
273284
Zreplace = s.replace('%', '%%')
274285
newformat.append(Zreplace)
275-
elif ch in 'YG' and object.year < 1000 and _need_normalize_century():
276-
# Note that datetime(1000, 1, 1).strftime('%G') == '1000' so
277-
# year 1000 for %G can go on the fast path.
286+
# Note that datetime(1000, 1, 1).strftime('%G') == '1000' so
287+
# year 1000 for %G can go on the fast path.
288+
elif ((ch in 'YG' or ch in 'FC' and _can_support_c99()) and
289+
object.year < 1000 and _need_normalize_century()):
278290
if ch == 'G':
279291
year = int(_time.strftime("%G", timetuple))
280292
else:
281293
year = object.year
282-
push('{:04}'.format(year))
294+
if ch == 'C':
295+
push('{:02}'.format(year // 100))
296+
else:
297+
push('{:04}'.format(year))
298+
if ch == 'F':
299+
push('-{:02}-{:02}'.format(*timetuple[1:3]))
283300
else:
284301
push('%')
285302
push(ch)

Lib/test/datetimetester.py

+13-4
Original file line numberDiff line numberDiff line change
@@ -1710,13 +1710,22 @@ def test_strftime_y2k(self):
17101710
(1000, 0),
17111711
(1970, 0),
17121712
)
1713-
for year, offset in dataset:
1714-
for specifier in 'YG':
1713+
specifiers = 'YG'
1714+
if _time.strftime('%F', (1900, 1, 1, 0, 0, 0, 0, 1, 0)) == '1900-01-01':
1715+
specifiers += 'FC'
1716+
for year, g_offset in dataset:
1717+
for specifier in specifiers:
17151718
with self.subTest(year=year, specifier=specifier):
17161719
d = self.theclass(year, 1, 1)
17171720
if specifier == 'G':
1718-
year += offset
1719-
self.assertEqual(d.strftime(f"%{specifier}"), f"{year:04d}")
1721+
year += g_offset
1722+
if specifier == 'C':
1723+
expected = f"{year // 100:02d}"
1724+
else:
1725+
expected = f"{year:04d}"
1726+
if specifier == 'F':
1727+
expected += f"-01-01"
1728+
self.assertEqual(d.strftime(f"%{specifier}"), expected)
17201729

17211730
def test_replace(self):
17221731
cls = self.theclass
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
On some platforms such as Linux, year with century was not 0-padded when formatted by :meth:`~.datetime.strftime` with C99-specific specifiers ``'%C'`` or ``'%F'``. The 0-padding behavior is now guaranteed when the format specifiers ``'%C'`` and ``'%F'`` are supported by the C library.
2+
Patch by Ben Hsing

Modules/_datetimemodule.c

+26-6
Original file line numberDiff line numberDiff line change
@@ -1853,7 +1853,12 @@ wrap_strftime(PyObject *object, PyObject *format, PyObject *timetuple,
18531853

18541854
#ifdef Py_NORMALIZE_CENTURY
18551855
/* Buffer of maximum size of formatted year permitted by long. */
1856-
char buf[SIZEOF_LONG*5/2+2];
1856+
char buf[SIZEOF_LONG * 5 / 2 + 2
1857+
#ifdef Py_STRFTIME_C99_SUPPORT
1858+
/* Need 6 more to accomodate dashes, 2-digit month and day for %F. */
1859+
+ 6
1860+
#endif
1861+
];
18571862
#endif
18581863

18591864
assert(object && format && timetuple);
@@ -1950,11 +1955,18 @@ wrap_strftime(PyObject *object, PyObject *format, PyObject *timetuple,
19501955
ntoappend = PyBytes_GET_SIZE(freplacement);
19511956
}
19521957
#ifdef Py_NORMALIZE_CENTURY
1953-
else if (ch == 'Y' || ch == 'G') {
1958+
else if (ch == 'Y' || ch == 'G'
1959+
#ifdef Py_STRFTIME_C99_SUPPORT
1960+
|| ch == 'F' || ch == 'C'
1961+
#endif
1962+
) {
19541963
/* 0-pad year with century as necessary */
1955-
PyObject *item = PyTuple_GET_ITEM(timetuple, 0);
1964+
PyObject *item = PySequence_GetItem(timetuple, 0);
1965+
if (item == NULL) {
1966+
goto Done;
1967+
}
19561968
long year_long = PyLong_AsLong(item);
1957-
1969+
Py_DECREF(item);
19581970
if (year_long == -1 && PyErr_Occurred()) {
19591971
goto Done;
19601972
}
@@ -1980,8 +1992,16 @@ wrap_strftime(PyObject *object, PyObject *format, PyObject *timetuple,
19801992
goto Done;
19811993
}
19821994
}
1983-
1984-
ntoappend = PyOS_snprintf(buf, sizeof(buf), "%04ld", year_long);
1995+
ntoappend = PyOS_snprintf(buf, sizeof(buf),
1996+
#ifdef Py_STRFTIME_C99_SUPPORT
1997+
ch == 'F' ? "%04ld-%%m-%%d" :
1998+
#endif
1999+
"%04ld", year_long);
2000+
#ifdef Py_STRFTIME_C99_SUPPORT
2001+
if (ch == 'C') {
2002+
ntoappend -= 2;
2003+
}
2004+
#endif
19852005
ptoappend = buf;
19862006
}
19872007
#endif

configure

+52
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

configure.ac

+28
Original file line numberDiff line numberDiff line change
@@ -6703,6 +6703,34 @@ then
67036703
[Define if year with century should be normalized for strftime.])
67046704
fi
67056705

6706+
AC_CACHE_CHECK([whether C99-specific strftime specifiers are supported], [ac_cv_strftime_c99_support], [
6707+
AC_RUN_IFELSE([AC_LANG_SOURCE([[
6708+
#include <time.h>
6709+
#include <string.h>
6710+
6711+
int main(void)
6712+
{
6713+
char full_date[11];
6714+
struct tm date = {
6715+
.tm_year = 0,
6716+
.tm_mon = 0,
6717+
.tm_mday = 1
6718+
};
6719+
if (strftime(full_date, sizeof(full_date), "%F", &date) && !strcmp(full_date, "1900-01-01")) {
6720+
return 0;
6721+
}
6722+
return 1;
6723+
}
6724+
]])],
6725+
[ac_cv_strftime_c99_support=yes],
6726+
[ac_cv_strftime_c99_support=no],
6727+
[ac_cv_strftime_c99_support=no])])
6728+
if test "$ac_cv_strftime_c99_support" = yes
6729+
then
6730+
AC_DEFINE([Py_STRFTIME_C99_SUPPORT], [1],
6731+
[Define if C99-specific strftime specifiers are supported.])
6732+
fi
6733+
67066734
dnl check for ncursesw/ncurses and panelw/panel
67076735
dnl NOTE: old curses is not detected.
67086736
dnl have_curses=[no, yes]

pyconfig.h.in

+3
Original file line numberDiff line numberDiff line change
@@ -1701,6 +1701,9 @@
17011701
/* Define if you want to enable internal statistics gathering. */
17021702
#undef Py_STATS
17031703

1704+
/* Define if C99-specific strftime specifiers are supported. */
1705+
#undef Py_STRFTIME_C99_SUPPORT
1706+
17041707
/* The version of SunOS/Solaris as reported by `uname -r' without the dot. */
17051708
#undef Py_SUNOS_VERSION
17061709

0 commit comments

Comments
 (0)