Commit 231aad38 authored by Serhiy Storchaka's avatar Serhiy Storchaka Committed by GitHub

bpo-37315: Deprecate accepting floats in math.factorial(). (GH-14147)

parent 1ce2656f
...@@ -71,6 +71,9 @@ Number-theoretic and representation functions ...@@ -71,6 +71,9 @@ Number-theoretic and representation functions
Return *x* factorial as an integer. Raises :exc:`ValueError` if *x* is not integral or Return *x* factorial as an integer. Raises :exc:`ValueError` if *x* is not integral or
is negative. is negative.
.. deprecated:: 3.9
Accepting floats with integral values (like ``5.0``) is deprecated.
.. function:: floor(x) .. function:: floor(x)
......
...@@ -109,6 +109,11 @@ Build and C API Changes ...@@ -109,6 +109,11 @@ Build and C API Changes
Deprecated Deprecated
========== ==========
* Currently :func:`math.factorial` accepts :class:`float` instances with
non-negative integer values (like ``5.0``). It raises a :exc:`ValueError`
for non-integral and negative floats. It is deprecated now. In future
Python versions it will raise a :exc:`TypeError` for all floats.
(Contributed by Serhiy Storchaka in :issue:`37315`.)
Removed Removed
......
...@@ -501,21 +501,25 @@ class MathTests(unittest.TestCase): ...@@ -501,21 +501,25 @@ class MathTests(unittest.TestCase):
def testFactorial(self): def testFactorial(self):
self.assertEqual(math.factorial(0), 1) self.assertEqual(math.factorial(0), 1)
self.assertEqual(math.factorial(0.0), 1)
total = 1 total = 1
for i in range(1, 1000): for i in range(1, 1000):
total *= i total *= i
self.assertEqual(math.factorial(i), total) self.assertEqual(math.factorial(i), total)
self.assertEqual(math.factorial(float(i)), total)
self.assertEqual(math.factorial(i), py_factorial(i)) self.assertEqual(math.factorial(i), py_factorial(i))
self.assertRaises(ValueError, math.factorial, -1) self.assertRaises(ValueError, math.factorial, -1)
self.assertRaises(ValueError, math.factorial, -1.0)
self.assertRaises(ValueError, math.factorial, -10**100) self.assertRaises(ValueError, math.factorial, -10**100)
self.assertRaises(ValueError, math.factorial, -1e100)
self.assertRaises(ValueError, math.factorial, math.pi)
def testFactorialNonIntegers(self): def testFactorialNonIntegers(self):
self.assertRaises(TypeError, math.factorial, decimal.Decimal(5.2)) with self.assertWarns(DeprecationWarning):
self.assertEqual(math.factorial(5.0), 120)
with self.assertWarns(DeprecationWarning):
self.assertRaises(ValueError, math.factorial, 5.2)
with self.assertWarns(DeprecationWarning):
self.assertRaises(ValueError, math.factorial, -1.0)
with self.assertWarns(DeprecationWarning):
self.assertRaises(ValueError, math.factorial, -1e100)
self.assertRaises(TypeError, math.factorial, decimal.Decimal('5'))
self.assertRaises(TypeError, math.factorial, decimal.Decimal('5.2'))
self.assertRaises(TypeError, math.factorial, "5") self.assertRaises(TypeError, math.factorial, "5")
# Other implementations may place different upper bounds. # Other implementations may place different upper bounds.
...@@ -524,6 +528,7 @@ class MathTests(unittest.TestCase): ...@@ -524,6 +528,7 @@ class MathTests(unittest.TestCase):
# Currently raises ValueError for inputs that are too large # Currently raises ValueError for inputs that are too large
# to fit into a C long. # to fit into a C long.
self.assertRaises(OverflowError, math.factorial, 10**100) self.assertRaises(OverflowError, math.factorial, 10**100)
with self.assertWarns(DeprecationWarning):
self.assertRaises(OverflowError, math.factorial, 1e100) self.assertRaises(OverflowError, math.factorial, 1e100)
def testFloor(self): def testFloor(self):
......
Deprecated accepting floats with integral value (like ``5.0``) in
:func:`math.factorial`.
...@@ -1981,6 +1981,12 @@ math_factorial(PyObject *module, PyObject *arg) ...@@ -1981,6 +1981,12 @@ math_factorial(PyObject *module, PyObject *arg)
PyObject *result, *odd_part, *pyint_form; PyObject *result, *odd_part, *pyint_form;
if (PyFloat_Check(arg)) { if (PyFloat_Check(arg)) {
if (PyErr_WarnEx(PyExc_DeprecationWarning,
"Using factorial() with floats is deprecated",
1) < 0)
{
return NULL;
}
PyObject *lx; PyObject *lx;
double dx = PyFloat_AS_DOUBLE((PyFloatObject *)arg); double dx = PyFloat_AS_DOUBLE((PyFloatObject *)arg);
if (!(Py_IS_FINITE(dx) && dx == floor(dx))) { if (!(Py_IS_FINITE(dx) && dx == floor(dx))) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment