[3.12] gh-108732: include comprehension locals in frame.f_locals (GH-109026) (#109097)

gh-108732: include comprehension locals in frame.f_locals (GH-109026)
(cherry picked from commit f2584eade3)

Co-authored-by: Carl Meyer <carl@oddbird.net>
Co-authored-by: Radislav Chugunov <52372310+chgnrdv@users.noreply.github.com>
Co-authored-by: Jelle Zijlstra <jelle.zijlstra@gmail.com>
This commit is contained in:
Miss Islington (bot) 2023-09-12 06:50:29 -07:00 committed by GitHub
parent af83d1e821
commit d533ab17ec
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 19 additions and 4 deletions

View file

@ -596,6 +596,13 @@ class ListComprehensionTest(unittest.TestCase):
""" """
self._check_in_scopes(code, {"value": [1, None]}) self._check_in_scopes(code, {"value": [1, None]})
def test_frame_locals(self):
code = """
val = [sys._getframe().f_locals for a in [0]][0]["a"]
"""
import sys
self._check_in_scopes(code, {"val": 0}, ns={"sys": sys})
__test__ = {'doctests' : doctests} __test__ = {'doctests' : doctests}

View file

@ -0,0 +1,2 @@
Make iteration variables of module- and class-scoped comprehensions visible
to pdb and other tools that use ``frame.f_locals`` again.

View file

@ -24,10 +24,16 @@ static PyMemberDef frame_memberlist[] = {
static PyObject * static PyObject *
frame_getlocals(PyFrameObject *f, void *closure) frame_getlocals(PyFrameObject *f, void *closure)
{ {
if (PyFrame_FastToLocalsWithError(f) < 0) if (f == NULL) {
PyErr_BadInternalCall();
return NULL; return NULL;
PyObject *locals = f->f_frame->f_locals; }
return Py_NewRef(locals); assert(!_PyFrame_IsIncomplete(f->f_frame));
PyObject *locals = _PyFrame_GetLocals(f->f_frame, 1);
if (locals) {
f->f_fast_as_locals = 1;
}
return locals;
} }
int int
@ -1351,11 +1357,11 @@ PyFrame_GetVarString(PyFrameObject *frame, const char *name)
int int
PyFrame_FastToLocalsWithError(PyFrameObject *f) PyFrame_FastToLocalsWithError(PyFrameObject *f)
{ {
assert(!_PyFrame_IsIncomplete(f->f_frame));
if (f == NULL) { if (f == NULL) {
PyErr_BadInternalCall(); PyErr_BadInternalCall();
return -1; return -1;
} }
assert(!_PyFrame_IsIncomplete(f->f_frame));
int err = _PyFrame_FastToLocalsWithError(f->f_frame); int err = _PyFrame_FastToLocalsWithError(f->f_frame);
if (err == 0) { if (err == 0) {
f->f_fast_as_locals = 1; f->f_fast_as_locals = 1;