mirror of https://github.com/python/cpython
Issue #25455: Fixed a crash in repr of recursive functools.partial objects.
This commit is contained in:
parent
cbe6142135
commit
179f960d47
|
@ -217,6 +217,24 @@ class TestPartialC(TestPartial, unittest.TestCase):
|
||||||
['{}({!r}, {}, {})'.format(name, capture, args_repr, kwargs_repr)
|
['{}({!r}, {}, {})'.format(name, capture, args_repr, kwargs_repr)
|
||||||
for kwargs_repr in kwargs_reprs])
|
for kwargs_repr in kwargs_reprs])
|
||||||
|
|
||||||
|
def test_recursive_repr(self):
|
||||||
|
if self.partial is c_functools.partial:
|
||||||
|
name = 'functools.partial'
|
||||||
|
else:
|
||||||
|
name = self.partial.__name__
|
||||||
|
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((f, (), {}, {}))
|
||||||
|
self.assertEqual(repr(f), '%s(%s(...))' % (name, name))
|
||||||
|
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((capture, (f,), {}, {}))
|
||||||
|
self.assertEqual(repr(f), '%s(%r, %s(...))' % (name, capture, name))
|
||||||
|
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((capture, (), {'a': f}, {}))
|
||||||
|
self.assertEqual(repr(f), '%s(%r, a=%s(...))' % (name, capture, name))
|
||||||
|
|
||||||
def test_pickle(self):
|
def test_pickle(self):
|
||||||
f = self.partial(signature, ['asdf'], bar=[True])
|
f = self.partial(signature, ['asdf'], bar=[True])
|
||||||
f.attr = []
|
f.attr = []
|
||||||
|
@ -297,6 +315,25 @@ class TestPartialC(TestPartial, unittest.TestCase):
|
||||||
self.assertEqual(r, ((1, 2), {}))
|
self.assertEqual(r, ((1, 2), {}))
|
||||||
self.assertIs(type(r[0]), tuple)
|
self.assertIs(type(r[0]), tuple)
|
||||||
|
|
||||||
|
def test_recursive_pickle(self):
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((f, (), {}, {}))
|
||||||
|
for proto in range(pickle.HIGHEST_PROTOCOL + 1):
|
||||||
|
with self.assertRaises(RecursionError):
|
||||||
|
pickle.dumps(f, proto)
|
||||||
|
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((capture, (f,), {}, {}))
|
||||||
|
for proto in range(pickle.HIGHEST_PROTOCOL + 1):
|
||||||
|
f_copy = pickle.loads(pickle.dumps(f, proto))
|
||||||
|
self.assertIs(f_copy.args[0], f_copy)
|
||||||
|
|
||||||
|
f = self.partial(capture)
|
||||||
|
f.__setstate__((capture, (), {'a': f}, {}))
|
||||||
|
for proto in range(pickle.HIGHEST_PROTOCOL + 1):
|
||||||
|
f_copy = pickle.loads(pickle.dumps(f, proto))
|
||||||
|
self.assertIs(f_copy.keywords['a'], f_copy)
|
||||||
|
|
||||||
# Issue 6083: Reference counting bug
|
# Issue 6083: Reference counting bug
|
||||||
def test_setstate_refcount(self):
|
def test_setstate_refcount(self):
|
||||||
class BadSequence:
|
class BadSequence:
|
||||||
|
|
|
@ -143,7 +143,8 @@ Core and Builtins
|
||||||
Library
|
Library
|
||||||
-------
|
-------
|
||||||
|
|
||||||
- Issue #25455: Fixed a crash in repr of ElementTree.Element with recursive tag.
|
- Issue #25455: Fixed crashes in repr of recursive ElementTree.Element and
|
||||||
|
functools.partial objects.
|
||||||
|
|
||||||
- Issue #26556: Update expat to 2.1.1, fixes CVE-2015-1283.
|
- Issue #26556: Update expat to 2.1.1, fixes CVE-2015-1283.
|
||||||
|
|
||||||
|
|
|
@ -203,40 +203,45 @@ static PyGetSetDef partial_getsetlist[] = {
|
||||||
static PyObject *
|
static PyObject *
|
||||||
partial_repr(partialobject *pto)
|
partial_repr(partialobject *pto)
|
||||||
{
|
{
|
||||||
PyObject *result;
|
PyObject *result = NULL;
|
||||||
PyObject *arglist;
|
PyObject *arglist;
|
||||||
PyObject *tmp;
|
|
||||||
Py_ssize_t i, n;
|
Py_ssize_t i, n;
|
||||||
PyObject *key, *value;
|
PyObject *key, *value;
|
||||||
|
int status;
|
||||||
|
|
||||||
|
status = Py_ReprEnter((PyObject *)pto);
|
||||||
|
if (status != 0) {
|
||||||
|
if (status < 0)
|
||||||
|
return NULL;
|
||||||
|
return PyUnicode_FromFormat("%s(...)", Py_TYPE(pto)->tp_name);
|
||||||
|
}
|
||||||
|
|
||||||
arglist = PyUnicode_FromString("");
|
arglist = PyUnicode_FromString("");
|
||||||
if (arglist == NULL) {
|
if (arglist == NULL)
|
||||||
return NULL;
|
goto done;
|
||||||
}
|
|
||||||
/* Pack positional arguments */
|
/* Pack positional arguments */
|
||||||
assert (PyTuple_Check(pto->args));
|
assert (PyTuple_Check(pto->args));
|
||||||
n = PyTuple_GET_SIZE(pto->args);
|
n = PyTuple_GET_SIZE(pto->args);
|
||||||
for (i = 0; i < n; i++) {
|
for (i = 0; i < n; i++) {
|
||||||
tmp = PyUnicode_FromFormat("%U, %R", arglist,
|
Py_SETREF(arglist, PyUnicode_FromFormat("%U, %R", arglist,
|
||||||
PyTuple_GET_ITEM(pto->args, i));
|
PyTuple_GET_ITEM(pto->args, i)));
|
||||||
Py_DECREF(arglist);
|
if (arglist == NULL)
|
||||||
if (tmp == NULL)
|
goto done;
|
||||||
return NULL;
|
|
||||||
arglist = tmp;
|
|
||||||
}
|
}
|
||||||
/* Pack keyword arguments */
|
/* Pack keyword arguments */
|
||||||
assert (PyDict_Check(pto->kw));
|
assert (PyDict_Check(pto->kw));
|
||||||
for (i = 0; PyDict_Next(pto->kw, &i, &key, &value);) {
|
for (i = 0; PyDict_Next(pto->kw, &i, &key, &value);) {
|
||||||
tmp = PyUnicode_FromFormat("%U, %U=%R", arglist,
|
Py_SETREF(arglist, PyUnicode_FromFormat("%U, %U=%R", arglist,
|
||||||
key, value);
|
key, value));
|
||||||
Py_DECREF(arglist);
|
if (arglist == NULL)
|
||||||
if (tmp == NULL)
|
goto done;
|
||||||
return NULL;
|
|
||||||
arglist = tmp;
|
|
||||||
}
|
}
|
||||||
result = PyUnicode_FromFormat("%s(%R%U)", Py_TYPE(pto)->tp_name,
|
result = PyUnicode_FromFormat("%s(%R%U)", Py_TYPE(pto)->tp_name,
|
||||||
pto->fn, arglist);
|
pto->fn, arglist);
|
||||||
Py_DECREF(arglist);
|
Py_DECREF(arglist);
|
||||||
|
|
||||||
|
done:
|
||||||
|
Py_ReprLeave((PyObject *)pto);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue