cli: fix typo in error message
[babeltrace.git] / tests / bindings / python / bt2 / test_graph.py
CommitLineData
32d2d479
MJ
1#
2# Copyright (C) 2019 EfficiOS Inc.
3#
4# This program is free software; you can redistribute it and/or
5# modify it under the terms of the GNU General Public License
6# as published by the Free Software Foundation; only version 2
7# of the License.
8#
9# This program is distributed in the hope that it will be useful,
10# but WITHOUT ANY WARRANTY; without even the implied warranty of
11# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12# GNU General Public License for more details.
13#
14# You should have received a copy of the GNU General Public License
15# along with this program; if not, write to the Free Software
16# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17#
18
ae0bfae8 19from bt2 import value
f6a5e476
PP
20import collections
21import unittest
22import copy
23import bt2
24
25
871a292a 26class _MyIter(bt2._UserMessageIterator):
a4dcfa96 27 def __init__(self, self_output_port):
871a292a
SM
28 self._build_meta()
29 self._at = 0
30
31 def _build_meta(self):
32 self._tc = self._component._create_trace_class()
33 self._t = self._tc()
34 self._sc = self._tc.create_stream_class()
35 self._ec = self._sc.create_event_class(name='salut')
36 self._my_int_ft = self._tc.create_signed_integer_field_class(32)
37 payload_ft = self._tc.create_structure_field_class()
38 payload_ft += collections.OrderedDict([
39 ('my_int', self._my_int_ft),
40 ])
41 self._ec.payload_field_type = payload_ft
42 self._stream = self._t.create_stream(self._sc)
43 self._packet = self._stream.create_packet()
44
45 def _create_event(self, value):
46 ev = self._ec()
47 ev.payload_field['my_int'] = value
48 ev.packet = self._packet
49 return ev
50
51
f6a5e476
PP
52class GraphTestCase(unittest.TestCase):
53 def setUp(self):
54 self._graph = bt2.Graph()
55
56 def tearDown(self):
57 del self._graph
58
59 def test_create_empty(self):
60 graph = bt2.Graph()
61
62 def test_add_component_user_cls(self):
63 class MySink(bt2._UserSinkComponent):
64 def _consume(self):
65 pass
66
67 comp = self._graph.add_component(MySink, 'salut')
68 self.assertEqual(comp.name, 'salut')
69
70 def test_add_component_gen_cls(self):
71 class MySink(bt2._UserSinkComponent):
72 def _consume(self):
73 pass
74
75 comp = self._graph.add_component(MySink, 'salut')
871a292a 76 assert comp
c88be1c8 77 comp2 = self._graph.add_component(comp.cls, 'salut2')
f6a5e476
PP
78 self.assertEqual(comp2.name, 'salut2')
79
80 def test_add_component_params(self):
81 comp_params = None
82
83 class MySink(bt2._UserSinkComponent):
84 def __init__(self, params):
85 nonlocal comp_params
86 comp_params = params
87
88 def _consume(self):
89 pass
90
91 params = {'hello': 23, 'path': '/path/to/stuff'}
92 comp = self._graph.add_component(MySink, 'salut', params)
93 self.assertEqual(params, comp_params)
94 del comp_params
95
96 def test_add_component_invalid_cls_type(self):
97 with self.assertRaises(TypeError):
98 self._graph.add_component(int, 'salut')
99
78c432bb
PP
100 def test_add_component_invalid_logging_level_type(self):
101 class MySink(bt2._UserSinkComponent):
102 def _consume(self):
103 pass
104
105 with self.assertRaises(TypeError):
106 self._graph.add_component(MySink, 'salut', logging_level='yo')
107
108 def test_add_component_invalid_logging_level_value(self):
109 class MySink(bt2._UserSinkComponent):
110 def _consume(self):
111 pass
112
113 with self.assertRaises(ValueError):
114 self._graph.add_component(MySink, 'salut', logging_level=12345)
115
116 def test_add_component_logging_level(self):
117 class MySink(bt2._UserSinkComponent):
118 def _consume(self):
119 pass
120
121 comp = self._graph.add_component(MySink, 'salut',
122 logging_level=bt2.LoggingLevel.DEBUG)
123 self.assertEqual(comp.logging_level, bt2.LoggingLevel.DEBUG)
124
f6a5e476 125 def test_connect_ports(self):
fa4c33e3 126 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
127 def __next__(self):
128 raise bt2.Stop
129
130 class MySource(bt2._UserSourceComponent,
fa4c33e3 131 message_iterator_class=MyIter):
f6a5e476
PP
132 def __init__(self, params):
133 self._add_output_port('out')
134
135 class MySink(bt2._UserSinkComponent):
136 def __init__(self, params):
137 self._add_input_port('in')
138
139 def _consume(self):
140 raise bt2.Stop
141
142 src = self._graph.add_component(MySource, 'src')
143 sink = self._graph.add_component(MySink, 'sink')
871a292a 144
f6a5e476
PP
145 conn = self._graph.connect_ports(src.output_ports['out'],
146 sink.input_ports['in'])
147 self.assertTrue(src.output_ports['out'].is_connected)
148 self.assertTrue(sink.input_ports['in'].is_connected)
871a292a
SM
149 self.assertEqual(src.output_ports['out'].connection._ptr, conn._ptr)
150 self.assertEqual(sink.input_ports['in'].connection._ptr, conn._ptr)
f6a5e476
PP
151
152 def test_connect_ports_invalid_direction(self):
fa4c33e3 153 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
154 def __next__(self):
155 raise bt2.Stop
156
157 class MySource(bt2._UserSourceComponent,
fa4c33e3 158 message_iterator_class=MyIter):
f6a5e476
PP
159 def __init__(self, params):
160 self._add_output_port('out')
161
162 class MySink(bt2._UserSinkComponent):
163 def __init__(self, params):
164 self._add_input_port('in')
165
166 def _consume(self):
167 raise bt2.Stop
168
169 src = self._graph.add_component(MySource, 'src')
170 sink = self._graph.add_component(MySink, 'sink')
171
172 with self.assertRaises(TypeError):
173 conn = self._graph.connect_ports(sink.input_ports['in'],
174 src.output_ports['out'])
175
871a292a
SM
176 def test_cancel(self):
177 self.assertFalse(self._graph.is_canceled)
f6a5e476 178 self._graph.cancel()
871a292a 179 self.assertTrue(self._graph.is_canceled)
f6a5e476 180
fb25b9e3 181 # Test that Graph.run() raises bt2.Canceled if the graph gets canceled
871a292a
SM
182 # during execution.
183 def test_cancel_while_running(self):
184 class MyIter(_MyIter):
9ef22b36 185 def __next__(self):
871a292a 186 return self._create_stream_beginning_message(self._stream)
9ef22b36
PP
187
188 class MySource(bt2._UserSourceComponent,
fa4c33e3 189 message_iterator_class=MyIter):
9ef22b36
PP
190 def __init__(self, params):
191 self._add_output_port('out')
192
193 class MySink(bt2._UserSinkComponent):
194 def __init__(self, params):
195 self._add_input_port('in')
196
197 def _consume(self):
871a292a
SM
198 # Pretend that somebody asynchronously cancelled the graph.
199 nonlocal graph
200 graph.cancel()
9ef22b36 201
871a292a 202 return next(self._msg_iter)
9ef22b36 203
871a292a
SM
204 def _graph_is_configured(self):
205 self._msg_iter = self._input_ports['in'].create_message_iterator()
9ef22b36 206
871a292a
SM
207 graph = bt2.Graph()
208 up = graph.add_component(MySource, 'down')
209 down = graph.add_component(MySink, 'up')
210 graph.connect_ports(up.output_ports['out'], down.input_ports['in'])
fb25b9e3 211 with self.assertRaises(bt2.Canceled):
871a292a 212 graph.run()
f6a5e476
PP
213
214 def test_run(self):
871a292a 215 class MyIter(_MyIter):
f6a5e476 216 def __next__(self):
871a292a
SM
217 if self._at == 9:
218 raise StopIteration
219
220 if self._at == 0:
221 msg = self._create_stream_beginning_message(self._stream)
222 elif self._at == 1:
223 msg = self._create_packet_beginning_message(self._packet)
224 elif self._at == 7:
225 msg = self._create_packet_end_message(self._packet)
226 elif self._at == 8:
227 msg = self._create_stream_end_message(self._stream)
228 else:
229 msg = self._create_event_message(self._ec, self._packet)
f6a5e476 230
f6a5e476 231 self._at += 1
fa4c33e3 232 return msg
f6a5e476
PP
233
234 class MySource(bt2._UserSourceComponent,
fa4c33e3 235 message_iterator_class=MyIter):
f6a5e476
PP
236 def __init__(self, params):
237 self._add_output_port('out')
238
239 class MySink(bt2._UserSinkComponent):
240 def __init__(self, params):
871a292a 241 self._input_port = self._add_input_port('in')
f6a5e476
PP
242 self._at = 0
243
244 def _consume(comp_self):
fa4c33e3 245 msg = next(comp_self._msg_iter)
f6a5e476
PP
246
247 if comp_self._at == 0:
871a292a 248 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 249 elif comp_self._at == 1:
871a292a 250 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
f6a5e476 251 elif comp_self._at >= 2 and comp_self._at <= 6:
871a292a 252 self.assertIsInstance(msg, bt2.message._EventMessage)
c88be1c8 253 self.assertEqual(msg.event.cls.name, 'salut')
f6a5e476 254 elif comp_self._at == 7:
871a292a 255 self.assertIsInstance(msg, bt2.message._PacketEndMessage)
f6a5e476 256 elif comp_self._at == 8:
871a292a 257 self.assertIsInstance(msg, bt2.message._StreamEndMessage)
f6a5e476
PP
258
259 comp_self._at += 1
260
871a292a
SM
261 def _graph_is_configured(self):
262 self._msg_iter = self._input_port.create_message_iterator()
f6a5e476
PP
263
264 src = self._graph.add_component(MySource, 'src')
265 sink = self._graph.add_component(MySink, 'sink')
266 conn = self._graph.connect_ports(src.output_ports['out'],
267 sink.input_ports['in'])
268 self._graph.run()
269
270 def test_run_again(self):
871a292a 271 class MyIter(_MyIter):
f6a5e476 272 def __next__(self):
871a292a 273 if self._at == 3:
f6a5e476
PP
274 raise bt2.TryAgain
275
871a292a
SM
276 if self._at == 0:
277 msg = self._create_stream_beginning_message(self._stream)
278 elif self._at == 1:
279 msg = self._create_packet_beginning_message(self._packet)
280 elif self._at == 2:
281 msg = self._create_event_message(self._ec, self._packet)
282
f6a5e476 283 self._at += 1
fa4c33e3 284 return msg
f6a5e476
PP
285
286 class MySource(bt2._UserSourceComponent,
fa4c33e3 287 message_iterator_class=MyIter):
f6a5e476
PP
288 def __init__(self, params):
289 self._add_output_port('out')
290
291 class MySink(bt2._UserSinkComponent):
292 def __init__(self, params):
871a292a 293 self._input_port = self._add_input_port('in')
f6a5e476
PP
294 self._at = 0
295
296 def _consume(comp_self):
871a292a 297 msg = next(comp_self._msg_iter)
f6a5e476 298 if comp_self._at == 0:
871a292a 299 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 300 elif comp_self._at == 1:
871a292a
SM
301 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
302 elif comp_self._at == 2:
303 self.assertIsInstance(msg, bt2.message._EventMessage)
f6a5e476 304 raise bt2.TryAgain
871a292a
SM
305 else:
306 pass
f6a5e476
PP
307
308 comp_self._at += 1
309
871a292a
SM
310 def _graph_is_configured(self):
311 self._msg_iter = self._input_port.create_message_iterator()
f6a5e476
PP
312
313 src = self._graph.add_component(MySource, 'src')
314 sink = self._graph.add_component(MySink, 'sink')
315 conn = self._graph.connect_ports(src.output_ports['out'],
316 sink.input_ports['in'])
317
318 with self.assertRaises(bt2.TryAgain):
319 self._graph.run()
320
f6a5e476 321 def test_run_error(self):
871a292a 322 raised_in_sink = False
f6a5e476 323
871a292a 324 class MyIter(_MyIter):
f6a5e476 325 def __next__(self):
871a292a
SM
326 # If this gets called after the sink raised an exception, it is
327 # an error.
328 nonlocal raised_in_sink
329 assert raised_in_sink is False
330
331 if self._at == 0:
332 msg = self._create_stream_beginning_message(self._stream)
333 elif self._at == 1:
334 msg = self._create_packet_beginning_message(self._packet)
335 elif self._at == 2 or self._at == 3:
336 msg = self._create_event_message(self._ec, self._packet)
337 else:
f6a5e476 338 raise bt2.TryAgain
f6a5e476 339 self._at += 1
fa4c33e3 340 return msg
f6a5e476
PP
341
342 class MySource(bt2._UserSourceComponent,
fa4c33e3 343 message_iterator_class=MyIter):
f6a5e476
PP
344 def __init__(self, params):
345 self._add_output_port('out')
346
347 class MySink(bt2._UserSinkComponent):
348 def __init__(self, params):
871a292a 349 self._input_port = self._add_input_port('in')
f6a5e476
PP
350 self._at = 0
351
352 def _consume(comp_self):
871a292a 353 msg = next(comp_self._msg_iter)
f6a5e476 354 if comp_self._at == 0:
871a292a 355 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 356 elif comp_self._at == 1:
871a292a
SM
357 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
358 elif comp_self._at == 2:
359 self.assertIsInstance(msg, bt2.message._EventMessage)
360 elif comp_self._at == 3:
361 nonlocal raised_in_sink
362 raised_in_sink = True
f6a5e476
PP
363 raise RuntimeError('error!')
364
365 comp_self._at += 1
366
871a292a
SM
367 def _graph_is_configured(self):
368 self._msg_iter = self._input_port.create_message_iterator()
f6a5e476
PP
369
370 src = self._graph.add_component(MySource, 'src')
371 sink = self._graph.add_component(MySink, 'sink')
372 conn = self._graph.connect_ports(src.output_ports['out'],
373 sink.input_ports['in'])
374
375 with self.assertRaises(bt2.Error):
376 self._graph.run()
377
871a292a 378 def test_listeners(self):
fa4c33e3 379 class MyIter(bt2._UserMessageIterator):
871a292a
SM
380 def __next__(self):
381 raise bt2.Stop
9ef22b36
PP
382
383 class MySource(bt2._UserSourceComponent,
fa4c33e3 384 message_iterator_class=MyIter):
9ef22b36
PP
385 def __init__(self, params):
386 self._add_output_port('out')
871a292a 387 self._add_output_port('zero')
9ef22b36
PP
388
389 class MySink(bt2._UserSinkComponent):
390 def __init__(self, params):
391 self._add_input_port('in')
9ef22b36 392
871a292a
SM
393 def _consume(self):
394 raise bt2.Stop
9ef22b36 395
871a292a
SM
396 def _port_connected(self, port, other_port):
397 self._add_input_port('taste')
9ef22b36 398
871a292a
SM
399 def port_added_listener(component, port):
400 nonlocal calls
401 calls.append((port_added_listener, component, port))
9ef22b36 402
871a292a
SM
403 def ports_connected_listener(upstream_component, upstream_port,
404 downstream_component, downstream_port):
405 nonlocal calls
406 calls.append((ports_connected_listener,
407 upstream_component, upstream_port,
408 downstream_component, downstream_port))
409
410 calls = []
411 self._graph.add_port_added_listener(port_added_listener)
412 self._graph.add_ports_connected_listener(ports_connected_listener)
9ef22b36
PP
413 src = self._graph.add_component(MySource, 'src')
414 sink = self._graph.add_component(MySink, 'sink')
871a292a
SM
415 self._graph.connect_ports(src.output_ports['out'],
416 sink.input_ports['in'])
9ef22b36 417
871a292a
SM
418 self.assertEqual(len(calls), 5)
419
420 self.assertIs(calls[0][0], port_added_listener)
421 self.assertEqual(calls[0][1].name, 'src')
422 self.assertEqual(calls[0][2].name, 'out')
423
424 self.assertIs(calls[1][0], port_added_listener)
425 self.assertEqual(calls[1][1].name, 'src')
426 self.assertEqual(calls[1][2].name, 'zero')
427
428 self.assertIs(calls[2][0], port_added_listener)
429 self.assertEqual(calls[2][1].name, 'sink')
430 self.assertEqual(calls[2][2].name, 'in')
431
432 self.assertIs(calls[3][0], port_added_listener)
433 self.assertEqual(calls[3][1].name, 'sink')
434 self.assertEqual(calls[3][2].name, 'taste')
435
436 self.assertIs(calls[4][0], ports_connected_listener)
437 self.assertEqual(calls[4][1].name, 'src')
438 self.assertEqual(calls[4][2].name, 'out')
439 self.assertEqual(calls[4][3].name, 'sink')
440 self.assertEqual(calls[4][4].name, 'in')
441
442 def test_invalid_listeners(self):
fa4c33e3 443 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
444 def __next__(self):
445 raise bt2.Stop
446
447 class MySource(bt2._UserSourceComponent,
fa4c33e3 448 message_iterator_class=MyIter):
f6a5e476
PP
449 def __init__(self, params):
450 self._add_output_port('out')
451 self._add_output_port('zero')
452
f6a5e476
PP
453 class MySink(bt2._UserSinkComponent):
454 def __init__(self, params):
455 self._add_input_port('in')
456
457 def _consume(self):
458 raise bt2.Stop
459
460 def _port_connected(self, port, other_port):
461 self._add_input_port('taste')
462
871a292a
SM
463 with self.assertRaises(TypeError):
464 self._graph.add_port_added_listener(1234)
465 with self.assertRaises(TypeError):
466 self._graph.add_ports_connected_listener(1234)
f6a5e476 467
871a292a
SM
468 def test_raise_in_component_init(self):
469 class MySink(bt2._UserSinkComponent):
470 def __init__(self, params):
471 raise ValueError('oops!')
f6a5e476 472
871a292a
SM
473 def _consume(self):
474 raise bt2.Stop
475
476 graph = bt2.Graph()
477
478 with self.assertRaises(bt2.Error):
479 graph.add_component(MySink, 'comp')
480
481 def test_raise_in_port_added_listener(self):
482 class MySink(bt2._UserSinkComponent):
483 def __init__(self, params):
484 self._add_input_port('in')
485
486 def _consume(self):
487 raise bt2.Stop
488
489 def port_added_listener(component, port):
490 raise ValueError('oh noes!')
491
492 graph = bt2.Graph()
493 graph.add_port_added_listener(port_added_listener)
494
495 with self.assertRaises(bt2.Error):
496 graph.add_component(MySink, 'comp')
497
498 def test_raise_in_ports_connected_listener(self):
499 class MyIter(bt2._UserMessageIterator):
500 def __next__(self):
501 raise bt2.Stop
502
503 class MySource(bt2._UserSourceComponent,
504 message_iterator_class=MyIter):
505 def __init__(self, params):
506 self._add_output_port('out')
507
508 class MySink(bt2._UserSinkComponent):
509 def __init__(self, params):
510 self._add_input_port('in')
511
512 def _consume(self):
513 raise bt2.Stop
f6a5e476 514
a4dcfa96
SM
515 def ports_connected_listener(upstream_component, upstream_port,
516 downstream_component, downstream_port):
871a292a 517 raise ValueError('oh noes!')
f6a5e476 518
871a292a
SM
519 graph = bt2.Graph()
520 graph.add_ports_connected_listener(ports_connected_listener)
521 up = graph.add_component(MySource, 'down')
522 down = graph.add_component(MySink, 'up')
f6a5e476 523
871a292a
SM
524 with self.assertRaises(bt2.Error):
525 graph.connect_ports(up.output_ports['out'], down.input_ports['in'])
This page took 0.054405 seconds and 4 git commands to generate.