summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJeremy Hylton <jeremy@alum.mit.edu>2000-10-30 17:15:20 (GMT)
committerJeremy Hylton <jeremy@alum.mit.edu>2000-10-30 17:15:20 (GMT)
commit6b4ec5135b72eddcb7ed062bb5b466a92795f911 (patch)
tree187dccda3de56b1e2f592f40194eb9142dfafe5e
parent5942b439b366b83880a2678793d36b931acef75b (diff)
downloadcpython-6b4ec5135b72eddcb7ed062bb5b466a92795f911.zip
cpython-6b4ec5135b72eddcb7ed062bb5b466a92795f911.tar.gz
cpython-6b4ec5135b72eddcb7ed062bb5b466a92795f911.tar.bz2
Fix for SF bug #117241
When a method is called with no regular arguments and * args, defer the first arg is subclass check until after the * args have been expanded. N.B. The CALL_FUNCTION implementation is getting really hairy; should review it to see if it can be simplified.
-rw-r--r--Lib/test/test_extcall.py18
-rw-r--r--Python/ceval.c16
2 files changed, 33 insertions, 1 deletions
diff --git a/Lib/test/test_extcall.py b/Lib/test/test_extcall.py
index 10e4d6c..7dddabc 100644
--- a/Lib/test/test_extcall.py
+++ b/Lib/test/test_extcall.py
@@ -144,3 +144,21 @@ for i in range(512):
d[key] = i
a, b = f2(1, *(2, 3), **d)
print len(a), len(b), b == d
+
+class Foo:
+ def method(self, arg1, arg2):
+ return arg1 + arg2
+
+x = Foo()
+print Foo.method(*(x, 1, 2))
+print Foo.method(x, *(1, 2))
+try:
+ print Foo.method(*(1, 2, 3))
+except TypeError, err:
+ print err
+try:
+ print Foo.method(1, *(2, 3))
+except TypeError, err:
+ print err
+
+
diff --git a/Python/ceval.c b/Python/ceval.c
index df057b7..ad6e792 100644
--- a/Python/ceval.c
+++ b/Python/ceval.c
@@ -1822,7 +1822,7 @@ eval_code2(PyCodeObject *co, PyObject *globals, PyObject *locals,
na++;
n++;
}
- else {
+ else if (!((flags & 1) && na == 0)) {
/* Unbound methods must be called with an
instance of the class (or a derived
class) as first argument */
@@ -1895,6 +1895,20 @@ eval_code2(PyCodeObject *co, PyObject *globals, PyObject *locals,
if (nstar < 0) {
goto extcall_fail;
}
+ if (class && self == NULL && na == 0) {
+ /* * arg is first argument of method,
+ so check it is isinstance of class */
+ self = PyTuple_GET_ITEM(stararg, 0);
+ if (!(PyInstance_Check(self) &&
+ PyClass_IsSubclass((PyObject *)
+ (((PyInstanceObject *)self)->in_class),
+ class))) {
+ PyErr_SetString(PyExc_TypeError,
+ "unbound method must be called with instance as first argument");
+ x = NULL;
+ break;
+ }
+ }
}
if (nk > 0) {
if (kwdict == NULL) {