Skip to content

Commit 936bd14

Browse files
committed
gh-106572: Deprecate PyObject_SetAttr(v, name, NULL)
If the alue is NULL, PyObject_SetAttr() and PyObject_SetAttrString() emit a DeprecationWarning in Python Development Mode or if Python is built in debug mode. weakref proxy_setattr() calls PyObject_DelAttr() if value is NULL.
1 parent 3590c45 commit 936bd14

File tree

7 files changed

+118
-13
lines changed

7 files changed

+118
-13
lines changed

Doc/c-api/object.rst

+14-5
Original file line numberDiff line numberDiff line change
@@ -122,9 +122,13 @@ Object Protocol
122122
return ``0`` on success. This is the equivalent of the Python statement
123123
``o.attr_name = v``.
124124
125-
If *v* is ``NULL``, the attribute is deleted. This behaviour is deprecated
126-
in favour of using :c:func:`PyObject_DelAttr`, but there are currently no
127-
plans to remove it.
125+
If *v* is ``NULL``, emit a :exc:`DeprecationWarning` in :ref:`Python
126+
Development Mode <devmode>` or if :ref:`Python is built in debug mode
127+
<debug-build>`.
128+
129+
.. deprecated:: 3.13
130+
Calling ``PyObject_SetAttrString(o, attr_name, NULL)`` is deprecated:
131+
``PyObject_DelAttr(o, attr_name)`` must be used instead.
128132
129133
130134
.. c:function:: int PyObject_SetAttrString(PyObject *o, const char *attr_name, PyObject *v)
@@ -134,8 +138,13 @@ Object Protocol
134138
return ``0`` on success. This is the equivalent of the Python statement
135139
``o.attr_name = v``.
136140
137-
If *v* is ``NULL``, the attribute is deleted, but this feature is
138-
deprecated in favour of using :c:func:`PyObject_DelAttrString`.
141+
If *v* is ``NULL``, emit a :exc:`DeprecationWarning` in :ref:`Python
142+
Development Mode <devmode>` or if :ref:`Python is built in debug mode
143+
<debug-build>`.
144+
145+
.. deprecated:: 3.13
146+
Calling ``PyObject_SetAttrString(o, attr_name, NULL)`` is deprecated:
147+
``PyObject_DelAttrString(o, attr_name)`` must be used instead.
139148
140149
141150
.. c:function:: int PyObject_GenericSetAttr(PyObject *o, PyObject *name, PyObject *value)

Doc/whatsnew/3.13.rst

+11
Original file line numberDiff line numberDiff line change
@@ -795,6 +795,17 @@ Deprecated
795795
:c:func:`PyWeakref_GetRef` on Python 3.12 and older.
796796
(Contributed by Victor Stinner in :gh:`105927`.)
797797

798+
* Deprecate :c:func:`PyObject_SetAttr` and :c:func:`PyObject_SetAttrString` to
799+
remove an attribute (if *value* is ``NULL``): ``PyObject_DelAttr(v, name)``
800+
and ``PyObject_DelAttrString(v, name)`` must be used instead. If the
801+
development mode is enabled or if Python is built in debug mode, these
802+
deprecated functions now emit a :exc:`DeprecationWarning` if *value* is
803+
``NULL``. When :c:func:`PyObject_SetAttr` is called with a ``NULL`` value,
804+
it's unclear if the caller wants to remove the attribute on purpose, or if
805+
the value is ``NULL`` because of an error. Such API is error prone and should
806+
be avoided.
807+
(Contributed by Victor Stinner in :gh:`106572`.)
808+
798809
Removed
799810
-------
800811

Lib/test/test_capi/test_misc.py

+18
Original file line numberDiff line numberDiff line change
@@ -1115,6 +1115,24 @@ class Data(_testcapi.ObjExtraData):
11151115
del d.extra
11161116
self.assertIsNone(d.extra)
11171117

1118+
def test_setattr(self):
1119+
class MyType:
1120+
pass
1121+
1122+
PyObject_SetAttr = _testcapi.PyObject_SetAttr
1123+
1124+
obj = MyType()
1125+
PyObject_SetAttr(obj, "attr", 123)
1126+
self.assertEqual(obj.attr, 123)
1127+
# PyObject_SetAttr(obj, name, NULL) emits a DeprecationWarning
1128+
# in the Python Development Mode or if Python is built in debug mode
1129+
if support.Py_DEBUG or sys.flags.dev_mode:
1130+
with self.assertWarns(DeprecationWarning):
1131+
PyObject_SetAttr(obj, "attr")
1132+
else:
1133+
PyObject_SetAttr(obj, "attr")
1134+
self.assertFalse(hasattr(obj, "attr"))
1135+
11181136

11191137
@requires_limited_api
11201138
class TestHeapTypeRelative(unittest.TestCase):
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
Deprecate :c:func:`PyObject_SetAttr` and :c:func:`PyObject_SetAttrString` to
2+
remove an attribute (if *value* is ``NULL``): ``PyObject_DelAttr(v, name)`` and
3+
``PyObject_DelAttrString(v, name)`` must be used instead. If the development
4+
mode is enabled or if Python is built in debug mode, these deprecated functions
5+
now emit a :exc:`DeprecationWarning` if *value* is ``NULL``. When
6+
:c:func:`PyObject_SetAttr` is called with a ``NULL`` value, it's unclear if the
7+
caller wants to remove the attribute on purpose, or if the value is ``NULL``
8+
because of an error. Such API is error prone and should be avoided.
9+
Patch by Victor Stinner.

Modules/_testcapimodule.c

+15
Original file line numberDiff line numberDiff line change
@@ -3464,6 +3464,20 @@ test_weakref_capi(PyObject *Py_UNUSED(module), PyObject *Py_UNUSED(args))
34643464
}
34653465

34663466

3467+
static PyObject *
3468+
test_pyobject_setattr(PyObject *Py_UNUSED(module), PyObject *args)
3469+
{
3470+
PyObject *obj, *name, *value = NULL;
3471+
if (!PyArg_ParseTuple(args, "OO|O", &obj, &name, &value)) {
3472+
return NULL;
3473+
}
3474+
if (PyObject_SetAttr(obj, name, value) < 0) {
3475+
return NULL;
3476+
}
3477+
Py_RETURN_NONE;
3478+
}
3479+
3480+
34673481
static PyMethodDef TestMethods[] = {
34683482
{"set_errno", set_errno, METH_VARARGS},
34693483
{"test_config", test_config, METH_NOARGS},
@@ -3609,6 +3623,7 @@ static PyMethodDef TestMethods[] = {
36093623
{"function_set_kw_defaults", function_set_kw_defaults, METH_VARARGS, NULL},
36103624
{"check_pyimport_addmodule", check_pyimport_addmodule, METH_VARARGS},
36113625
{"test_weakref_capi", test_weakref_capi, METH_NOARGS},
3626+
{"PyObject_SetAttr", test_pyobject_setattr, METH_VARARGS},
36123627
{NULL, NULL} /* sentinel */
36133628
};
36143629

Objects/object.c

+44-7
Original file line numberDiff line numberDiff line change
@@ -926,8 +926,11 @@ PyObject_HasAttrString(PyObject *v, const char *name)
926926
return ok;
927927
}
928928

929-
int
930-
PyObject_SetAttrString(PyObject *v, const char *name, PyObject *w)
929+
// Forward declaration
930+
static int object_set_attr(PyObject *v, PyObject *name, PyObject *value);
931+
932+
static int
933+
object_set_attr_string(PyObject *v, const char *name, PyObject *w)
931934
{
932935
PyObject *s;
933936
int res;
@@ -937,15 +940,32 @@ PyObject_SetAttrString(PyObject *v, const char *name, PyObject *w)
937940
s = PyUnicode_InternFromString(name);
938941
if (s == NULL)
939942
return -1;
940-
res = PyObject_SetAttr(v, s, w);
943+
res = object_set_attr(v, s, w);
941944
Py_XDECREF(s);
942945
return res;
943946
}
944947

948+
int
949+
PyObject_SetAttrString(PyObject *v, const char *name, PyObject *w)
950+
{
951+
if (w == NULL
952+
#ifndef Py_DEBUG
953+
&& _Py_GetConfig()->dev_mode
954+
#endif
955+
&& PyErr_WarnEx(PyExc_DeprecationWarning,
956+
"PyObject_SetAttrString(v, name, NULL) is deprecated: "
957+
"use PyObject_DelAttrString(v, name) instead",
958+
1) < 0)
959+
{
960+
return -1;
961+
}
962+
return object_set_attr_string(v, name, w);
963+
}
964+
945965
int
946966
PyObject_DelAttrString(PyObject *v, const char *name)
947967
{
948-
return PyObject_SetAttrString(v, name, NULL);
968+
return object_set_attr_string(v, name, NULL);
949969
}
950970

951971
int
@@ -1156,8 +1176,8 @@ PyObject_HasAttr(PyObject *v, PyObject *name)
11561176
return 1;
11571177
}
11581178

1159-
int
1160-
PyObject_SetAttr(PyObject *v, PyObject *name, PyObject *value)
1179+
static int
1180+
object_set_attr(PyObject *v, PyObject *name, PyObject *value)
11611181
{
11621182
PyTypeObject *tp = Py_TYPE(v);
11631183
int err;
@@ -1205,10 +1225,27 @@ PyObject_SetAttr(PyObject *v, PyObject *name, PyObject *value)
12051225
return -1;
12061226
}
12071227

1228+
int
1229+
PyObject_SetAttr(PyObject *v, PyObject *name, PyObject *value)
1230+
{
1231+
if (value == NULL
1232+
#ifndef Py_DEBUG
1233+
&& _Py_GetConfig()->dev_mode
1234+
#endif
1235+
&& PyErr_WarnEx(PyExc_DeprecationWarning,
1236+
"PyObject_SetAttr(v, name, NULL) is deprecated: "
1237+
"use PyObject_DelAttr(v, name) instead",
1238+
1) < 0)
1239+
{
1240+
return -1;
1241+
}
1242+
return object_set_attr(v, name, value);
1243+
}
1244+
12081245
int
12091246
PyObject_DelAttr(PyObject *v, PyObject *name)
12101247
{
1211-
return PyObject_SetAttr(v, name, NULL);
1248+
return object_set_attr(v, name, NULL);
12121249
}
12131250

12141251
PyObject **

Objects/weakrefobject.c

+7-1
Original file line numberDiff line numberDiff line change
@@ -485,7 +485,13 @@ proxy_setattr(PyObject *proxy, PyObject *name, PyObject *value)
485485
if (!proxy_check_ref(obj)) {
486486
return -1;
487487
}
488-
int res = PyObject_SetAttr(obj, name, value);
488+
int res;
489+
if (value != NULL) {
490+
res = PyObject_SetAttr(obj, name, value);
491+
}
492+
else {
493+
res = PyObject_DelAttr(obj, name);
494+
}
489495
Py_DECREF(obj);
490496
return res;
491497
}

0 commit comments

Comments
 (0)