From: Serhiy Storchaka Date: Fri, 7 Oct 2016 20:32:41 +0000 (+0300) Subject: Issue #28257: Improved error message when pass a non-mapping as a var-keyword X-Git-Tag: v3.6.0b2~32^2 X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=5665301baea21200771829516fc8edeec3b059b9;p=python Issue #28257: Improved error message when pass a non-mapping as a var-keyword argument. --- diff --git a/Lib/test/test_extcall.py b/Lib/test/test_extcall.py index 015cdfcf1a..9cb0d38669 100644 --- a/Lib/test/test_extcall.py +++ b/Lib/test/test_extcall.py @@ -269,6 +269,16 @@ not function ... TypeError: h() argument after ** must be a mapping, not list + >>> h(**{'a': 1}, **h) + Traceback (most recent call last): + ... + TypeError: h() argument after ** must be a mapping, not function + + >>> h(**{'a': 1}, **[]) + Traceback (most recent call last): + ... + TypeError: h() argument after ** must be a mapping, not list + >>> dir(**h) Traceback (most recent call last): ... diff --git a/Python/ceval.c b/Python/ceval.c index 9f1af788aa..e9d0cbb976 100644 --- a/Python/ceval.c +++ b/Python/ceval.c @@ -2663,7 +2663,8 @@ PyEval_EvalFrameEx(PyFrameObject *f, int throwflag) PyObject *intersection = _PyDictView_Intersect(sum, arg); if (intersection == NULL) { - if (PyErr_ExceptionMatches(PyExc_AttributeError)) { + if (PyErr_ExceptionMatches(PyExc_AttributeError) || + !PyMapping_Check(arg)) { int function_location = (oparg>>8) & 0xff; PyObject *func = ( PEEK(function_location + num_maps)); @@ -2707,9 +2708,21 @@ PyEval_EvalFrameEx(PyFrameObject *f, int throwflag) if (PyDict_Update(sum, arg) < 0) { if (PyErr_ExceptionMatches(PyExc_AttributeError)) { - PyErr_Format(PyExc_TypeError, - "'%.200s' object is not a mapping", - arg->ob_type->tp_name); + if (with_call) { + int function_location = (oparg>>8) & 0xff; + PyObject *func = PEEK(function_location + num_maps); + PyErr_Format(PyExc_TypeError, + "%.200s%.200s argument after ** " + "must be a mapping, not %.200s", + PyEval_GetFuncName(func), + PyEval_GetFuncDesc(func), + arg->ob_type->tp_name); + } + else { + PyErr_Format(PyExc_TypeError, + "'%.200s' object is not a mapping", + arg->ob_type->tp_name); + } } Py_DECREF(sum); goto error;