Commit a608e3bb authored by Jeremy Hylton's avatar Jeremy Hylton

Ignore descriptors in persistent_id() calls.

Apparent fix for the bug Sidnei reported today.  He didn't have a
simple test case, rather reported that starting Zope with Formulator
caused a crash.  I think this will fix the problem.

We had to do the same thing in Python code in ZODB4, but weren't
worrying about performance there.
parent 8fdb3f34
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
static char coptimizations_doc_string[] = static char coptimizations_doc_string[] =
"C optimization for new_persistent_id().\n" "C optimization for new_persistent_id().\n"
"\n" "\n"
"$Id: coptimizations.c,v 1.24 2003/11/28 16:44:49 jim Exp $\n"; "$Id: coptimizations.c,v 1.25 2003/12/11 05:16:50 jeremy Exp $\n";
#include "cPersistence.h" #include "cPersistence.h"
...@@ -173,9 +173,39 @@ persistent_id_call(persistent_id *self, PyObject *args, PyObject *kwargs) ...@@ -173,9 +173,39 @@ persistent_id_call(persistent_id *self, PyObject *args, PyObject *kwargs)
PyErr_Clear(); PyErr_Clear();
goto return_none; goto return_none;
} }
if (oid != Py_None) { if (oid != Py_None) {
PyObject *jar = PyObject_GetAttr(object, py__p_jar); PyObject *jar;
if (!PyString_Check(oid)) {
/* If the object is a class, then asking for _p_oid or
_p_jar will return a descriptor. There is no API to
ask whether something is a descriptor; the best you
can do is call anything with an __get__ a descriptor.
The getattr check is potentially expensive so do the
cheap PyString_Check() first, assuming that most oids
that aren't None are real oids. ZODB always uses
strings, although some other user of Persistent could
use something else.
*/
static PyObject *__get__;
PyObject *descr;
if (!__get__) {
__get__ = PyString_InternFromString("__get__");
if (!__get__)
goto err;
}
descr = PyObject_GetAttr(oid, __get__);
if (descr)
goto return_none;
/* Otherwise it's not a descriptor and it's just some
weird value. Maybe we'll get an error later.
*/
/* XXX should check that this was an AttributeError */
PyErr_Clear();
}
jar = PyObject_GetAttr(object, py__p_jar);
if (!jar) if (!jar)
PyErr_Clear(); PyErr_Clear();
else { else {
......
...@@ -15,6 +15,8 @@ ...@@ -15,6 +15,8 @@
import random import random
import unittest import unittest
from persistent import Persistent
import ZODB.coptimizations
NUM = 100 NUM = 100
...@@ -44,6 +46,13 @@ class TestUtils(unittest.TestCase): ...@@ -44,6 +46,13 @@ class TestUtils(unittest.TestCase):
self.assertEquals(u64("\000\000\000\001\000\000\000\000"), 1L<<32) self.assertEquals(u64("\000\000\000\001\000\000\000\000"), 1L<<32)
self.assertEquals(U64("\000\000\000\001\000\000\000\000"), 1L<<32) self.assertEquals(U64("\000\000\000\001\000\000\000\000"), 1L<<32)
def checkPersistentIdHandlesDescriptor(self):
class P(Persistent):
pass
L = []
pid = ZODB.coptimizations.new_persistent_id(None, L)
pid(P)
def test_suite(): def test_suite():
return unittest.makeSuite(TestUtils, 'check') return unittest.makeSuite(TestUtils, 'check')
......
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