self.assertRaises(TypeError, math.atan2, 1.0)
self.assertRaises(TypeError, math.atan2, 1.0, 2.0, 3.0)
+ def test_exception_messages(self):
+ x = -1.1
+ with self.assertRaisesRegex(ValueError,
+ f"expected a nonnegative input, got {x}"):
+ math.sqrt(x)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log(x)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log(123, x)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log(x, 123)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log2(x)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log10(x)
+ x = decimal.Decimal('-1.1')
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log(x)
+ x = fractions.Fraction(1, 10**400)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {float(x)}"):
+ math.log(x)
+ x = -123
+ with self.assertRaisesRegex(ValueError,
+ f"expected a positive input, got {x}"):
+ math.log(x)
+ with self.assertRaisesRegex(ValueError,
+ f"expected a float or nonnegative integer, got {x}"):
+ math.gamma(x)
+ x = 1.0
+ with self.assertRaisesRegex(ValueError,
+ f"expected a number between -1 and 1, got {x}"):
+ math.atanh(x)
+
# Custom assertions.
def assertIsNaN(self, value):
* true (1), but may return false (0) without setting up an exception.
*/
static int
-is_error(double x)
+is_error(double x, int raise_edom)
{
int result = 1; /* presumption of guilt */
assert(errno); /* non-zero errno is a precondition for calling */
- if (errno == EDOM)
- PyErr_SetString(PyExc_ValueError, "math domain error");
+ if (errno == EDOM) {
+ if (raise_edom) {
+ PyErr_SetString(PyExc_ValueError, "math domain error");
+ }
+ }
else if (errno == ERANGE) {
/* ANSI C generally requires libm functions to set ERANGE
*/
static PyObject *
-math_1(PyObject *arg, double (*func) (double), int can_overflow)
+math_1(PyObject *arg, double (*func) (double), int can_overflow,
+ const char *err_msg)
{
double x, r;
x = PyFloat_AsDouble(arg);
return NULL;
errno = 0;
r = (*func)(x);
- if (isnan(r) && !isnan(x)) {
- PyErr_SetString(PyExc_ValueError,
- "math domain error"); /* invalid arg */
- return NULL;
- }
+ if (isnan(r) && !isnan(x))
+ goto domain_err; /* domain error */
if (isinf(r) && isfinite(x)) {
if (can_overflow)
PyErr_SetString(PyExc_OverflowError,
"math range error"); /* overflow */
else
- PyErr_SetString(PyExc_ValueError,
- "math domain error"); /* singularity */
+ goto domain_err; /* singularity */
return NULL;
}
- if (isfinite(r) && errno && is_error(r))
+ if (isfinite(r) && errno && is_error(r, 1))
/* this branch unnecessary on most platforms */
return NULL;
return PyFloat_FromDouble(r);
+
+domain_err:
+ if (err_msg) {
+ char *buf = PyOS_double_to_string(x, 'r', 0, Py_DTSF_ADD_DOT_0, NULL);
+ if (buf) {
+ PyErr_Format(PyExc_ValueError, err_msg, buf);
+ PyMem_Free(buf);
+ }
+ }
+ else {
+ PyErr_SetString(PyExc_ValueError, "math domain error");
+ }
+ return NULL;
}
/* variant of math_1, to be used when the function being wrapped is known to
errno = ERANGE for overflow). */
static PyObject *
-math_1a(PyObject *arg, double (*func) (double))
+math_1a(PyObject *arg, double (*func) (double), const char *err_msg)
{
double x, r;
x = PyFloat_AsDouble(arg);
return NULL;
errno = 0;
r = (*func)(x);
- if (errno && is_error(r))
+ if (errno && is_error(r, err_msg ? 0 : 1)) {
+ if (err_msg && errno == EDOM) {
+ assert(!PyErr_Occurred()); /* exception is not set by is_error() */
+ char *buf = PyOS_double_to_string(x, 'r', 0, Py_DTSF_ADD_DOT_0, NULL);
+ if (buf) {
+ PyErr_Format(PyExc_ValueError, err_msg, buf);
+ PyMem_Free(buf);
+ }
+ }
return NULL;
+ }
return PyFloat_FromDouble(r);
}
else
errno = 0;
}
- if (errno && is_error(r))
+ if (errno && is_error(r, 1))
return NULL;
else
return PyFloat_FromDouble(r);
#define FUNC1(funcname, func, can_overflow, docstring) \
static PyObject * math_##funcname(PyObject *self, PyObject *args) { \
- return math_1(args, func, can_overflow); \
+ return math_1(args, func, can_overflow, NULL); \
+ }\
+ PyDoc_STRVAR(math_##funcname##_doc, docstring);
+
+#define FUNC1D(funcname, func, can_overflow, docstring, err_msg) \
+ static PyObject * math_##funcname(PyObject *self, PyObject *args) { \
+ return math_1(args, func, can_overflow, err_msg); \
}\
PyDoc_STRVAR(math_##funcname##_doc, docstring);
#define FUNC1A(funcname, func, docstring) \
static PyObject * math_##funcname(PyObject *self, PyObject *args) { \
- return math_1a(args, func); \
+ return math_1a(args, func, NULL); \
+ }\
+ PyDoc_STRVAR(math_##funcname##_doc, docstring);
+
+#define FUNC1AD(funcname, func, docstring, err_msg) \
+ static PyObject * math_##funcname(PyObject *self, PyObject *args) { \
+ return math_1a(args, func, err_msg); \
}\
PyDoc_STRVAR(math_##funcname##_doc, docstring);
"atan2($module, y, x, /)\n--\n\n"
"Return the arc tangent (measured in radians) of y/x.\n\n"
"Unlike atan(y/x), the signs of both x and y are considered.")
-FUNC1(atanh, atanh, 0,
+FUNC1D(atanh, atanh, 0,
"atanh($module, x, /)\n--\n\n"
- "Return the inverse hyperbolic tangent of x.")
+ "Return the inverse hyperbolic tangent of x.",
+ "expected a number between -1 and 1, got %s")
FUNC1(cbrt, cbrt, 0,
"cbrt($module, x, /)\n--\n\n"
"Return the cube root of x.")
return PyLong_FromDouble(floor(x));
}
-FUNC1A(gamma, m_tgamma,
+FUNC1AD(gamma, m_tgamma,
"gamma($module, x, /)\n--\n\n"
- "Gamma function at x.")
+ "Gamma function at x.",
+ "expected a float or nonnegative integer, got %s")
FUNC1A(lgamma, m_lgamma,
"lgamma($module, x, /)\n--\n\n"
"Natural logarithm of absolute value of Gamma function at x.")
FUNC1(sinh, sinh, 1,
"sinh($module, x, /)\n--\n\n"
"Return the hyperbolic sine of x.")
-FUNC1(sqrt, sqrt, 0,
+FUNC1D(sqrt, sqrt, 0,
"sqrt($module, x, /)\n--\n\n"
- "Return the square root of x.")
+ "Return the square root of x.",
+ "expected a nonnegative input, got %s")
FUNC1(tan, tan, 0,
"tan($module, x, /)\n--\n\n"
"Return the tangent of x (measured in radians).")
errno = ERANGE;
}
- if (errno && is_error(r))
+ if (errno && is_error(r, 1))
return NULL;
return PyFloat_FromDouble(r);
}
/* Negative or zero inputs give a ValueError. */
if (!_PyLong_IsPositive((PyLongObject *)arg)) {
- PyErr_SetString(PyExc_ValueError,
- "math domain error");
+ PyErr_Format(PyExc_ValueError,
+ "expected a positive input, got %S", arg);
return NULL;
}
}
/* Else let libm handle it by itself. */
- return math_1(arg, func, 0);
+ return math_1(arg, func, 0, "expected a positive input, got %s");
}
else
errno = 0;
}
- if (errno && is_error(r))
+ if (errno && is_error(r, 1))
return NULL;
else
return PyFloat_FromDouble(r);
}
}
- if (errno && is_error(r))
+ if (errno && is_error(r, 1))
return NULL;
else
return PyFloat_FromDouble(r);