test_kwargs_and_defaults.py
Go to the documentation of this file.
1 import pytest
2 
3 from pybind11_tests import kwargs_and_defaults as m
4 
5 
7  assert doc(m.kw_func0) == "kw_func0(arg0: int, arg1: int) -> str"
8  assert doc(m.kw_func1) == "kw_func1(x: int, y: int) -> str"
9  assert doc(m.kw_func2) == "kw_func2(x: int = 100, y: int = 200) -> str"
10  assert doc(m.kw_func3) == "kw_func3(data: str = 'Hello world!') -> None"
11  assert doc(m.kw_func4) == "kw_func4(myList: List[int] = [13, 17]) -> str"
12  assert doc(m.kw_func_udl) == "kw_func_udl(x: int, y: int = 300) -> str"
13  assert doc(m.kw_func_udl_z) == "kw_func_udl_z(x: int, y: int = 0) -> str"
14  assert doc(m.args_function) == "args_function(*args) -> tuple"
15  assert (
16  doc(m.args_kwargs_function) == "args_kwargs_function(*args, **kwargs) -> tuple"
17  )
18  assert (
19  doc(m.KWClass.foo0)
20  == "foo0(self: m.kwargs_and_defaults.KWClass, arg0: int, arg1: float) -> None"
21  )
22  assert (
23  doc(m.KWClass.foo1)
24  == "foo1(self: m.kwargs_and_defaults.KWClass, x: int, y: float) -> None"
25  )
26 
27 
29  assert m.kw_func0(5, 10) == "x=5, y=10"
30 
31  assert m.kw_func1(5, 10) == "x=5, y=10"
32  assert m.kw_func1(5, y=10) == "x=5, y=10"
33  assert m.kw_func1(y=10, x=5) == "x=5, y=10"
34 
35  assert m.kw_func2() == "x=100, y=200"
36  assert m.kw_func2(5) == "x=5, y=200"
37  assert m.kw_func2(x=5) == "x=5, y=200"
38  assert m.kw_func2(y=10) == "x=100, y=10"
39  assert m.kw_func2(5, 10) == "x=5, y=10"
40  assert m.kw_func2(x=5, y=10) == "x=5, y=10"
41 
42  with pytest.raises(TypeError) as excinfo:
43  # noinspection PyArgumentList
44  m.kw_func2(x=5, y=10, z=12)
45  assert excinfo.match(
46  r"(?s)^kw_func2\(\): incompatible.*Invoked with: kwargs: ((x=5|y=10|z=12)(, |$)){3}$"
47  )
48 
49  assert m.kw_func4() == "{13 17}"
50  assert m.kw_func4(myList=[1, 2, 3]) == "{1 2 3}"
51 
52  assert m.kw_func_udl(x=5, y=10) == "x=5, y=10"
53  assert m.kw_func_udl_z(x=5) == "x=5, y=0"
54 
55 
57  args = "arg1_value", "arg2_value", 3
58  assert m.args_function(*args) == args
59 
60  args = "a1", "a2"
61  kwargs = {"arg3": "a3", "arg4": 4}
62  assert m.args_kwargs_function(*args, **kwargs) == (args, kwargs)
63 
64 
66  mpa = m.mixed_plus_args
67  mpk = m.mixed_plus_kwargs
68  mpak = m.mixed_plus_args_kwargs
69  mpakd = m.mixed_plus_args_kwargs_defaults
70 
71  assert mpa(1, 2.5, 4, 99.5, None) == (1, 2.5, (4, 99.5, None))
72  assert mpa(1, 2.5) == (1, 2.5, ())
73  with pytest.raises(TypeError) as excinfo:
74  assert mpa(1)
75  assert (
76  msg(excinfo.value)
77  == """
78  mixed_plus_args(): incompatible function arguments. The following argument types are supported:
79  1. (arg0: int, arg1: float, *args) -> tuple
80 
81  Invoked with: 1
82  """
83  )
84  with pytest.raises(TypeError) as excinfo:
85  assert mpa()
86  assert (
87  msg(excinfo.value)
88  == """
89  mixed_plus_args(): incompatible function arguments. The following argument types are supported:
90  1. (arg0: int, arg1: float, *args) -> tuple
91 
92  Invoked with:
93  """
94  )
95 
96  assert mpk(-2, 3.5, pi=3.14159, e=2.71828) == (
97  -2,
98  3.5,
99  {"e": 2.71828, "pi": 3.14159},
100  )
101  assert mpak(7, 7.7, 7.77, 7.777, 7.7777, minusseven=-7) == (
102  7,
103  7.7,
104  (7.77, 7.777, 7.7777),
105  {"minusseven": -7},
106  )
107  assert mpakd() == (1, 3.14159, (), {})
108  assert mpakd(3) == (3, 3.14159, (), {})
109  assert mpakd(j=2.71828) == (1, 2.71828, (), {})
110  assert mpakd(k=42) == (1, 3.14159, (), {"k": 42})
111  assert mpakd(1, 1, 2, 3, 5, 8, then=13, followedby=21) == (
112  1,
113  1,
114  (2, 3, 5, 8),
115  {"then": 13, "followedby": 21},
116  )
117  # Arguments specified both positionally and via kwargs should fail:
118  with pytest.raises(TypeError) as excinfo:
119  assert mpakd(1, i=1)
120  assert (
121  msg(excinfo.value)
122  == """
123  mixed_plus_args_kwargs_defaults(): incompatible function arguments. The following argument types are supported:
124  1. (i: int = 1, j: float = 3.14159, *args, **kwargs) -> tuple
125 
126  Invoked with: 1; kwargs: i=1
127  """
128  )
129  with pytest.raises(TypeError) as excinfo:
130  assert mpakd(1, 2, j=1)
131  assert (
132  msg(excinfo.value)
133  == """
134  mixed_plus_args_kwargs_defaults(): incompatible function arguments. The following argument types are supported:
135  1. (i: int = 1, j: float = 3.14159, *args, **kwargs) -> tuple
136 
137  Invoked with: 1, 2; kwargs: j=1
138  """
139  )
140 
141  # Arguments after a py::args are automatically keyword-only (pybind 2.9+)
142  assert m.args_kwonly(2, 2.5, z=22) == (2, 2.5, (), 22)
143  assert m.args_kwonly(2, 2.5, "a", "b", "c", z=22) == (2, 2.5, ("a", "b", "c"), 22)
144  assert m.args_kwonly(z=22, i=4, j=16) == (4, 16, (), 22)
145 
146  with pytest.raises(TypeError) as excinfo:
147  assert m.args_kwonly(2, 2.5, 22) # missing z= keyword
148  assert (
149  msg(excinfo.value)
150  == """
151  args_kwonly(): incompatible function arguments. The following argument types are supported:
152  1. (i: int, j: float, *args, z: int) -> tuple
153 
154  Invoked with: 2, 2.5, 22
155  """
156  )
157 
158  assert m.args_kwonly_kwargs(i=1, k=4, j=10, z=-1, y=9) == (
159  1,
160  10,
161  (),
162  -1,
163  {"k": 4, "y": 9},
164  )
165  assert m.args_kwonly_kwargs(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, z=11, y=12) == (
166  1,
167  2,
168  (3, 4, 5, 6, 7, 8, 9, 10),
169  11,
170  {"y": 12},
171  )
172  assert (
173  m.args_kwonly_kwargs.__doc__
174  == "args_kwonly_kwargs(i: int, j: float, *args, z: int, **kwargs) -> tuple\n"
175  )
176 
177  assert (
178  m.args_kwonly_kwargs_defaults.__doc__
179  == "args_kwonly_kwargs_defaults(i: int = 1, j: float = 3.14159, *args, z: int = 42, **kwargs) -> tuple\n"
180  )
181  assert m.args_kwonly_kwargs_defaults() == (1, 3.14159, (), 42, {})
182  assert m.args_kwonly_kwargs_defaults(2) == (2, 3.14159, (), 42, {})
183  assert m.args_kwonly_kwargs_defaults(z=-99) == (1, 3.14159, (), -99, {})
184  assert m.args_kwonly_kwargs_defaults(5, 6, 7, 8) == (5, 6, (7, 8), 42, {})
185  assert m.args_kwonly_kwargs_defaults(5, 6, 7, m=8) == (5, 6, (7,), 42, {"m": 8})
186  assert m.args_kwonly_kwargs_defaults(5, 6, 7, m=8, z=9) == (5, 6, (7,), 9, {"m": 8})
187 
188 
190  assert m.kw_only_all(i=1, j=2) == (1, 2)
191  assert m.kw_only_all(j=1, i=2) == (2, 1)
192 
193  with pytest.raises(TypeError) as excinfo:
194  assert m.kw_only_all(i=1) == (1,)
195  assert "incompatible function arguments" in str(excinfo.value)
196 
197  with pytest.raises(TypeError) as excinfo:
198  assert m.kw_only_all(1, 2) == (1, 2)
199  assert "incompatible function arguments" in str(excinfo.value)
200 
201  assert m.kw_only_some(1, k=3, j=2) == (1, 2, 3)
202 
203  assert m.kw_only_with_defaults(z=8) == (3, 4, 5, 8)
204  assert m.kw_only_with_defaults(2, z=8) == (2, 4, 5, 8)
205  assert m.kw_only_with_defaults(2, j=7, k=8, z=9) == (2, 7, 8, 9)
206  assert m.kw_only_with_defaults(2, 7, z=9, k=8) == (2, 7, 8, 9)
207 
208  assert m.kw_only_mixed(1, j=2) == (1, 2)
209  assert m.kw_only_mixed(j=2, i=3) == (3, 2)
210  assert m.kw_only_mixed(i=2, j=3) == (2, 3)
211 
212  assert m.kw_only_plus_more(4, 5, k=6, extra=7) == (4, 5, 6, {"extra": 7})
213  assert m.kw_only_plus_more(3, k=5, j=4, extra=6) == (3, 4, 5, {"extra": 6})
214  assert m.kw_only_plus_more(2, k=3, extra=4) == (2, -1, 3, {"extra": 4})
215 
216  with pytest.raises(TypeError) as excinfo:
217  assert m.kw_only_mixed(i=1) == (1,)
218  assert "incompatible function arguments" in str(excinfo.value)
219 
220  with pytest.raises(RuntimeError) as excinfo:
221  m.register_invalid_kw_only(m)
222  assert (
223  msg(excinfo.value)
224  == """
225  arg(): cannot specify an unnamed argument after a kw_only() annotation or args() argument
226  """
227  )
228 
229  # https://github.com/pybind/pybind11/pull/3402#issuecomment-963341987
230  x = m.first_arg_kw_only(i=1)
231  x.method()
232  x.method(i=1, j=2)
233  assert (
234  m.first_arg_kw_only.__init__.__doc__
235  == "__init__(self: pybind11_tests.kwargs_and_defaults.first_arg_kw_only, *, i: int = 0) -> None\n"
236  )
237  assert (
238  m.first_arg_kw_only.method.__doc__
239  == "method(self: pybind11_tests.kwargs_and_defaults.first_arg_kw_only, *, i: int = 1, j: int = 2) -> None\n"
240  )
241 
242 
244  assert m.pos_only_all(1, 2) == (1, 2)
245  assert m.pos_only_all(2, 1) == (2, 1)
246 
247  with pytest.raises(TypeError) as excinfo:
248  m.pos_only_all(i=1, j=2)
249  assert "incompatible function arguments" in str(excinfo.value)
250 
251  assert m.pos_only_mix(1, 2) == (1, 2)
252  assert m.pos_only_mix(2, j=1) == (2, 1)
253 
254  with pytest.raises(TypeError) as excinfo:
255  m.pos_only_mix(i=1, j=2)
256  assert "incompatible function arguments" in str(excinfo.value)
257 
258  assert m.pos_kw_only_mix(1, 2, k=3) == (1, 2, 3)
259  assert m.pos_kw_only_mix(1, j=2, k=3) == (1, 2, 3)
260 
261  with pytest.raises(TypeError) as excinfo:
262  m.pos_kw_only_mix(i=1, j=2, k=3)
263  assert "incompatible function arguments" in str(excinfo.value)
264 
265  with pytest.raises(TypeError) as excinfo:
266  m.pos_kw_only_mix(1, 2, 3)
267  assert "incompatible function arguments" in str(excinfo.value)
268 
269  with pytest.raises(TypeError) as excinfo:
270  m.pos_only_def_mix()
271  assert "incompatible function arguments" in str(excinfo.value)
272 
273  assert m.pos_only_def_mix(1) == (1, 2, 3)
274  assert m.pos_only_def_mix(1, 4) == (1, 4, 3)
275  assert m.pos_only_def_mix(1, 4, 7) == (1, 4, 7)
276  assert m.pos_only_def_mix(1, 4, k=7) == (1, 4, 7)
277 
278  with pytest.raises(TypeError) as excinfo:
279  m.pos_only_def_mix(1, j=4)
280  assert "incompatible function arguments" in str(excinfo.value)
281 
282  # Mix it with args and kwargs:
283  assert (
284  m.args_kwonly_full_monty.__doc__
285  == "args_kwonly_full_monty(arg0: int = 1, arg1: int = 2, /, j: float = 3.14159, *args, z: int = 42, **kwargs) -> tuple\n"
286  )
287  assert m.args_kwonly_full_monty() == (1, 2, 3.14159, (), 42, {})
288  assert m.args_kwonly_full_monty(8) == (8, 2, 3.14159, (), 42, {})
289  assert m.args_kwonly_full_monty(8, 9) == (8, 9, 3.14159, (), 42, {})
290  assert m.args_kwonly_full_monty(8, 9, 10) == (8, 9, 10.0, (), 42, {})
291  assert m.args_kwonly_full_monty(3, 4, 5, 6, 7, m=8, z=9) == (
292  3,
293  4,
294  5.0,
295  (
296  6,
297  7,
298  ),
299  9,
300  {"m": 8},
301  )
302  assert m.args_kwonly_full_monty(3, 4, 5, 6, 7, m=8, z=9) == (
303  3,
304  4,
305  5.0,
306  (
307  6,
308  7,
309  ),
310  9,
311  {"m": 8},
312  )
313  assert m.args_kwonly_full_monty(5, j=7, m=8, z=9) == (5, 2, 7.0, (), 9, {"m": 8})
314  assert m.args_kwonly_full_monty(i=5, j=7, m=8, z=9) == (
315  1,
316  2,
317  7.0,
318  (),
319  9,
320  {"i": 5, "m": 8},
321  )
322 
323  # pos_only at the beginning of the argument list was "broken" in how it was displayed (though
324  # this is fairly useless in practice). Related to:
325  # https://github.com/pybind/pybind11/pull/3402#issuecomment-963341987
326  assert (
327  m.first_arg_kw_only.pos_only.__doc__
328  == "pos_only(self: pybind11_tests.kwargs_and_defaults.first_arg_kw_only, /, i: int, j: int) -> None\n"
329  )
330 
331 
333  assert m.kw_only_all.__doc__ == "kw_only_all(*, i: int, j: int) -> tuple\n"
334  assert m.kw_only_mixed.__doc__ == "kw_only_mixed(i: int, *, j: int) -> tuple\n"
335  assert m.pos_only_all.__doc__ == "pos_only_all(i: int, j: int, /) -> tuple\n"
336  assert m.pos_only_mix.__doc__ == "pos_only_mix(i: int, /, j: int) -> tuple\n"
337  assert (
338  m.pos_kw_only_mix.__doc__
339  == "pos_kw_only_mix(i: int, /, j: int, *, k: int) -> tuple\n"
340  )
341 
342 
344  """Issue/PR #1216 - py::args elements get double-inc_ref()ed when combined with regular
345  arguments"""
346  refcount = m.arg_refcount_h
347 
348  myval = 54321
349  expected = refcount(myval)
350  assert m.arg_refcount_h(myval) == expected
351  assert m.arg_refcount_o(myval) == expected + 1
352  assert m.arg_refcount_h(myval) == expected
353  assert refcount(myval) == expected
354 
355  assert m.mixed_plus_args(1, 2.0, "a", myval) == (1, 2.0, ("a", myval))
356  assert refcount(myval) == expected
357 
358  assert m.mixed_plus_kwargs(3, 4.0, a=1, b=myval) == (3, 4.0, {"a": 1, "b": myval})
359  assert refcount(myval) == expected
360 
361  assert m.args_function(-1, myval) == (-1, myval)
362  assert refcount(myval) == expected
363 
364  assert m.mixed_plus_args_kwargs(5, 6.0, myval, a=myval) == (
365  5,
366  6.0,
367  (myval,),
368  {"a": myval},
369  )
370  assert refcount(myval) == expected
371 
372  assert m.args_kwargs_function(7, 8, myval, a=1, b=myval) == (
373  (7, 8, myval),
374  {"a": 1, "b": myval},
375  )
376  assert refcount(myval) == expected
377 
378  exp3 = refcount(myval, myval, myval)
379  assert m.args_refcount(myval, myval, myval) == (exp3, exp3, exp3)
380  assert refcount(myval) == expected
381 
382  # This function takes the first arg as a `py::object` and the rest as a `py::args`. Unlike the
383  # previous case, when we have both positional and `py::args` we need to construct a new tuple
384  # for the `py::args`; in the previous case, we could simply inc_ref and pass on Python's input
385  # tuple without having to inc_ref the individual elements, but here we can't, hence the extra
386  # refs.
387  assert m.mixed_args_refcount(myval, myval, myval) == (exp3 + 3, exp3 + 3, exp3 + 3)
388 
389  assert m.class_default_argument() == "<class 'decimal.Decimal'>"
test_kwargs_and_defaults.test_named_arguments
def test_named_arguments()
Definition: test_kwargs_and_defaults.py:28
test_kwargs_and_defaults.test_arg_and_kwargs
def test_arg_and_kwargs()
Definition: test_kwargs_and_defaults.py:56
test_kwargs_and_defaults.test_signatures
def test_signatures()
Definition: test_kwargs_and_defaults.py:332
doc
Annotation for documentation.
Definition: attr.h:45
test_kwargs_and_defaults.test_mixed_args_and_kwargs
def test_mixed_args_and_kwargs(msg)
Definition: test_kwargs_and_defaults.py:65
test_kwargs_and_defaults.test_keyword_only_args
def test_keyword_only_args(msg)
Definition: test_kwargs_and_defaults.py:189
str
Definition: pytypes.h:1524
test_kwargs_and_defaults.test_positional_only_args
def test_positional_only_args()
Definition: test_kwargs_and_defaults.py:243
test_kwargs_and_defaults.test_args_refcount
def test_args_refcount()
Definition: test_kwargs_and_defaults.py:343
pybind11.msg
msg
Definition: wrap/pybind11/pybind11/__init__.py:4
test_kwargs_and_defaults.test_function_signatures
def test_function_signatures(doc)
Definition: test_kwargs_and_defaults.py:6


gtsam
Author(s):
autogenerated on Tue Jun 25 2024 03:05:28