https://github.com/python/cpython/commit/93ef7aa03c64a97d8615ad3975083392ad07b379
commit: 93ef7aa03c64a97d8615ad3975083392ad07b379
branch: 3.13
author: Miss Islington (bot) <[email protected]>
committer: gvanrossum <[email protected]>
date: 2024-05-10T23:15:54Z
summary:

[3.13] gh-118921: Add `copy()` method for `FrameLocalsProxy` (GH-118923) 
(#118933)

(cherry picked from commit 35c436186b849f8f2f9fb866c59015c9d034d448)

Co-authored-by: Tian Gao <[email protected]>

files:
A Misc/NEWS.d/next/Core and 
Builtins/2024-05-10-19-54-18.gh-issue-118921.O4ztZG.rst
M Lib/test/test_frame.py
M Objects/frameobject.c

diff --git a/Lib/test/test_frame.py b/Lib/test/test_frame.py
index 212255374bddd1..aee8d374b22710 100644
--- a/Lib/test/test_frame.py
+++ b/Lib/test/test_frame.py
@@ -371,6 +371,15 @@ def test_local_objects(self):
         f_locals['o'] = f_locals['k']
         self.assertEqual(o, 'a.b.c')
 
+    def test_copy(self):
+        x = 0
+        d = sys._getframe().f_locals
+        d_copy = d.copy()
+        self.assertIsInstance(d_copy, dict)
+        self.assertEqual(d_copy['x'], 0)
+        d_copy['x'] = 1
+        self.assertEqual(x, 0)
+
     def test_update_with_self(self):
         def f():
             f_locals = sys._getframe().f_locals
@@ -405,9 +414,6 @@ def test_sizeof(self):
     def test_unsupport(self):
         x = 1
         d = sys._getframe().f_locals
-        with self.assertRaises(AttributeError):
-            d.copy()
-
         with self.assertRaises(TypeError):
             copy.copy(d)
 
diff --git a/Misc/NEWS.d/next/Core and 
Builtins/2024-05-10-19-54-18.gh-issue-118921.O4ztZG.rst b/Misc/NEWS.d/next/Core 
and Builtins/2024-05-10-19-54-18.gh-issue-118921.O4ztZG.rst
new file mode 100644
index 00000000000000..39ccf472067cfd
--- /dev/null
+++ b/Misc/NEWS.d/next/Core and 
Builtins/2024-05-10-19-54-18.gh-issue-118921.O4ztZG.rst 
@@ -0,0 +1 @@
+Add ``copy()`` method for ``FrameLocalsProxy`` which returns a snapshot 
``dict`` for local variables.
diff --git a/Objects/frameobject.c b/Objects/frameobject.c
index d7fcb1925d286c..64fded85de1468 100644
--- a/Objects/frameobject.c
+++ b/Objects/frameobject.c
@@ -637,6 +637,23 @@ framelocalsproxy_setdefault(PyObject* self, PyObject 
*const *args, Py_ssize_t na
     return result;
 }
 
+static PyObject*
+framelocalsproxy_copy(PyObject *self, PyObject *Py_UNUSED(ignored))
+{
+    PyObject* result = PyDict_New();
+
+    if (result == NULL) {
+        return NULL;
+    }
+
+    if (PyDict_Update(result, self) < 0) {
+        Py_DECREF(result);
+        return NULL;
+    }
+
+    return result;
+}
+
 static PyObject*
 framelocalsproxy_reversed(PyObject *self, void *Py_UNUSED(ignored))
 {
@@ -677,6 +694,8 @@ static PyMethodDef framelocalsproxy_methods[] = {
      NULL},
     {"__reversed__", _PyCFunction_CAST(framelocalsproxy_reversed),       
METH_NOARGS,
      NULL},
+    {"copy",         _PyCFunction_CAST(framelocalsproxy_copy),           
METH_NOARGS,
+     NULL},
     {"keys",         _PyCFunction_CAST(framelocalsproxy_keys),           
METH_NOARGS,
      NULL},
     {"values",       _PyCFunction_CAST(framelocalsproxy_values),         
METH_NOARGS,

_______________________________________________
Python-checkins mailing list -- [email protected]
To unsubscribe send an email to [email protected]
https://mail.python.org/mailman3/lists/python-checkins.python.org/
Member address: [email protected]

Reply via email to