blob: baa0f4e7fcbff0e200e5c99372e79c11d5bf4daa [file] [log] [blame]
Henry Schreinerd8c7ee02020-07-20 13:35:21 -04001# -*- coding: utf-8 -*-
Patrick Stewart54679792016-11-08 13:03:34 +00002import pytest
3import sys
Jason Rhinelander391c7542017-07-25 16:47:36 -04004from pybind11_tests import stl_binders as m
Patrick Stewart54679792016-11-08 13:03:34 +00005
6with pytest.suppress(ImportError):
7 import numpy as np
8
9
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020010def test_vector_int():
Jason Rhinelander391c7542017-07-25 16:47:36 -040011 v_int = m.VectorInt([0, 0])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020012 assert len(v_int) == 2
13 assert bool(v_int) is True
14
Chris Rusby22859bb2018-08-22 22:38:27 +010015 # test construction from a generator
16 v_int1 = m.VectorInt(x for x in range(5))
17 assert v_int1 == m.VectorInt([0, 1, 2, 3, 4])
18
Jason Rhinelander391c7542017-07-25 16:47:36 -040019 v_int2 = m.VectorInt([0, 0])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020020 assert v_int == v_int2
21 v_int2[1] = 1
22 assert v_int != v_int2
23
24 v_int2.append(2)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020025 v_int2.insert(0, 1)
26 v_int2.insert(0, 2)
27 v_int2.insert(0, 3)
Bruce Merryeee4f4f2017-05-25 15:17:36 +020028 v_int2.insert(6, 3)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020029 assert str(v_int2) == "VectorInt[3, 2, 1, 0, 1, 2, 3]"
Bruce Merryeee4f4f2017-05-25 15:17:36 +020030 with pytest.raises(IndexError):
31 v_int2.insert(8, 4)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020032
33 v_int.append(99)
34 v_int2[2:-2] = v_int
Jason Rhinelander391c7542017-07-25 16:47:36 -040035 assert v_int2 == m.VectorInt([3, 2, 0, 0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020036 del v_int2[1:3]
Jason Rhinelander391c7542017-07-25 16:47:36 -040037 assert v_int2 == m.VectorInt([3, 0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020038 del v_int2[0]
Jason Rhinelander391c7542017-07-25 16:47:36 -040039 assert v_int2 == m.VectorInt([0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020040
Chris Rusby22859bb2018-08-22 22:38:27 +010041 v_int2.extend(m.VectorInt([4, 5]))
42 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5])
43
44 v_int2.extend([6, 7])
45 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7])
46
47 # test error handling, and that the vector is unchanged
48 with pytest.raises(RuntimeError):
49 v_int2.extend([8, 'a'])
50
51 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7])
52
53 # test extending from a generator
54 v_int2.extend(x for x in range(5))
55 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 4])
56
ali-beep5ef13eb2019-08-15 13:41:12 -040057 # test negative indexing
58 assert v_int2[-1] == 4
59
60 # insert with negative index
61 v_int2.insert(-1, 88)
62 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 88, 4])
63
64 # delete negative index
65 del v_int2[-1]
66 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 88])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020067
fwjavoxe97c7352020-01-17 01:16:56 +010068 v_int2.clear()
69 assert len(v_int2) == 0
70
Henry Schreinera38e5332020-05-31 00:29:30 -040071
Jason Rhinelander90bac962017-04-14 17:50:44 -040072# related to the PyPy's buffer protocol.
Patrick Stewart54679792016-11-08 13:03:34 +000073@pytest.unsupported_on_pypy
74def test_vector_buffer():
Patrick Stewart54679792016-11-08 13:03:34 +000075 b = bytearray([1, 2, 3, 4])
Jason Rhinelander391c7542017-07-25 16:47:36 -040076 v = m.VectorUChar(b)
Patrick Stewart54679792016-11-08 13:03:34 +000077 assert v[1] == 2
78 v[2] = 5
Jason Rhinelander391c7542017-07-25 16:47:36 -040079 mv = memoryview(v) # We expose the buffer interface
Patrick Stewart54679792016-11-08 13:03:34 +000080 if sys.version_info.major > 2:
Jason Rhinelander391c7542017-07-25 16:47:36 -040081 assert mv[2] == 5
82 mv[2] = 6
Patrick Stewart54679792016-11-08 13:03:34 +000083 else:
Jason Rhinelander391c7542017-07-25 16:47:36 -040084 assert mv[2] == '\x05'
85 mv[2] = '\x06'
Patrick Stewart54679792016-11-08 13:03:34 +000086 assert v[2] == 6
87
marc-chiesa830adda2020-08-13 16:47:23 -040088 if sys.version_info.major > 2:
89 mv = memoryview(b)
90 v = m.VectorUChar(mv[::2])
91 assert v[1] == 3
92
Jason Rhinelander391c7542017-07-25 16:47:36 -040093 with pytest.raises(RuntimeError) as excinfo:
94 m.create_undeclstruct() # Undeclared struct contents, no buffer interface
95 assert "NumPy type info missing for " in str(excinfo.value)
Patrick Stewart54679792016-11-08 13:03:34 +000096
97
Jason Rhinelander90bac962017-04-14 17:50:44 -040098@pytest.unsupported_on_pypy
Patrick Stewart54679792016-11-08 13:03:34 +000099@pytest.requires_numpy
100def test_vector_buffer_numpy():
Patrick Stewart54679792016-11-08 13:03:34 +0000101 a = np.array([1, 2, 3, 4], dtype=np.int32)
102 with pytest.raises(TypeError):
Jason Rhinelander391c7542017-07-25 16:47:36 -0400103 m.VectorInt(a)
Patrick Stewart54679792016-11-08 13:03:34 +0000104
105 a = np.array([[1, 2, 3, 4], [5, 6, 7, 8], [9, 10, 11, 12]], dtype=np.uintc)
Jason Rhinelander391c7542017-07-25 16:47:36 -0400106 v = m.VectorInt(a[0, :])
Patrick Stewart54679792016-11-08 13:03:34 +0000107 assert len(v) == 4
108 assert v[2] == 3
Jason Rhinelander391c7542017-07-25 16:47:36 -0400109 ma = np.asarray(v)
110 ma[2] = 5
Patrick Stewart54679792016-11-08 13:03:34 +0000111 assert v[2] == 5
112
Jason Rhinelander391c7542017-07-25 16:47:36 -0400113 v = m.VectorInt(a[:, 1])
Patrick Stewart54679792016-11-08 13:03:34 +0000114 assert len(v) == 3
115 assert v[2] == 10
116
Jason Rhinelander391c7542017-07-25 16:47:36 -0400117 v = m.get_vectorstruct()
Patrick Stewart54679792016-11-08 13:03:34 +0000118 assert v[0].x == 5
Jason Rhinelander391c7542017-07-25 16:47:36 -0400119 ma = np.asarray(v)
120 ma[1]['x'] = 99
Patrick Stewart54679792016-11-08 13:03:34 +0000121 assert v[1].x == 99
122
Jason Rhinelander391c7542017-07-25 16:47:36 -0400123 v = m.VectorStruct(np.zeros(3, dtype=np.dtype([('w', 'bool'), ('x', 'I'),
124 ('y', 'float64'), ('z', 'bool')], align=True)))
Patrick Stewart0b6d08a2016-11-21 17:40:43 +0000125 assert len(v) == 3
126
marc-chiesa830adda2020-08-13 16:47:23 -0400127 b = np.array([1, 2, 3, 4], dtype=np.uint8)
128 v = m.VectorUChar(b[::2])
129 assert v[1] == 3
130
Patrick Stewart54679792016-11-08 13:03:34 +0000131
Dean Moldovana0c1ccf2016-08-12 13:50:00 +0200132def test_vector_bool():
Dean Moldovan3c4933c2017-09-01 21:42:20 +0200133 import pybind11_cross_module_tests as cm
134
135 vv_c = cm.VectorBool()
Dean Moldovana0c1ccf2016-08-12 13:50:00 +0200136 for i in range(10):
137 vv_c.append(i % 2 == 0)
138 for i in range(10):
139 assert vv_c[i] == (i % 2 == 0)
140 assert str(vv_c) == "VectorBool[1, 0, 1, 0, 1, 0, 1, 0, 1, 0]"
Sergey Lyskov75204182016-08-29 22:50:38 -0400141
Dean Moldovanbad17402016-11-20 21:21:54 +0100142
Jason Rhinelander391c7542017-07-25 16:47:36 -0400143def test_vector_custom():
144 v_a = m.VectorEl()
145 v_a.append(m.El(1))
146 v_a.append(m.El(2))
147 assert str(v_a) == "VectorEl[El{1}, El{2}]"
148
149 vv_a = m.VectorVectorEl()
150 vv_a.append(v_a)
151 vv_b = vv_a[0]
152 assert str(vv_b) == "VectorEl[El{1}, El{2}]"
153
154
Sergey Lyskov75204182016-08-29 22:50:38 -0400155def test_map_string_double():
Jason Rhinelander391c7542017-07-25 16:47:36 -0400156 mm = m.MapStringDouble()
157 mm['a'] = 1
158 mm['b'] = 2.5
Sergey Lyskov75204182016-08-29 22:50:38 -0400159
Jason Rhinelander391c7542017-07-25 16:47:36 -0400160 assert list(mm) == ['a', 'b']
161 assert list(mm.items()) == [('a', 1), ('b', 2.5)]
162 assert str(mm) == "MapStringDouble{a: 1, b: 2.5}"
Sergey Lyskov75204182016-08-29 22:50:38 -0400163
Jason Rhinelander391c7542017-07-25 16:47:36 -0400164 um = m.UnorderedMapStringDouble()
Sergey Lyskov75204182016-08-29 22:50:38 -0400165 um['ua'] = 1.1
166 um['ub'] = 2.6
167
Dean Moldovanbad17402016-11-20 21:21:54 +0100168 assert sorted(list(um)) == ['ua', 'ub']
169 assert sorted(list(um.items())) == [('ua', 1.1), ('ub', 2.6)]
170 assert "UnorderedMapStringDouble" in str(um)
Sergey Lyskov75204182016-08-29 22:50:38 -0400171
172
173def test_map_string_double_const():
Jason Rhinelander391c7542017-07-25 16:47:36 -0400174 mc = m.MapStringDoubleConst()
Sergey Lyskov75204182016-08-29 22:50:38 -0400175 mc['a'] = 10
176 mc['b'] = 20.5
177 assert str(mc) == "MapStringDoubleConst{a: 10, b: 20.5}"
178
Jason Rhinelander391c7542017-07-25 16:47:36 -0400179 umc = m.UnorderedMapStringDoubleConst()
Sergey Lyskov75204182016-08-29 22:50:38 -0400180 umc['a'] = 11
181 umc['b'] = 21.5
182
183 str(umc)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500184
Dean Moldovanbad17402016-11-20 21:21:54 +0100185
Jason Rhinelander391c7542017-07-25 16:47:36 -0400186def test_noncopyable_containers():
187 # std::vector
188 vnc = m.get_vnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500189 for i in range(0, 5):
Dean Moldovanbad17402016-11-20 21:21:54 +0100190 assert vnc[i].value == i + 1
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500191
Dean Moldovanbad17402016-11-20 21:21:54 +0100192 for i, j in enumerate(vnc, start=1):
193 assert j.value == i
194
Jason Rhinelander391c7542017-07-25 16:47:36 -0400195 # std::deque
196 dnc = m.get_dnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500197 for i in range(0, 5):
Dean Moldovanbad17402016-11-20 21:21:54 +0100198 assert dnc[i].value == i + 1
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500199
200 i = 1
201 for j in dnc:
202 assert(j.value == i)
203 i += 1
204
Jason Rhinelander391c7542017-07-25 16:47:36 -0400205 # std::map
206 mnc = m.get_mnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500207 for i in range(1, 6):
Dean Moldovanbad17402016-11-20 21:21:54 +0100208 assert mnc[i].value == 10 * i
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500209
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500210 vsum = 0
211 for k, v in mnc.items():
Dean Moldovanbad17402016-11-20 21:21:54 +0100212 assert v.value == 10 * k
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500213 vsum += v.value
214
Dean Moldovanbad17402016-11-20 21:21:54 +0100215 assert vsum == 150
216
Jason Rhinelander391c7542017-07-25 16:47:36 -0400217 # std::unordered_map
218 mnc = m.get_umnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500219 for i in range(1, 6):
Dean Moldovanbad17402016-11-20 21:21:54 +0100220 assert mnc[i].value == 10 * i
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500221
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500222 vsum = 0
223 for k, v in mnc.items():
Dean Moldovanbad17402016-11-20 21:21:54 +0100224 assert v.value == 10 * k
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500225 vsum += v.value
226
Dean Moldovanbad17402016-11-20 21:21:54 +0100227 assert vsum == 150
Jeff VanOss05d379a2018-01-11 18:43:37 -0500228
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100229 # nested std::map<std::vector>
230 nvnc = m.get_nvnc(5)
231 for i in range(1, 6):
232 for j in range(0, 5):
233 assert nvnc[i][j].value == j + 1
234
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400235 # Note: maps do not have .values()
236 for _, v in nvnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100237 for i, j in enumerate(v, start=1):
238 assert j.value == i
239
240 # nested std::map<std::map>
241 nmnc = m.get_nmnc(5)
242 for i in range(1, 6):
243 for j in range(10, 60, 10):
244 assert nmnc[i][j].value == 10 * j
245
246 vsum = 0
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400247 for _, v_o in nmnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100248 for k_i, v_i in v_o.items():
249 assert v_i.value == 10 * k_i
250 vsum += v_i.value
251
252 assert vsum == 7500
253
254 # nested std::unordered_map<std::unordered_map>
255 numnc = m.get_numnc(5)
256 for i in range(1, 6):
257 for j in range(10, 60, 10):
258 assert numnc[i][j].value == 10 * j
259
260 vsum = 0
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400261 for _, v_o in numnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100262 for k_i, v_i in v_o.items():
263 assert v_i.value == 10 * k_i
264 vsum += v_i.value
265
266 assert vsum == 7500
267
Jeff VanOss05d379a2018-01-11 18:43:37 -0500268
269def test_map_delitem():
270 mm = m.MapStringDouble()
271 mm['a'] = 1
272 mm['b'] = 2.5
273
274 assert list(mm) == ['a', 'b']
275 assert list(mm.items()) == [('a', 1), ('b', 2.5)]
276 del mm['a']
277 assert list(mm) == ['b']
278 assert list(mm.items()) == [('b', 2.5)]
279
280 um = m.UnorderedMapStringDouble()
281 um['ua'] = 1.1
282 um['ub'] = 2.6
283
284 assert sorted(list(um)) == ['ua', 'ub']
285 assert sorted(list(um.items())) == [('ua', 1.1), ('ub', 2.6)]
286 del um['ua']
287 assert sorted(list(um)) == ['ub']
288 assert sorted(list(um.items())) == [('ub', 2.6)]