blob: f9b8ea4af2a3156ecb09dd9d61d3464bb85ceefb [file] [log] [blame]
Henry Schreinerd8c7ee02020-07-20 13:35:21 -04001# -*- coding: utf-8 -*-
Patrick Stewart54679792016-11-08 13:03:34 +00002import pytest
Patrick Stewart54679792016-11-08 13:03:34 +00003
Henry Schreiner4d9024e2020-08-16 16:02:12 -04004import env # noqa: F401
5
6from pybind11_tests import stl_binders as m
Patrick Stewart54679792016-11-08 13:03:34 +00007
8
Dean Moldovana0c1ccf2016-08-12 13:50:00 +02009def test_vector_int():
Jason Rhinelander391c7542017-07-25 16:47:36 -040010 v_int = m.VectorInt([0, 0])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020011 assert len(v_int) == 2
12 assert bool(v_int) is True
13
Chris Rusby22859bb2018-08-22 22:38:27 +010014 # test construction from a generator
15 v_int1 = m.VectorInt(x for x in range(5))
16 assert v_int1 == m.VectorInt([0, 1, 2, 3, 4])
17
Jason Rhinelander391c7542017-07-25 16:47:36 -040018 v_int2 = m.VectorInt([0, 0])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020019 assert v_int == v_int2
20 v_int2[1] = 1
21 assert v_int != v_int2
22
23 v_int2.append(2)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020024 v_int2.insert(0, 1)
25 v_int2.insert(0, 2)
26 v_int2.insert(0, 3)
Bruce Merryeee4f4f2017-05-25 15:17:36 +020027 v_int2.insert(6, 3)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020028 assert str(v_int2) == "VectorInt[3, 2, 1, 0, 1, 2, 3]"
Bruce Merryeee4f4f2017-05-25 15:17:36 +020029 with pytest.raises(IndexError):
30 v_int2.insert(8, 4)
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020031
32 v_int.append(99)
33 v_int2[2:-2] = v_int
Jason Rhinelander391c7542017-07-25 16:47:36 -040034 assert v_int2 == m.VectorInt([3, 2, 0, 0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020035 del v_int2[1:3]
Jason Rhinelander391c7542017-07-25 16:47:36 -040036 assert v_int2 == m.VectorInt([3, 0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020037 del v_int2[0]
Jason Rhinelander391c7542017-07-25 16:47:36 -040038 assert v_int2 == m.VectorInt([0, 99, 2, 3])
Dean Moldovana0c1ccf2016-08-12 13:50:00 +020039
Chris Rusby22859bb2018-08-22 22:38:27 +010040 v_int2.extend(m.VectorInt([4, 5]))
41 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5])
42
43 v_int2.extend([6, 7])
44 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7])
45
46 # test error handling, and that the vector is unchanged
47 with pytest.raises(RuntimeError):
48 v_int2.extend([8, 'a'])
49
50 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7])
51
52 # test extending from a generator
53 v_int2.extend(x for x in range(5))
54 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 4])
55
ali-beep5ef13eb2019-08-15 13:41:12 -040056 # test negative indexing
57 assert v_int2[-1] == 4
58
59 # insert with negative index
60 v_int2.insert(-1, 88)
61 assert v_int2 == m.VectorInt([0, 99, 2, 3, 4, 5, 6, 7, 0, 1, 2, 3, 88, 4])
62
63 # delete negative index
64 del v_int2[-1]
65 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 +020066
fwjavoxe97c7352020-01-17 01:16:56 +010067 v_int2.clear()
68 assert len(v_int2) == 0
69
Henry Schreinera38e5332020-05-31 00:29:30 -040070
Henry Schreiner4d9024e2020-08-16 16:02:12 -040071# Older PyPy's failed here, related to the PyPy's buffer protocol.
Patrick Stewart54679792016-11-08 13:03:34 +000072def test_vector_buffer():
Patrick Stewart54679792016-11-08 13:03:34 +000073 b = bytearray([1, 2, 3, 4])
Jason Rhinelander391c7542017-07-25 16:47:36 -040074 v = m.VectorUChar(b)
Patrick Stewart54679792016-11-08 13:03:34 +000075 assert v[1] == 2
76 v[2] = 5
Jason Rhinelander391c7542017-07-25 16:47:36 -040077 mv = memoryview(v) # We expose the buffer interface
Henry Schreiner4d9024e2020-08-16 16:02:12 -040078 if not env.PY2:
Jason Rhinelander391c7542017-07-25 16:47:36 -040079 assert mv[2] == 5
80 mv[2] = 6
Patrick Stewart54679792016-11-08 13:03:34 +000081 else:
Jason Rhinelander391c7542017-07-25 16:47:36 -040082 assert mv[2] == '\x05'
83 mv[2] = '\x06'
Patrick Stewart54679792016-11-08 13:03:34 +000084 assert v[2] == 6
85
Henry Schreiner4d9024e2020-08-16 16:02:12 -040086 if not env.PY2:
marc-chiesa830adda2020-08-13 16:47:23 -040087 mv = memoryview(b)
88 v = m.VectorUChar(mv[::2])
89 assert v[1] == 3
90
Jason Rhinelander391c7542017-07-25 16:47:36 -040091 with pytest.raises(RuntimeError) as excinfo:
92 m.create_undeclstruct() # Undeclared struct contents, no buffer interface
93 assert "NumPy type info missing for " in str(excinfo.value)
Patrick Stewart54679792016-11-08 13:03:34 +000094
95
Patrick Stewart54679792016-11-08 13:03:34 +000096def test_vector_buffer_numpy():
Henry Schreiner4d9024e2020-08-16 16:02:12 -040097 np = pytest.importorskip("numpy")
Patrick Stewart54679792016-11-08 13:03:34 +000098 a = np.array([1, 2, 3, 4], dtype=np.int32)
99 with pytest.raises(TypeError):
Jason Rhinelander391c7542017-07-25 16:47:36 -0400100 m.VectorInt(a)
Patrick Stewart54679792016-11-08 13:03:34 +0000101
102 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 -0400103 v = m.VectorInt(a[0, :])
Patrick Stewart54679792016-11-08 13:03:34 +0000104 assert len(v) == 4
105 assert v[2] == 3
Jason Rhinelander391c7542017-07-25 16:47:36 -0400106 ma = np.asarray(v)
107 ma[2] = 5
Patrick Stewart54679792016-11-08 13:03:34 +0000108 assert v[2] == 5
109
Jason Rhinelander391c7542017-07-25 16:47:36 -0400110 v = m.VectorInt(a[:, 1])
Patrick Stewart54679792016-11-08 13:03:34 +0000111 assert len(v) == 3
112 assert v[2] == 10
113
Jason Rhinelander391c7542017-07-25 16:47:36 -0400114 v = m.get_vectorstruct()
Patrick Stewart54679792016-11-08 13:03:34 +0000115 assert v[0].x == 5
Jason Rhinelander391c7542017-07-25 16:47:36 -0400116 ma = np.asarray(v)
117 ma[1]['x'] = 99
Patrick Stewart54679792016-11-08 13:03:34 +0000118 assert v[1].x == 99
119
Jason Rhinelander391c7542017-07-25 16:47:36 -0400120 v = m.VectorStruct(np.zeros(3, dtype=np.dtype([('w', 'bool'), ('x', 'I'),
121 ('y', 'float64'), ('z', 'bool')], align=True)))
Patrick Stewart0b6d08a2016-11-21 17:40:43 +0000122 assert len(v) == 3
123
marc-chiesa830adda2020-08-13 16:47:23 -0400124 b = np.array([1, 2, 3, 4], dtype=np.uint8)
125 v = m.VectorUChar(b[::2])
126 assert v[1] == 3
127
Patrick Stewart54679792016-11-08 13:03:34 +0000128
Dean Moldovana0c1ccf2016-08-12 13:50:00 +0200129def test_vector_bool():
Dean Moldovan3c4933c2017-09-01 21:42:20 +0200130 import pybind11_cross_module_tests as cm
131
132 vv_c = cm.VectorBool()
Dean Moldovana0c1ccf2016-08-12 13:50:00 +0200133 for i in range(10):
134 vv_c.append(i % 2 == 0)
135 for i in range(10):
136 assert vv_c[i] == (i % 2 == 0)
137 assert str(vv_c) == "VectorBool[1, 0, 1, 0, 1, 0, 1, 0, 1, 0]"
Sergey Lyskov75204182016-08-29 22:50:38 -0400138
Dean Moldovanbad17402016-11-20 21:21:54 +0100139
Jason Rhinelander391c7542017-07-25 16:47:36 -0400140def test_vector_custom():
141 v_a = m.VectorEl()
142 v_a.append(m.El(1))
143 v_a.append(m.El(2))
144 assert str(v_a) == "VectorEl[El{1}, El{2}]"
145
146 vv_a = m.VectorVectorEl()
147 vv_a.append(v_a)
148 vv_b = vv_a[0]
149 assert str(vv_b) == "VectorEl[El{1}, El{2}]"
150
151
Sergey Lyskov75204182016-08-29 22:50:38 -0400152def test_map_string_double():
Jason Rhinelander391c7542017-07-25 16:47:36 -0400153 mm = m.MapStringDouble()
154 mm['a'] = 1
155 mm['b'] = 2.5
Sergey Lyskov75204182016-08-29 22:50:38 -0400156
Jason Rhinelander391c7542017-07-25 16:47:36 -0400157 assert list(mm) == ['a', 'b']
158 assert list(mm.items()) == [('a', 1), ('b', 2.5)]
159 assert str(mm) == "MapStringDouble{a: 1, b: 2.5}"
Sergey Lyskov75204182016-08-29 22:50:38 -0400160
Jason Rhinelander391c7542017-07-25 16:47:36 -0400161 um = m.UnorderedMapStringDouble()
Sergey Lyskov75204182016-08-29 22:50:38 -0400162 um['ua'] = 1.1
163 um['ub'] = 2.6
164
Dean Moldovanbad17402016-11-20 21:21:54 +0100165 assert sorted(list(um)) == ['ua', 'ub']
166 assert sorted(list(um.items())) == [('ua', 1.1), ('ub', 2.6)]
167 assert "UnorderedMapStringDouble" in str(um)
Sergey Lyskov75204182016-08-29 22:50:38 -0400168
169
170def test_map_string_double_const():
Jason Rhinelander391c7542017-07-25 16:47:36 -0400171 mc = m.MapStringDoubleConst()
Sergey Lyskov75204182016-08-29 22:50:38 -0400172 mc['a'] = 10
173 mc['b'] = 20.5
174 assert str(mc) == "MapStringDoubleConst{a: 10, b: 20.5}"
175
Jason Rhinelander391c7542017-07-25 16:47:36 -0400176 umc = m.UnorderedMapStringDoubleConst()
Sergey Lyskov75204182016-08-29 22:50:38 -0400177 umc['a'] = 11
178 umc['b'] = 21.5
179
180 str(umc)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500181
Dean Moldovanbad17402016-11-20 21:21:54 +0100182
Jason Rhinelander391c7542017-07-25 16:47:36 -0400183def test_noncopyable_containers():
184 # std::vector
185 vnc = m.get_vnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500186 for i in range(0, 5):
Dean Moldovanbad17402016-11-20 21:21:54 +0100187 assert vnc[i].value == i + 1
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500188
Dean Moldovanbad17402016-11-20 21:21:54 +0100189 for i, j in enumerate(vnc, start=1):
190 assert j.value == i
191
Jason Rhinelander391c7542017-07-25 16:47:36 -0400192 # std::deque
193 dnc = m.get_dnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500194 for i in range(0, 5):
Dean Moldovanbad17402016-11-20 21:21:54 +0100195 assert dnc[i].value == i + 1
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500196
197 i = 1
198 for j in dnc:
199 assert(j.value == i)
200 i += 1
201
Jason Rhinelander391c7542017-07-25 16:47:36 -0400202 # std::map
203 mnc = m.get_mnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500204 for i in range(1, 6):
Dean Moldovanbad17402016-11-20 21:21:54 +0100205 assert mnc[i].value == 10 * i
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500206
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500207 vsum = 0
208 for k, v in mnc.items():
Dean Moldovanbad17402016-11-20 21:21:54 +0100209 assert v.value == 10 * k
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500210 vsum += v.value
211
Dean Moldovanbad17402016-11-20 21:21:54 +0100212 assert vsum == 150
213
Jason Rhinelander391c7542017-07-25 16:47:36 -0400214 # std::unordered_map
215 mnc = m.get_umnc(5)
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500216 for i in range(1, 6):
Dean Moldovanbad17402016-11-20 21:21:54 +0100217 assert mnc[i].value == 10 * i
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500218
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500219 vsum = 0
220 for k, v in mnc.items():
Dean Moldovanbad17402016-11-20 21:21:54 +0100221 assert v.value == 10 * k
Jason Rhinelander617fbcf2016-11-15 06:30:38 -0500222 vsum += v.value
223
Dean Moldovanbad17402016-11-20 21:21:54 +0100224 assert vsum == 150
Jeff VanOss05d379a2018-01-11 18:43:37 -0500225
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100226 # nested std::map<std::vector>
227 nvnc = m.get_nvnc(5)
228 for i in range(1, 6):
229 for j in range(0, 5):
230 assert nvnc[i][j].value == j + 1
231
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400232 # Note: maps do not have .values()
233 for _, v in nvnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100234 for i, j in enumerate(v, start=1):
235 assert j.value == i
236
237 # nested std::map<std::map>
238 nmnc = m.get_nmnc(5)
239 for i in range(1, 6):
240 for j in range(10, 60, 10):
241 assert nmnc[i][j].value == 10 * j
242
243 vsum = 0
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400244 for _, v_o in nmnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100245 for k_i, v_i in v_o.items():
246 assert v_i.value == 10 * k_i
247 vsum += v_i.value
248
249 assert vsum == 7500
250
251 # nested std::unordered_map<std::unordered_map>
252 numnc = m.get_numnc(5)
253 for i in range(1, 6):
254 for j in range(10, 60, 10):
255 assert numnc[i][j].value == 10 * j
256
257 vsum = 0
Henry Schreinerd8c7ee02020-07-20 13:35:21 -0400258 for _, v_o in numnc.items():
Sebastian Gsängera83d69e2019-10-31 12:38:24 +0100259 for k_i, v_i in v_o.items():
260 assert v_i.value == 10 * k_i
261 vsum += v_i.value
262
263 assert vsum == 7500
264
Jeff VanOss05d379a2018-01-11 18:43:37 -0500265
266def test_map_delitem():
267 mm = m.MapStringDouble()
268 mm['a'] = 1
269 mm['b'] = 2.5
270
271 assert list(mm) == ['a', 'b']
272 assert list(mm.items()) == [('a', 1), ('b', 2.5)]
273 del mm['a']
274 assert list(mm) == ['b']
275 assert list(mm.items()) == [('b', 2.5)]
276
277 um = m.UnorderedMapStringDouble()
278 um['ua'] = 1.1
279 um['ub'] = 2.6
280
281 assert sorted(list(um)) == ['ua', 'ub']
282 assert sorted(list(um.items())) == [('ua', 1.1), ('ub', 2.6)]
283 del um['ua']
284 assert sorted(list(um)) == ['ub']
285 assert sorted(list(um.items())) == [('ub', 2.6)]