Commit a0d416f0 authored by Serhiy Storchaka's avatar Serhiy Storchaka

Issue #26482: Allowed pickling recursive dequeues.

parent 841b930a
...@@ -622,20 +622,22 @@ class TestBasic(unittest.TestCase): ...@@ -622,20 +622,22 @@ class TestBasic(unittest.TestCase):
self.assertEqual(list(d), list(e)) self.assertEqual(list(d), list(e))
def test_pickle(self): def test_pickle(self):
d = deque(range(200)) for d in deque(range(200)), deque(range(200), 100):
for i in range(pickle.HIGHEST_PROTOCOL + 1): for i in range(pickle.HIGHEST_PROTOCOL + 1):
s = pickle.dumps(d, i) s = pickle.dumps(d, i)
e = pickle.loads(s) e = pickle.loads(s)
self.assertNotEqual(id(d), id(e)) self.assertNotEqual(id(e), id(d))
self.assertEqual(list(d), list(e)) self.assertEqual(list(e), list(d))
self.assertEqual(e.maxlen, d.maxlen)
## def test_pickle_recursive(self): def test_pickle_recursive(self):
## d = deque('abc') for d in deque('abc'), deque('abc', 3):
## d.append(d) d.append(d)
## for i in range(pickle.HIGHEST_PROTOCOL + 1): for i in range(pickle.HIGHEST_PROTOCOL + 1):
## e = pickle.loads(pickle.dumps(d, i)) e = pickle.loads(pickle.dumps(d, i))
## self.assertNotEqual(id(d), id(e)) self.assertNotEqual(id(e), id(d))
## self.assertEqual(id(e), id(e[-1])) self.assertEqual(id(e[-1]), id(e))
self.assertEqual(e.maxlen, d.maxlen)
def test_iterator_pickle(self): def test_iterator_pickle(self):
data = deque(range(200)) data = deque(range(200))
...@@ -827,24 +829,26 @@ class TestSubclass(unittest.TestCase): ...@@ -827,24 +829,26 @@ class TestSubclass(unittest.TestCase):
self.assertEqual(type(d), type(e)) self.assertEqual(type(d), type(e))
self.assertEqual(list(d), list(e)) self.assertEqual(list(d), list(e))
## def test_pickle(self): def test_pickle_recursive(self):
## d = Deque('abc') for proto in range(pickle.HIGHEST_PROTOCOL + 1):
## d.append(d) for d in Deque('abc'), Deque('abc', 3):
## d.append(d)
## e = pickle.loads(pickle.dumps(d))
## self.assertNotEqual(id(d), id(e)) e = pickle.loads(pickle.dumps(d, proto))
## self.assertEqual(type(d), type(e)) self.assertNotEqual(id(e), id(d))
## dd = d.pop() self.assertEqual(type(e), type(d))
## ee = e.pop() self.assertEqual(e.maxlen, d.maxlen)
## self.assertEqual(id(e), id(ee)) dd = d.pop()
## self.assertEqual(d, e) ee = e.pop()
## self.assertEqual(id(ee), id(e))
## d.x = d self.assertEqual(e, d)
## e = pickle.loads(pickle.dumps(d))
## self.assertEqual(id(e), id(e.x)) d.x = d
## e = pickle.loads(pickle.dumps(d, proto))
## d = DequeWithBadIter('abc') self.assertEqual(id(e.x), id(e))
## self.assertRaises(TypeError, pickle.dumps, d)
for d in DequeWithBadIter('abc'), DequeWithBadIter('abc', 2):
self.assertRaises(TypeError, pickle.dumps, d, proto)
def test_weakref(self): def test_weakref(self):
d = deque('gallahad') d = deque('gallahad')
......
...@@ -201,6 +201,8 @@ Core and Builtins ...@@ -201,6 +201,8 @@ Core and Builtins
Library Library
------- -------
- Issue #26482: Allowed pickling recursive dequeues.
- Issue #26335: Make mmap.write() return the number of bytes written like - Issue #26335: Make mmap.write() return the number of bytes written like
other write methods. Patch by Jakub Stasiak. other write methods. Patch by Jakub Stasiak.
......
...@@ -1296,31 +1296,31 @@ deque_traverse(dequeobject *deque, visitproc visit, void *arg) ...@@ -1296,31 +1296,31 @@ deque_traverse(dequeobject *deque, visitproc visit, void *arg)
static PyObject * static PyObject *
deque_reduce(dequeobject *deque) deque_reduce(dequeobject *deque)
{ {
PyObject *dict, *result, *aslist; PyObject *dict, *it;
_Py_IDENTIFIER(__dict__); _Py_IDENTIFIER(__dict__);
dict = _PyObject_GetAttrId((PyObject *)deque, &PyId___dict__); dict = _PyObject_GetAttrId((PyObject *)deque, &PyId___dict__);
if (dict == NULL) if (dict == NULL) {
if (!PyErr_ExceptionMatches(PyExc_AttributeError)) {
return NULL;
}
PyErr_Clear(); PyErr_Clear();
aslist = PySequence_List((PyObject *)deque); dict = Py_None;
if (aslist == NULL) { Py_INCREF(dict);
Py_XDECREF(dict); }
it = PyObject_GetIter((PyObject *)deque);
if (it == NULL) {
Py_DECREF(dict);
return NULL; return NULL;
} }
if (dict == NULL) {
if (deque->maxlen < 0) if (deque->maxlen < 0) {
result = Py_BuildValue("O(O)", Py_TYPE(deque), aslist); return Py_BuildValue("O()NN", Py_TYPE(deque), dict, it);
else }
result = Py_BuildValue("O(On)", Py_TYPE(deque), aslist, deque->maxlen); else {
} else { return Py_BuildValue("O(()n)NN", Py_TYPE(deque), deque->maxlen, dict, it);
if (deque->maxlen < 0)
result = Py_BuildValue("O(OO)O", Py_TYPE(deque), aslist, Py_None, dict);
else
result = Py_BuildValue("O(On)O", Py_TYPE(deque), aslist, deque->maxlen, dict);
} }
Py_XDECREF(dict);
Py_DECREF(aslist);
return result;
} }
PyDoc_STRVAR(reduce_doc, "Return state information for pickling."); PyDoc_STRVAR(reduce_doc, "Return state information for pickling.");
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment