blob: 2c1848337b2d7f47dda7d50308b65cc7742fa0f1 [file] [log] [blame]
Benjamin Petersona567a772010-03-21 21:00:50 +00001
Christian Heimesb186d002008-03-18 15:15:01 +00002"""Doctest for method/function calls.
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +00003
Christian Heimesb186d002008-03-18 15:15:01 +00004We're going the use these types for extra testing
Raymond Hettinger40174c32003-05-31 07:04:16 +00005
Neal Norwitzd79bacc2008-03-18 20:58:39 +00006 >>> from collections import UserList
7 >>> from collections import UserDict
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +00008
Christian Heimesb186d002008-03-18 15:15:01 +00009We're defining four helper functions
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +000010
Christian Heimesb186d002008-03-18 15:15:01 +000011 >>> def e(a,b):
Neal Norwitzd79bacc2008-03-18 20:58:39 +000012 ... print(a, b)
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +000013
Christian Heimesb186d002008-03-18 15:15:01 +000014 >>> def f(*a, **k):
Benjamin Petersonee8712c2008-05-20 21:35:26 +000015 ... print(a, support.sortdict(k))
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +000016
Christian Heimesb186d002008-03-18 15:15:01 +000017 >>> def g(x, *y, **z):
Benjamin Petersonee8712c2008-05-20 21:35:26 +000018 ... print(x, y, support.sortdict(z))
Christian Heimesb186d002008-03-18 15:15:01 +000019
20 >>> def h(j=1, a=2, h=3):
Neal Norwitzd79bacc2008-03-18 20:58:39 +000021 ... print(j, a, h)
Christian Heimesb186d002008-03-18 15:15:01 +000022
23Argument list examples
24
25 >>> f()
26 () {}
27 >>> f(1)
28 (1,) {}
29 >>> f(1, 2)
30 (1, 2) {}
31 >>> f(1, 2, 3)
32 (1, 2, 3) {}
33 >>> f(1, 2, 3, *(4, 5))
34 (1, 2, 3, 4, 5) {}
35 >>> f(1, 2, 3, *[4, 5])
36 (1, 2, 3, 4, 5) {}
Benjamin Peterson025e9eb2015-05-05 20:16:41 -040037 >>> f(*[1, 2, 3], 4, 5)
38 (1, 2, 3, 4, 5) {}
Christian Heimesb186d002008-03-18 15:15:01 +000039 >>> f(1, 2, 3, *UserList([4, 5]))
40 (1, 2, 3, 4, 5) {}
Benjamin Peterson025e9eb2015-05-05 20:16:41 -040041 >>> f(1, 2, 3, *[4, 5], *[6, 7])
42 (1, 2, 3, 4, 5, 6, 7) {}
43 >>> f(1, *[2, 3], 4, *[5, 6], 7)
44 (1, 2, 3, 4, 5, 6, 7) {}
45 >>> f(*UserList([1, 2]), *UserList([3, 4]), 5, *UserList([6, 7]))
46 (1, 2, 3, 4, 5, 6, 7) {}
Christian Heimesb186d002008-03-18 15:15:01 +000047
48Here we add keyword arguments
49
50 >>> f(1, 2, 3, **{'a':4, 'b':5})
51 (1, 2, 3) {'a': 4, 'b': 5}
Benjamin Peterson025e9eb2015-05-05 20:16:41 -040052 >>> f(1, 2, **{'a': -1, 'b': 5}, **{'a': 4, 'c': 6})
53 Traceback (most recent call last):
54 ...
55 TypeError: f() got multiple values for keyword argument 'a'
56 >>> f(1, 2, **{'a': -1, 'b': 5}, a=4, c=6)
57 Traceback (most recent call last):
58 ...
59 TypeError: f() got multiple values for keyword argument 'a'
Serhiy Storchaka3c317e72016-06-12 09:22:01 +030060 >>> f(1, 2, a=3, **{'a': 4}, **{'a': 5})
61 Traceback (most recent call last):
62 ...
63 TypeError: f() got multiple values for keyword argument 'a'
Christian Heimesb186d002008-03-18 15:15:01 +000064 >>> f(1, 2, 3, *[4, 5], **{'a':6, 'b':7})
65 (1, 2, 3, 4, 5) {'a': 6, 'b': 7}
66 >>> f(1, 2, 3, x=4, y=5, *(6, 7), **{'a':8, 'b': 9})
67 (1, 2, 3, 6, 7) {'a': 8, 'b': 9, 'x': 4, 'y': 5}
Benjamin Peterson025e9eb2015-05-05 20:16:41 -040068 >>> f(1, 2, 3, *[4, 5], **{'c': 8}, **{'a':6, 'b':7})
69 (1, 2, 3, 4, 5) {'a': 6, 'b': 7, 'c': 8}
70 >>> f(1, 2, 3, *(4, 5), x=6, y=7, **{'a':8, 'b': 9})
71 (1, 2, 3, 4, 5) {'a': 8, 'b': 9, 'x': 6, 'y': 7}
Christian Heimesb186d002008-03-18 15:15:01 +000072
73 >>> f(1, 2, 3, **UserDict(a=4, b=5))
74 (1, 2, 3) {'a': 4, 'b': 5}
75 >>> f(1, 2, 3, *(4, 5), **UserDict(a=6, b=7))
76 (1, 2, 3, 4, 5) {'a': 6, 'b': 7}
77 >>> f(1, 2, 3, x=4, y=5, *(6, 7), **UserDict(a=8, b=9))
78 (1, 2, 3, 6, 7) {'a': 8, 'b': 9, 'x': 4, 'y': 5}
Benjamin Peterson025e9eb2015-05-05 20:16:41 -040079 >>> f(1, 2, 3, *(4, 5), x=6, y=7, **UserDict(a=8, b=9))
80 (1, 2, 3, 4, 5) {'a': 8, 'b': 9, 'x': 6, 'y': 7}
Christian Heimesb186d002008-03-18 15:15:01 +000081
82Examples with invalid arguments (TypeErrors). We're also testing the function
83names in the exception messages.
84
85Verify clearing of SF bug #733667
86
87 >>> e(c=4)
88 Traceback (most recent call last):
89 ...
90 TypeError: e() got an unexpected keyword argument 'c'
91
92 >>> g()
93 Traceback (most recent call last):
94 ...
Benjamin Petersone109c702011-06-24 09:37:26 -050095 TypeError: g() missing 1 required positional argument: 'x'
Christian Heimesb186d002008-03-18 15:15:01 +000096
97 >>> g(*())
98 Traceback (most recent call last):
99 ...
Benjamin Petersone109c702011-06-24 09:37:26 -0500100 TypeError: g() missing 1 required positional argument: 'x'
Christian Heimesb186d002008-03-18 15:15:01 +0000101
102 >>> g(*(), **{})
103 Traceback (most recent call last):
104 ...
Benjamin Petersone109c702011-06-24 09:37:26 -0500105 TypeError: g() missing 1 required positional argument: 'x'
Christian Heimesb186d002008-03-18 15:15:01 +0000106
107 >>> g(1)
108 1 () {}
109 >>> g(1, 2)
110 1 (2,) {}
111 >>> g(1, 2, 3)
112 1 (2, 3) {}
113 >>> g(1, 2, 3, *(4, 5))
114 1 (2, 3, 4, 5) {}
115
116 >>> class Nothing: pass
117 ...
118 >>> g(*Nothing())
119 Traceback (most recent call last):
120 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300121 TypeError: g() argument after * must be an iterable, not Nothing
Christian Heimesb186d002008-03-18 15:15:01 +0000122
123 >>> class Nothing:
124 ... def __len__(self): return 5
125 ...
126
127 >>> g(*Nothing())
128 Traceback (most recent call last):
129 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300130 TypeError: g() argument after * must be an iterable, not Nothing
Christian Heimesb186d002008-03-18 15:15:01 +0000131
132 >>> class Nothing():
133 ... def __len__(self): return 5
134 ... def __getitem__(self, i):
135 ... if i<3: return i
136 ... else: raise IndexError(i)
137 ...
138
139 >>> g(*Nothing())
140 0 (1, 2) {}
141
142 >>> class Nothing:
143 ... def __init__(self): self.c = 0
144 ... def __iter__(self): return self
Neal Norwitzd79bacc2008-03-18 20:58:39 +0000145 ... def __next__(self):
Christian Heimesb186d002008-03-18 15:15:01 +0000146 ... if self.c == 4:
147 ... raise StopIteration
148 ... c = self.c
149 ... self.c += 1
150 ... return c
151 ...
152
153 >>> g(*Nothing())
154 0 (1, 2, 3) {}
155
Martin Panterb5944222016-01-31 06:30:56 +0000156Check for issue #4806: Does a TypeError in a generator get propagated with the
157right error message? (Also check with other iterables.)
158
159 >>> def broken(): raise TypeError("myerror")
160 ...
161
162 >>> g(*(broken() for i in range(1)))
163 Traceback (most recent call last):
164 ...
165 TypeError: myerror
Serhiy Storchaka25e4f772017-08-03 11:37:15 +0300166 >>> g(*range(1), *(broken() for i in range(1)))
167 Traceback (most recent call last):
168 ...
169 TypeError: myerror
Martin Panterb5944222016-01-31 06:30:56 +0000170
171 >>> class BrokenIterable1:
172 ... def __iter__(self):
173 ... raise TypeError('myerror')
174 ...
175 >>> g(*BrokenIterable1())
176 Traceback (most recent call last):
177 ...
178 TypeError: myerror
Serhiy Storchaka25e4f772017-08-03 11:37:15 +0300179 >>> g(*range(1), *BrokenIterable1())
180 Traceback (most recent call last):
181 ...
182 TypeError: myerror
Martin Panterb5944222016-01-31 06:30:56 +0000183
184 >>> class BrokenIterable2:
185 ... def __iter__(self):
186 ... yield 0
187 ... raise TypeError('myerror')
188 ...
189 >>> g(*BrokenIterable2())
190 Traceback (most recent call last):
191 ...
192 TypeError: myerror
Serhiy Storchaka25e4f772017-08-03 11:37:15 +0300193 >>> g(*range(1), *BrokenIterable2())
194 Traceback (most recent call last):
195 ...
196 TypeError: myerror
Martin Panterb5944222016-01-31 06:30:56 +0000197
198 >>> class BrokenSequence:
199 ... def __getitem__(self, idx):
200 ... raise TypeError('myerror')
201 ...
202 >>> g(*BrokenSequence())
203 Traceback (most recent call last):
204 ...
205 TypeError: myerror
Serhiy Storchaka25e4f772017-08-03 11:37:15 +0300206 >>> g(*range(1), *BrokenSequence())
207 Traceback (most recent call last):
208 ...
209 TypeError: myerror
Martin Panterb5944222016-01-31 06:30:56 +0000210
Christian Heimesb186d002008-03-18 15:15:01 +0000211Make sure that the function doesn't stomp the dictionary
212
213 >>> d = {'a': 1, 'b': 2, 'c': 3}
214 >>> d2 = d.copy()
215 >>> g(1, d=4, **d)
216 1 () {'a': 1, 'b': 2, 'c': 3, 'd': 4}
217 >>> d == d2
218 True
219
220What about willful misconduct?
221
222 >>> def saboteur(**kw):
223 ... kw['x'] = 'm'
224 ... return kw
225
226 >>> d = {}
227 >>> kw = saboteur(a=1, **d)
228 >>> d
229 {}
Jeremy Hyltonaed0d8d2000-03-28 23:51:17 +0000230
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000231
Christian Heimesb186d002008-03-18 15:15:01 +0000232 >>> g(1, 2, 3, **{'x': 4, 'y': 5})
233 Traceback (most recent call last):
234 ...
Benjamin Petersonb204a422011-06-05 22:04:07 -0500235 TypeError: g() got multiple values for argument 'x'
Christian Heimesb186d002008-03-18 15:15:01 +0000236
237 >>> f(**{1:2})
238 Traceback (most recent call last):
239 ...
240 TypeError: f() keywords must be strings
241
242 >>> h(**{'e': 2})
243 Traceback (most recent call last):
244 ...
245 TypeError: h() got an unexpected keyword argument 'e'
246
247 >>> h(*h)
248 Traceback (most recent call last):
249 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300250 TypeError: h() argument after * must be an iterable, not function
Christian Heimesb186d002008-03-18 15:15:01 +0000251
Serhiy Storchaka73442852016-10-02 10:33:46 +0300252 >>> h(1, *h)
253 Traceback (most recent call last):
254 ...
255 TypeError: h() argument after * must be an iterable, not function
256
257 >>> h(*[1], *h)
258 Traceback (most recent call last):
259 ...
260 TypeError: h() argument after * must be an iterable, not function
261
Christian Heimesb186d002008-03-18 15:15:01 +0000262 >>> dir(*h)
263 Traceback (most recent call last):
264 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300265 TypeError: dir() argument after * must be an iterable, not function
Christian Heimesb186d002008-03-18 15:15:01 +0000266
267 >>> None(*h)
268 Traceback (most recent call last):
269 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300270 TypeError: NoneType object argument after * must be an iterable, \
271not function
Christian Heimesb186d002008-03-18 15:15:01 +0000272
273 >>> h(**h)
274 Traceback (most recent call last):
275 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300276 TypeError: h() argument after ** must be a mapping, not function
Christian Heimesb186d002008-03-18 15:15:01 +0000277
Serhiy Storchakae036ef82016-10-02 11:06:43 +0300278 >>> h(**[])
279 Traceback (most recent call last):
280 ...
281 TypeError: h() argument after ** must be a mapping, not list
282
283 >>> h(a=1, **h)
284 Traceback (most recent call last):
285 ...
286 TypeError: h() argument after ** must be a mapping, not function
287
288 >>> h(a=1, **[])
289 Traceback (most recent call last):
290 ...
291 TypeError: h() argument after ** must be a mapping, not list
292
293 >>> h(**{'a': 1}, **h)
294 Traceback (most recent call last):
295 ...
296 TypeError: h() argument after ** must be a mapping, not function
297
298 >>> h(**{'a': 1}, **[])
299 Traceback (most recent call last):
300 ...
301 TypeError: h() argument after ** must be a mapping, not list
302
Christian Heimesb186d002008-03-18 15:15:01 +0000303 >>> dir(**h)
304 Traceback (most recent call last):
305 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300306 TypeError: dir() argument after ** must be a mapping, not function
Christian Heimesb186d002008-03-18 15:15:01 +0000307
308 >>> None(**h)
309 Traceback (most recent call last):
310 ...
Serhiy Storchakab7281052016-09-12 00:52:40 +0300311 TypeError: NoneType object argument after ** must be a mapping, \
312not function
Christian Heimesb186d002008-03-18 15:15:01 +0000313
314 >>> dir(b=1, **{'b': 1})
315 Traceback (most recent call last):
316 ...
317 TypeError: dir() got multiple values for keyword argument 'b'
318
319Another helper function
320
321 >>> def f2(*a, **b):
322 ... return a, b
Guido van Rossumd59da4b2007-05-22 18:11:13 +0000323
324
Christian Heimesb186d002008-03-18 15:15:01 +0000325 >>> d = {}
Neal Norwitzd79bacc2008-03-18 20:58:39 +0000326 >>> for i in range(512):
Christian Heimesb186d002008-03-18 15:15:01 +0000327 ... key = 'k%d' % i
328 ... d[key] = i
329 >>> a, b = f2(1, *(2,3), **d)
330 >>> len(a), len(b), b == d
331 (3, 512, True)
Raymond Hettinger40174c32003-05-31 07:04:16 +0000332
Christian Heimesb186d002008-03-18 15:15:01 +0000333 >>> class Foo:
334 ... def method(self, arg1, arg2):
335 ... return arg1+arg2
Fred Drake004d5e62000-10-23 17:22:08 +0000336
Christian Heimesb186d002008-03-18 15:15:01 +0000337 >>> x = Foo()
338 >>> Foo.method(*(x, 1, 2))
339 3
340 >>> Foo.method(x, *(1, 2))
341 3
342 >>> Foo.method(*(1, 2, 3))
Neal Norwitzd79bacc2008-03-18 20:58:39 +0000343 5
Christian Heimesb186d002008-03-18 15:15:01 +0000344 >>> Foo.method(1, *[2, 3])
Neal Norwitzd79bacc2008-03-18 20:58:39 +0000345 5
Fred Drake004d5e62000-10-23 17:22:08 +0000346
Ezio Melotti13925002011-03-16 11:05:33 +0200347A PyCFunction that takes only positional parameters should allow an
Christian Heimesb186d002008-03-18 15:15:01 +0000348empty keyword dictionary to pass without a complaint, but raise a
349TypeError if te dictionary is not empty
Jeremy Hylton074c3e62000-03-30 23:55:31 +0000350
Christian Heimesb186d002008-03-18 15:15:01 +0000351 >>> try:
352 ... silence = id(1, *{})
353 ... True
354 ... except:
355 ... False
356 True
Fred Drake004d5e62000-10-23 17:22:08 +0000357
Christian Heimesb186d002008-03-18 15:15:01 +0000358 >>> id(1, **{'foo': 1})
359 Traceback (most recent call last):
360 ...
361 TypeError: id() takes no keyword arguments
Jeremy Hylton074c3e62000-03-30 23:55:31 +0000362
Benjamin Peterson0289b152009-06-28 17:22:03 +0000363A corner case of keyword dictionary items being deleted during
364the function call setup. See <http://bugs.python.org/issue2016>.
365
366 >>> class Name(str):
367 ... def __eq__(self, other):
368 ... try:
369 ... del x[self]
370 ... except KeyError:
371 ... pass
372 ... return str.__eq__(self, other)
373 ... def __hash__(self):
374 ... return str.__hash__(self)
375
376 >>> x = {Name("a"):1, Name("b"):2}
377 >>> def f(a, b):
378 ... print(a,b)
379 >>> f(**x)
380 1 2
Benjamin Petersona567a772010-03-21 21:00:50 +0000381
Benjamin Petersone109c702011-06-24 09:37:26 -0500382Too many arguments:
Benjamin Petersona567a772010-03-21 21:00:50 +0000383
Benjamin Petersone109c702011-06-24 09:37:26 -0500384 >>> def f(): pass
Benjamin Petersonb204a422011-06-05 22:04:07 -0500385 >>> f(1)
386 Traceback (most recent call last):
387 ...
Benjamin Petersone109c702011-06-24 09:37:26 -0500388 TypeError: f() takes 0 positional arguments but 1 was given
389 >>> def f(a): pass
390 >>> f(1, 2)
Benjamin Petersonb204a422011-06-05 22:04:07 -0500391 Traceback (most recent call last):
392 ...
Benjamin Petersone109c702011-06-24 09:37:26 -0500393 TypeError: f() takes 1 positional argument but 2 were given
394 >>> def f(a, b=1): pass
395 >>> f(1, 2, 3)
Benjamin Petersonb204a422011-06-05 22:04:07 -0500396 Traceback (most recent call last):
397 ...
Benjamin Petersone109c702011-06-24 09:37:26 -0500398 TypeError: f() takes from 1 to 2 positional arguments but 3 were given
399 >>> def f(*, kw): pass
400 >>> f(1, kw=3)
Benjamin Petersonb204a422011-06-05 22:04:07 -0500401 Traceback (most recent call last):
402 ...
403 TypeError: f() takes 0 positional arguments but 1 positional argument (and 1 keyword-only argument) were given
Benjamin Petersone109c702011-06-24 09:37:26 -0500404 >>> def f(*, kw, b): pass
405 >>> f(1, 2, 3, b=3, kw=3)
406 Traceback (most recent call last):
407 ...
408 TypeError: f() takes 0 positional arguments but 3 positional arguments (and 2 keyword-only arguments) were given
409 >>> def f(a, b=2, *, kw): pass
410 >>> f(2, 3, 4, kw=4)
411 Traceback (most recent call last):
412 ...
413 TypeError: f() takes from 1 to 2 positional arguments but 3 positional arguments (and 1 keyword-only argument) were given
Benjamin Petersonb204a422011-06-05 22:04:07 -0500414
Benjamin Petersone109c702011-06-24 09:37:26 -0500415Too few and missing arguments:
416
417 >>> def f(a): pass
Benjamin Petersonb204a422011-06-05 22:04:07 -0500418 >>> f()
419 Traceback (most recent call last):
Benjamin Petersone109c702011-06-24 09:37:26 -0500420 ...
421 TypeError: f() missing 1 required positional argument: 'a'
422 >>> def f(a, b): pass
423 >>> f()
Benjamin Petersonb204a422011-06-05 22:04:07 -0500424 Traceback (most recent call last):
Benjamin Petersone109c702011-06-24 09:37:26 -0500425 ...
426 TypeError: f() missing 2 required positional arguments: 'a' and 'b'
427 >>> def f(a, b, c): pass
428 >>> f()
Benjamin Petersonb204a422011-06-05 22:04:07 -0500429 Traceback (most recent call last):
Benjamin Petersone109c702011-06-24 09:37:26 -0500430 ...
431 TypeError: f() missing 3 required positional arguments: 'a', 'b', and 'c'
432 >>> def f(a, b, c, d, e): pass
433 >>> f()
434 Traceback (most recent call last):
435 ...
436 TypeError: f() missing 5 required positional arguments: 'a', 'b', 'c', 'd', and 'e'
437 >>> def f(a, b=4, c=5, d=5): pass
438 >>> f(c=12, b=9)
439 Traceback (most recent call last):
440 ...
441 TypeError: f() missing 1 required positional argument: 'a'
442
443Same with keyword only args:
444
445 >>> def f(*, w): pass
446 >>> f()
447 Traceback (most recent call last):
448 ...
449 TypeError: f() missing 1 required keyword-only argument: 'w'
450 >>> def f(*, a, b, c, d, e): pass
451 >>> f()
452 Traceback (most recent call last):
453 ...
454 TypeError: f() missing 5 required keyword-only arguments: 'a', 'b', 'c', 'd', and 'e'
455
Christian Heimesb186d002008-03-18 15:15:01 +0000456"""
Samuele Pedroni8036c832004-02-21 21:03:30 +0000457
Benjamin Petersona567a772010-03-21 21:00:50 +0000458import sys
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000459from test import support
Samuele Pedroni8036c832004-02-21 21:03:30 +0000460
Christian Heimesb186d002008-03-18 15:15:01 +0000461def test_main():
Benjamin Petersona567a772010-03-21 21:00:50 +0000462 support.run_doctest(sys.modules[__name__], True)
Jeremy Hylton074c3e62000-03-30 23:55:31 +0000463
Christian Heimesb186d002008-03-18 15:15:01 +0000464if __name__ == '__main__':
465 test_main()