mirror of
https://github.com/python/cpython.git
synced 2025-10-08 16:11:51 +00:00
Merged revisions 72461 via svnmerge from
svn+ssh://pythondev@svn.python.org/python/trunk ........ r72461 | benjamin.peterson | 2009-05-07 22:06:00 -0500 (Thu, 07 May 2009) | 1 line add _PyObject_LookupSpecial to handle fetching special method lookup ........
This commit is contained in:
parent
c04dad772c
commit
224205fde2
4 changed files with 63 additions and 9 deletions
|
@ -414,6 +414,7 @@ PyAPI_FUNC(PyObject *) PyType_GenericAlloc(PyTypeObject *, Py_ssize_t);
|
||||||
PyAPI_FUNC(PyObject *) PyType_GenericNew(PyTypeObject *,
|
PyAPI_FUNC(PyObject *) PyType_GenericNew(PyTypeObject *,
|
||||||
PyObject *, PyObject *);
|
PyObject *, PyObject *);
|
||||||
PyAPI_FUNC(PyObject *) _PyType_Lookup(PyTypeObject *, PyObject *);
|
PyAPI_FUNC(PyObject *) _PyType_Lookup(PyTypeObject *, PyObject *);
|
||||||
|
PyAPI_FUNC(PyObject *) _PyObject_LookupSpecial(PyObject *, char *, PyObject **);
|
||||||
PyAPI_FUNC(unsigned int) PyType_ClearCache(void);
|
PyAPI_FUNC(unsigned int) PyType_ClearCache(void);
|
||||||
PyAPI_FUNC(void) PyType_Modified(PyTypeObject *);
|
PyAPI_FUNC(void) PyType_Modified(PyTypeObject *);
|
||||||
|
|
||||||
|
|
|
@ -1538,6 +1538,58 @@ order (MRO) for bases """
|
||||||
self.assertEqual(E().foo.__func__, C.foo) # i.e., unbound
|
self.assertEqual(E().foo.__func__, C.foo) # i.e., unbound
|
||||||
self.assert_(repr(C.foo.__get__(C(1))).startswith("<bound method "))
|
self.assert_(repr(C.foo.__get__(C(1))).startswith("<bound method "))
|
||||||
|
|
||||||
|
def test_special_method_lookup(self):
|
||||||
|
# The lookup of special methods bypasses __getattr__ and
|
||||||
|
# __getattribute__, but they still can be descriptors.
|
||||||
|
|
||||||
|
def run_context(manager):
|
||||||
|
with manager:
|
||||||
|
pass
|
||||||
|
def iden(self):
|
||||||
|
return self
|
||||||
|
def hello(self):
|
||||||
|
return b"hello"
|
||||||
|
|
||||||
|
# It would be nice to have every special method tested here, but I'm
|
||||||
|
# only listing the ones I can remember outside of typeobject.c, since it
|
||||||
|
# does it right.
|
||||||
|
specials = [
|
||||||
|
("__bytes__", bytes, hello),
|
||||||
|
# These two fail because the compiler generates LOAD_ATTR to look
|
||||||
|
# them up. We'd have to add a new opcode to fix this, and it's
|
||||||
|
# probably not worth it.
|
||||||
|
# ("__enter__", run_context, iden),
|
||||||
|
# ("__exit__", run_context, iden),
|
||||||
|
]
|
||||||
|
|
||||||
|
class Checker(object):
|
||||||
|
def __getattr__(self, attr, test=self):
|
||||||
|
test.fail("__getattr__ called with {0}".format(attr))
|
||||||
|
def __getattribute__(self, attr, test=self):
|
||||||
|
test.fail("__getattribute__ called with {0}".format(attr))
|
||||||
|
class SpecialDescr(object):
|
||||||
|
def __init__(self, impl):
|
||||||
|
self.impl = impl
|
||||||
|
def __get__(self, obj, owner):
|
||||||
|
record.append(1)
|
||||||
|
return self
|
||||||
|
def __call__(self, *args):
|
||||||
|
return self.impl(*args)
|
||||||
|
|
||||||
|
|
||||||
|
for name, runner, meth_impl in specials:
|
||||||
|
class X(Checker):
|
||||||
|
pass
|
||||||
|
setattr(X, name, staticmethod(meth_impl))
|
||||||
|
runner(X())
|
||||||
|
|
||||||
|
record = []
|
||||||
|
class X(Checker):
|
||||||
|
pass
|
||||||
|
setattr(X, name, SpecialDescr(meth_impl))
|
||||||
|
runner(X())
|
||||||
|
self.assertEqual(record, [1], name)
|
||||||
|
|
||||||
def test_specials(self):
|
def test_specials(self):
|
||||||
# Testing special operators...
|
# Testing special operators...
|
||||||
# Test operators like __hash__ for which a built-in default exists
|
# Test operators like __hash__ for which a built-in default exists
|
||||||
|
|
|
@ -474,12 +474,6 @@ PyObject_Bytes(PyObject *v)
|
||||||
PyObject *result, *func;
|
PyObject *result, *func;
|
||||||
static PyObject *bytesstring = NULL;
|
static PyObject *bytesstring = NULL;
|
||||||
|
|
||||||
if (bytesstring == NULL) {
|
|
||||||
bytesstring = PyUnicode_InternFromString("__bytes__");
|
|
||||||
if (bytesstring == NULL)
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (v == NULL)
|
if (v == NULL)
|
||||||
return PyBytes_FromString("<NULL>");
|
return PyBytes_FromString("<NULL>");
|
||||||
|
|
||||||
|
@ -488,10 +482,10 @@ PyObject_Bytes(PyObject *v)
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Doesn't create a reference */
|
func = _PyObject_LookupSpecial(v, "__bytes__", &bytesstring);
|
||||||
func = _PyType_Lookup(Py_TYPE(v), bytesstring);
|
|
||||||
if (func != NULL) {
|
if (func != NULL) {
|
||||||
result = PyObject_CallFunctionObjArgs(func, v, NULL);
|
result = PyObject_CallFunctionObjArgs(func, v, NULL);
|
||||||
|
Py_DECREF(func);
|
||||||
if (result == NULL)
|
if (result == NULL)
|
||||||
return NULL;
|
return NULL;
|
||||||
if (!PyBytes_Check(result)) {
|
if (!PyBytes_Check(result)) {
|
||||||
|
@ -503,7 +497,6 @@ PyObject_Bytes(PyObject *v)
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
PyErr_Clear();
|
|
||||||
return PyBytes_FromObject(v);
|
return PyBytes_FromObject(v);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1125,6 +1125,8 @@ PyType_IsSubtype(PyTypeObject *a, PyTypeObject *b)
|
||||||
when the _PyType_Lookup() call fails;
|
when the _PyType_Lookup() call fails;
|
||||||
|
|
||||||
- lookup_method() always raises an exception upon errors.
|
- lookup_method() always raises an exception upon errors.
|
||||||
|
|
||||||
|
- _PyObject_LookupSpecial() exported for the benefit of other places.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
|
@ -1157,6 +1159,12 @@ lookup_method(PyObject *self, char *attrstr, PyObject **attrobj)
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
PyObject *
|
||||||
|
_PyObject_LookupSpecial(PyObject *self, char *attrstr, PyObject **attrobj)
|
||||||
|
{
|
||||||
|
return lookup_maybe(self, attrstr, attrobj);
|
||||||
|
}
|
||||||
|
|
||||||
/* A variation of PyObject_CallMethod that uses lookup_method()
|
/* A variation of PyObject_CallMethod that uses lookup_method()
|
||||||
instead of PyObject_GetAttrString(). This uses the same convention
|
instead of PyObject_GetAttrString(). This uses the same convention
|
||||||
as lookup_method to cache the interned name string object. */
|
as lookup_method to cache the interned name string object. */
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue