import test.test_lazy_import.data.dunder_lazy_import_used
self.assertIn("test.test_lazy_import.data.basic2", sys.modules)
+ @support.requires_subprocess()
+ def test_dunder_lazy_import_fromlist_resolves_to_module(self):
+ for fromlist in ["basic2", ("basic2",)]:
+ with self.subTest(fromlist=fromlist):
+ code = textwrap.dedent(f"""
+ import sys
+ import types
+
+ lazy = __lazy_import__("test.test_lazy_import.data", fromlist={fromlist!r})
+
+ def check():
+ lazy_obj = globals()["lazy"]
+ assert type(lazy_obj) is types.LazyImportType, lazy_obj
+ assert "test.test_lazy_import.data.basic2" not in sys.modules
+
+ resolved = lazy_obj.resolve()
+ assert type(resolved) is types.ModuleType, resolved
+ assert "test.test_lazy_import.data.basic2" in sys.modules
+ assert resolved.__name__ == "test.test_lazy_import.data"
+ assert resolved.basic2.x == 42
+
+ check()
+ """)
+ assert_python_ok("-c", code)
+
def test_dunder_lazy_import_invalid_arguments(self):
"""__lazy_import__ should reject invalid arguments."""
for invalid_name in (b"", 123, None):
--- /dev/null
+Fix an inconsistency where calling ``__lazy_import__`` with a string
+``fromlist`` would return a :class:`types.LazyImportType` that resolves to
+the named member, rather than the module being imported.
}
if (fromlist == NULL) {
assert(!PyErr_Occurred());
- fromlist = Py_NewRef(Py_None);
+ fromlist = Py_None;
}
PyObject *args[] = {modname, abs_name, fromlist};
PyObject *res = PyObject_Vectorcall(filter, args, 3, NULL);
}
// here, 'filter' is either NULL or is equivalent to a borrowed reference
+ if (fromlist && PyUnicode_Check(fromlist)) {
+ fromlist = PyTuple_Pack(1, fromlist);
+ if (fromlist == NULL) {
+ Py_DECREF(abs_name);
+ return NULL;
+ }
+ }
+ else {
+ Py_XINCREF(fromlist);
+ }
PyObject *res = _PyLazyImport_New(frame, builtins, abs_name, fromlist);
if (res == NULL) {
+ Py_XDECREF(fromlist);
Py_DECREF(abs_name);
return NULL;
}
goto error;
}
- if (fromlist && PyUnicode_Check(fromlist)) {
- if (register_from_lazy_on_parent(tstate, abs_name, fromlist) < 0) {
- goto error;
- }
- }
- else if (fromlist && PyTuple_Check(fromlist) &&
- PyTuple_GET_SIZE(fromlist)) {
+ if (fromlist && PyTuple_Check(fromlist) && PyTuple_GET_SIZE(fromlist)) {
for (Py_ssize_t i = 0; i < PyTuple_GET_SIZE(fromlist); i++) {
if (register_from_lazy_on_parent(tstate, abs_name,
PyTuple_GET_ITEM(fromlist, i)) < 0)
goto error;
}
+ Py_XDECREF(fromlist);
Py_DECREF(abs_name);
return res;
error:
+ Py_XDECREF(fromlist);
Py_DECREF(abs_name);
Py_DECREF(res);
return NULL;