Kaydet (Commit) 1184e266 authored tarafından Benjamin Peterson's avatar Benjamin Peterson

do not override errors from descriptors on modules

üst 7b9ff0e6
...@@ -227,6 +227,14 @@ a = A(destroyed)""" ...@@ -227,6 +227,14 @@ a = A(destroyed)"""
b"len = len", b"len = len",
b"shutil.rmtree = rmtree"}) b"shutil.rmtree = rmtree"})
def test_descriptor_errors_propogate(self):
class Descr:
def __get__(self, o, t):
raise RuntimeError
class M(ModuleType):
melon = Descr()
self.assertRaises(RuntimeError, getattr, M("mymod"), "melon")
# frozen and namespace module reprs are tested in importlib. # frozen and namespace module reprs are tested in importlib.
......
...@@ -412,24 +412,23 @@ module_repr(PyModuleObject *m) ...@@ -412,24 +412,23 @@ module_repr(PyModuleObject *m)
} }
static PyObject* static PyObject*
module_getattr(PyObject *m, PyObject *name) module_getattro(PyModuleObject *m, PyObject *name)
{ {
PyModuleObject *module;
PyObject *attr, *mod_name; PyObject *attr, *mod_name;
attr = PyObject_GenericGetAttr(m, name); attr = PyObject_GenericGetAttr((PyObject *)m, name);
if (attr != NULL) if (attr || !PyErr_ExceptionMatches(PyExc_AttributeError))
return attr; return attr;
PyErr_Clear(); PyErr_Clear();
module = (PyModuleObject*)m; if (m->md_dict) {
if (module->md_dict != NULL) { mod_name = PyDict_GetItemString(m->md_dict, "__name__");
mod_name = PyDict_GetItemString(module->md_dict, "__name__"); if (mod_name) {
if (mod_name != NULL) {
PyErr_Format(PyExc_AttributeError, PyErr_Format(PyExc_AttributeError,
"module '%U' has no attribute '%U'", mod_name, name); "module '%U' has no attribute '%U'", mod_name, name);
return NULL; return NULL;
} }
else if (PyErr_Occurred()) else if (PyErr_Occurred()) {
PyErr_Clear(); PyErr_Clear();
}
} }
PyErr_Format(PyExc_AttributeError, PyErr_Format(PyExc_AttributeError,
"module has no attribute '%U'", name); "module has no attribute '%U'", name);
...@@ -512,7 +511,7 @@ PyTypeObject PyModule_Type = { ...@@ -512,7 +511,7 @@ PyTypeObject PyModule_Type = {
0, /* tp_hash */ 0, /* tp_hash */
0, /* tp_call */ 0, /* tp_call */
0, /* tp_str */ 0, /* tp_str */
module_getattr, /* tp_getattro */ (getattrofunc)module_getattro, /* tp_getattro */
PyObject_GenericSetAttr, /* tp_setattro */ PyObject_GenericSetAttr, /* tp_setattro */
0, /* tp_as_buffer */ 0, /* tp_as_buffer */
Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC | Py_TPFLAGS_DEFAULT | Py_TPFLAGS_HAVE_GC |
......
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