bt2: add interrupter support
[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()
37a93d41 34 self._sc = self._tc.create_stream_class(supports_packets=True)
871a292a
SM
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()
61d96b89 38 payload_ft += [('my_int', self._my_int_ft)]
871a292a
SM
39 self._ec.payload_field_type = payload_ft
40 self._stream = self._t.create_stream(self._sc)
41 self._packet = self._stream.create_packet()
42
43 def _create_event(self, value):
44 ev = self._ec()
45 ev.payload_field['my_int'] = value
46 ev.packet = self._packet
47 return ev
48
49
f6a5e476
PP
50class GraphTestCase(unittest.TestCase):
51 def setUp(self):
52 self._graph = bt2.Graph()
53
54 def tearDown(self):
55 del self._graph
56
57 def test_create_empty(self):
58 graph = bt2.Graph()
59
60 def test_add_component_user_cls(self):
61 class MySink(bt2._UserSinkComponent):
62 def _consume(self):
63 pass
61d96b89 64
8a08af82
SM
65 def _graph_is_configured(self):
66 pass
f6a5e476
PP
67
68 comp = self._graph.add_component(MySink, 'salut')
69 self.assertEqual(comp.name, 'salut')
70
71 def test_add_component_gen_cls(self):
72 class MySink(bt2._UserSinkComponent):
73 def _consume(self):
74 pass
61d96b89 75
8a08af82
SM
76 def _graph_is_configured(self):
77 pass
f6a5e476
PP
78
79 comp = self._graph.add_component(MySink, 'salut')
871a292a 80 assert comp
c88be1c8 81 comp2 = self._graph.add_component(comp.cls, 'salut2')
f6a5e476
PP
82 self.assertEqual(comp2.name, 'salut2')
83
84 def test_add_component_params(self):
85 comp_params = None
86
87 class MySink(bt2._UserSinkComponent):
88 def __init__(self, params):
89 nonlocal comp_params
90 comp_params = params
91
92 def _consume(self):
93 pass
61d96b89 94
8a08af82
SM
95 def _graph_is_configured(self):
96 pass
f6a5e476
PP
97
98 params = {'hello': 23, 'path': '/path/to/stuff'}
99 comp = self._graph.add_component(MySink, 'salut', params)
100 self.assertEqual(params, comp_params)
101 del comp_params
102
103 def test_add_component_invalid_cls_type(self):
104 with self.assertRaises(TypeError):
105 self._graph.add_component(int, 'salut')
106
78c432bb
PP
107 def test_add_component_invalid_logging_level_type(self):
108 class MySink(bt2._UserSinkComponent):
109 def _consume(self):
110 pass
61d96b89 111
8a08af82
SM
112 def _graph_is_configured(self):
113 pass
78c432bb
PP
114
115 with self.assertRaises(TypeError):
116 self._graph.add_component(MySink, 'salut', logging_level='yo')
117
118 def test_add_component_invalid_logging_level_value(self):
119 class MySink(bt2._UserSinkComponent):
120 def _consume(self):
121 pass
61d96b89 122
8a08af82
SM
123 def _graph_is_configured(self):
124 pass
78c432bb
PP
125
126 with self.assertRaises(ValueError):
127 self._graph.add_component(MySink, 'salut', logging_level=12345)
128
129 def test_add_component_logging_level(self):
130 class MySink(bt2._UserSinkComponent):
131 def _consume(self):
132 pass
61d96b89 133
8a08af82
SM
134 def _graph_is_configured(self):
135 pass
78c432bb 136
61d96b89
FD
137 comp = self._graph.add_component(
138 MySink, 'salut', logging_level=bt2.LoggingLevel.DEBUG
139 )
78c432bb
PP
140 self.assertEqual(comp.logging_level, bt2.LoggingLevel.DEBUG)
141
f6a5e476 142 def test_connect_ports(self):
fa4c33e3 143 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
144 def __next__(self):
145 raise bt2.Stop
146
61d96b89 147 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
148 def __init__(self, params):
149 self._add_output_port('out')
150
151 class MySink(bt2._UserSinkComponent):
152 def __init__(self, params):
153 self._add_input_port('in')
154
155 def _consume(self):
156 raise bt2.Stop
157
8a08af82
SM
158 def _graph_is_configured(self):
159 pass
160
f6a5e476
PP
161 src = self._graph.add_component(MySource, 'src')
162 sink = self._graph.add_component(MySink, 'sink')
871a292a 163
61d96b89
FD
164 conn = self._graph.connect_ports(
165 src.output_ports['out'], sink.input_ports['in']
166 )
f6a5e476
PP
167 self.assertTrue(src.output_ports['out'].is_connected)
168 self.assertTrue(sink.input_ports['in'].is_connected)
3e2cc5dc
FD
169 self.assertEqual(src.output_ports['out'].connection.addr, conn.addr)
170 self.assertEqual(sink.input_ports['in'].connection.addr, conn.addr)
f6a5e476
PP
171
172 def test_connect_ports_invalid_direction(self):
fa4c33e3 173 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
174 def __next__(self):
175 raise bt2.Stop
176
61d96b89 177 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
178 def __init__(self, params):
179 self._add_output_port('out')
180
181 class MySink(bt2._UserSinkComponent):
182 def __init__(self, params):
183 self._add_input_port('in')
184
185 def _consume(self):
186 raise bt2.Stop
187
8a08af82
SM
188 def _graph_is_configured(self):
189 pass
190
f6a5e476
PP
191 src = self._graph.add_component(MySource, 'src')
192 sink = self._graph.add_component(MySink, 'sink')
193
194 with self.assertRaises(TypeError):
61d96b89
FD
195 conn = self._graph.connect_ports(
196 sink.input_ports['in'], src.output_ports['out']
197 )
f6a5e476 198
871a292a
SM
199 def test_cancel(self):
200 self.assertFalse(self._graph.is_canceled)
f6a5e476 201 self._graph.cancel()
871a292a 202 self.assertTrue(self._graph.is_canceled)
f6a5e476 203
fb25b9e3 204 # Test that Graph.run() raises bt2.Canceled if the graph gets canceled
871a292a
SM
205 # during execution.
206 def test_cancel_while_running(self):
207 class MyIter(_MyIter):
9ef22b36 208 def __next__(self):
871a292a 209 return self._create_stream_beginning_message(self._stream)
9ef22b36 210
61d96b89 211 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
9ef22b36
PP
212 def __init__(self, params):
213 self._add_output_port('out')
214
215 class MySink(bt2._UserSinkComponent):
216 def __init__(self, params):
217 self._add_input_port('in')
218
219 def _consume(self):
871a292a
SM
220 # Pretend that somebody asynchronously cancelled the graph.
221 nonlocal graph
222 graph.cancel()
9ef22b36 223
871a292a 224 return next(self._msg_iter)
9ef22b36 225
871a292a 226 def _graph_is_configured(self):
692f1a01
PP
227 self._msg_iter = self._create_input_port_message_iterator(
228 self._input_ports['in']
229 )
9ef22b36 230
871a292a
SM
231 graph = bt2.Graph()
232 up = graph.add_component(MySource, 'down')
233 down = graph.add_component(MySink, 'up')
234 graph.connect_ports(up.output_ports['out'], down.input_ports['in'])
fb25b9e3 235 with self.assertRaises(bt2.Canceled):
871a292a 236 graph.run()
f6a5e476
PP
237
238 def test_run(self):
871a292a 239 class MyIter(_MyIter):
f6a5e476 240 def __next__(self):
871a292a
SM
241 if self._at == 9:
242 raise StopIteration
243
244 if self._at == 0:
245 msg = self._create_stream_beginning_message(self._stream)
246 elif self._at == 1:
247 msg = self._create_packet_beginning_message(self._packet)
248 elif self._at == 7:
249 msg = self._create_packet_end_message(self._packet)
250 elif self._at == 8:
251 msg = self._create_stream_end_message(self._stream)
252 else:
253 msg = self._create_event_message(self._ec, self._packet)
f6a5e476 254
f6a5e476 255 self._at += 1
fa4c33e3 256 return msg
f6a5e476 257
61d96b89 258 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
259 def __init__(self, params):
260 self._add_output_port('out')
261
262 class MySink(bt2._UserSinkComponent):
263 def __init__(self, params):
871a292a 264 self._input_port = self._add_input_port('in')
f6a5e476
PP
265 self._at = 0
266
267 def _consume(comp_self):
fa4c33e3 268 msg = next(comp_self._msg_iter)
f6a5e476
PP
269
270 if comp_self._at == 0:
871a292a 271 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 272 elif comp_self._at == 1:
871a292a 273 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
f6a5e476 274 elif comp_self._at >= 2 and comp_self._at <= 6:
871a292a 275 self.assertIsInstance(msg, bt2.message._EventMessage)
c88be1c8 276 self.assertEqual(msg.event.cls.name, 'salut')
f6a5e476 277 elif comp_self._at == 7:
871a292a 278 self.assertIsInstance(msg, bt2.message._PacketEndMessage)
f6a5e476 279 elif comp_self._at == 8:
871a292a 280 self.assertIsInstance(msg, bt2.message._StreamEndMessage)
f6a5e476
PP
281
282 comp_self._at += 1
283
871a292a 284 def _graph_is_configured(self):
692f1a01
PP
285 self._msg_iter = self._create_input_port_message_iterator(
286 self._input_port
287 )
f6a5e476
PP
288
289 src = self._graph.add_component(MySource, 'src')
290 sink = self._graph.add_component(MySink, 'sink')
61d96b89
FD
291 conn = self._graph.connect_ports(
292 src.output_ports['out'], sink.input_ports['in']
293 )
f6a5e476
PP
294 self._graph.run()
295
296 def test_run_again(self):
871a292a 297 class MyIter(_MyIter):
f6a5e476 298 def __next__(self):
871a292a 299 if self._at == 3:
f6a5e476
PP
300 raise bt2.TryAgain
301
871a292a
SM
302 if self._at == 0:
303 msg = self._create_stream_beginning_message(self._stream)
304 elif self._at == 1:
305 msg = self._create_packet_beginning_message(self._packet)
306 elif self._at == 2:
307 msg = self._create_event_message(self._ec, self._packet)
308
f6a5e476 309 self._at += 1
fa4c33e3 310 return msg
f6a5e476 311
61d96b89 312 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
313 def __init__(self, params):
314 self._add_output_port('out')
315
316 class MySink(bt2._UserSinkComponent):
317 def __init__(self, params):
871a292a 318 self._input_port = self._add_input_port('in')
f6a5e476
PP
319 self._at = 0
320
321 def _consume(comp_self):
871a292a 322 msg = next(comp_self._msg_iter)
f6a5e476 323 if comp_self._at == 0:
871a292a 324 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 325 elif comp_self._at == 1:
871a292a
SM
326 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
327 elif comp_self._at == 2:
328 self.assertIsInstance(msg, bt2.message._EventMessage)
f6a5e476 329 raise bt2.TryAgain
871a292a
SM
330 else:
331 pass
f6a5e476
PP
332
333 comp_self._at += 1
334
871a292a 335 def _graph_is_configured(self):
692f1a01
PP
336 self._msg_iter = self._create_input_port_message_iterator(
337 self._input_port
338 )
f6a5e476
PP
339
340 src = self._graph.add_component(MySource, 'src')
341 sink = self._graph.add_component(MySink, 'sink')
61d96b89
FD
342 conn = self._graph.connect_ports(
343 src.output_ports['out'], sink.input_ports['in']
344 )
f6a5e476
PP
345
346 with self.assertRaises(bt2.TryAgain):
347 self._graph.run()
348
f6a5e476 349 def test_run_error(self):
871a292a 350 raised_in_sink = False
f6a5e476 351
871a292a 352 class MyIter(_MyIter):
f6a5e476 353 def __next__(self):
871a292a
SM
354 # If this gets called after the sink raised an exception, it is
355 # an error.
356 nonlocal raised_in_sink
357 assert raised_in_sink is False
358
359 if self._at == 0:
360 msg = self._create_stream_beginning_message(self._stream)
361 elif self._at == 1:
362 msg = self._create_packet_beginning_message(self._packet)
363 elif self._at == 2 or self._at == 3:
364 msg = self._create_event_message(self._ec, self._packet)
365 else:
f6a5e476 366 raise bt2.TryAgain
f6a5e476 367 self._at += 1
fa4c33e3 368 return msg
f6a5e476 369
61d96b89 370 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
371 def __init__(self, params):
372 self._add_output_port('out')
373
374 class MySink(bt2._UserSinkComponent):
375 def __init__(self, params):
871a292a 376 self._input_port = self._add_input_port('in')
f6a5e476
PP
377 self._at = 0
378
379 def _consume(comp_self):
871a292a 380 msg = next(comp_self._msg_iter)
f6a5e476 381 if comp_self._at == 0:
871a292a 382 self.assertIsInstance(msg, bt2.message._StreamBeginningMessage)
f6a5e476 383 elif comp_self._at == 1:
871a292a
SM
384 self.assertIsInstance(msg, bt2.message._PacketBeginningMessage)
385 elif comp_self._at == 2:
386 self.assertIsInstance(msg, bt2.message._EventMessage)
387 elif comp_self._at == 3:
388 nonlocal raised_in_sink
389 raised_in_sink = True
f6a5e476
PP
390 raise RuntimeError('error!')
391
392 comp_self._at += 1
393
871a292a 394 def _graph_is_configured(self):
692f1a01
PP
395 self._msg_iter = self._create_input_port_message_iterator(
396 self._input_port
397 )
f6a5e476
PP
398
399 src = self._graph.add_component(MySource, 'src')
400 sink = self._graph.add_component(MySink, 'sink')
61d96b89
FD
401 conn = self._graph.connect_ports(
402 src.output_ports['out'], sink.input_ports['in']
403 )
f6a5e476 404
614743a5 405 with self.assertRaises(bt2._Error):
f6a5e476
PP
406 self._graph.run()
407
871a292a 408 def test_listeners(self):
fa4c33e3 409 class MyIter(bt2._UserMessageIterator):
871a292a
SM
410 def __next__(self):
411 raise bt2.Stop
9ef22b36 412
61d96b89 413 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
9ef22b36
PP
414 def __init__(self, params):
415 self._add_output_port('out')
871a292a 416 self._add_output_port('zero')
9ef22b36
PP
417
418 class MySink(bt2._UserSinkComponent):
419 def __init__(self, params):
420 self._add_input_port('in')
9ef22b36 421
871a292a
SM
422 def _consume(self):
423 raise bt2.Stop
9ef22b36 424
8a08af82
SM
425 def _graph_is_configured(self):
426 pass
427
871a292a
SM
428 def _port_connected(self, port, other_port):
429 self._add_input_port('taste')
9ef22b36 430
871a292a
SM
431 def port_added_listener(component, port):
432 nonlocal calls
433 calls.append((port_added_listener, component, port))
9ef22b36 434
61d96b89
FD
435 def ports_connected_listener(
436 upstream_component, upstream_port, downstream_component, downstream_port
437 ):
871a292a 438 nonlocal calls
61d96b89
FD
439 calls.append(
440 (
441 ports_connected_listener,
442 upstream_component,
443 upstream_port,
444 downstream_component,
445 downstream_port,
446 )
447 )
871a292a
SM
448
449 calls = []
450 self._graph.add_port_added_listener(port_added_listener)
451 self._graph.add_ports_connected_listener(ports_connected_listener)
9ef22b36
PP
452 src = self._graph.add_component(MySource, 'src')
453 sink = self._graph.add_component(MySink, 'sink')
61d96b89 454 self._graph.connect_ports(src.output_ports['out'], sink.input_ports['in'])
9ef22b36 455
871a292a
SM
456 self.assertEqual(len(calls), 5)
457
458 self.assertIs(calls[0][0], port_added_listener)
459 self.assertEqual(calls[0][1].name, 'src')
460 self.assertEqual(calls[0][2].name, 'out')
461
462 self.assertIs(calls[1][0], port_added_listener)
463 self.assertEqual(calls[1][1].name, 'src')
464 self.assertEqual(calls[1][2].name, 'zero')
465
466 self.assertIs(calls[2][0], port_added_listener)
467 self.assertEqual(calls[2][1].name, 'sink')
468 self.assertEqual(calls[2][2].name, 'in')
469
470 self.assertIs(calls[3][0], port_added_listener)
471 self.assertEqual(calls[3][1].name, 'sink')
472 self.assertEqual(calls[3][2].name, 'taste')
473
474 self.assertIs(calls[4][0], ports_connected_listener)
475 self.assertEqual(calls[4][1].name, 'src')
476 self.assertEqual(calls[4][2].name, 'out')
477 self.assertEqual(calls[4][3].name, 'sink')
478 self.assertEqual(calls[4][4].name, 'in')
479
480 def test_invalid_listeners(self):
fa4c33e3 481 class MyIter(bt2._UserMessageIterator):
f6a5e476
PP
482 def __next__(self):
483 raise bt2.Stop
484
61d96b89 485 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
f6a5e476
PP
486 def __init__(self, params):
487 self._add_output_port('out')
488 self._add_output_port('zero')
489
f6a5e476
PP
490 class MySink(bt2._UserSinkComponent):
491 def __init__(self, params):
492 self._add_input_port('in')
493
494 def _consume(self):
495 raise bt2.Stop
496
8a08af82
SM
497 def _graph_is_configured(self):
498 pass
499
f6a5e476
PP
500 def _port_connected(self, port, other_port):
501 self._add_input_port('taste')
502
871a292a
SM
503 with self.assertRaises(TypeError):
504 self._graph.add_port_added_listener(1234)
505 with self.assertRaises(TypeError):
506 self._graph.add_ports_connected_listener(1234)
f6a5e476 507
871a292a
SM
508 def test_raise_in_component_init(self):
509 class MySink(bt2._UserSinkComponent):
510 def __init__(self, params):
511 raise ValueError('oops!')
f6a5e476 512
871a292a
SM
513 def _consume(self):
514 raise bt2.Stop
515
8a08af82
SM
516 def _graph_is_configured(self):
517 pass
518
871a292a
SM
519 graph = bt2.Graph()
520
614743a5 521 with self.assertRaises(bt2._Error):
871a292a
SM
522 graph.add_component(MySink, 'comp')
523
524 def test_raise_in_port_added_listener(self):
525 class MySink(bt2._UserSinkComponent):
526 def __init__(self, params):
527 self._add_input_port('in')
528
529 def _consume(self):
530 raise bt2.Stop
531
8a08af82
SM
532 def _graph_is_configured(self):
533 pass
534
871a292a
SM
535 def port_added_listener(component, port):
536 raise ValueError('oh noes!')
537
538 graph = bt2.Graph()
539 graph.add_port_added_listener(port_added_listener)
540
614743a5 541 with self.assertRaises(bt2._Error):
871a292a
SM
542 graph.add_component(MySink, 'comp')
543
544 def test_raise_in_ports_connected_listener(self):
545 class MyIter(bt2._UserMessageIterator):
546 def __next__(self):
547 raise bt2.Stop
548
61d96b89 549 class MySource(bt2._UserSourceComponent, message_iterator_class=MyIter):
871a292a
SM
550 def __init__(self, params):
551 self._add_output_port('out')
552
553 class MySink(bt2._UserSinkComponent):
554 def __init__(self, params):
555 self._add_input_port('in')
556
557 def _consume(self):
558 raise bt2.Stop
f6a5e476 559
8a08af82
SM
560 def _graph_is_configured(self):
561 pass
562
61d96b89
FD
563 def ports_connected_listener(
564 upstream_component, upstream_port, downstream_component, downstream_port
565 ):
871a292a 566 raise ValueError('oh noes!')
f6a5e476 567
871a292a
SM
568 graph = bt2.Graph()
569 graph.add_ports_connected_listener(ports_connected_listener)
570 up = graph.add_component(MySource, 'down')
571 down = graph.add_component(MySink, 'up')
f6a5e476 572
614743a5 573 with self.assertRaises(bt2._Error):
871a292a 574 graph.connect_ports(up.output_ports['out'], down.input_ports['in'])
This page took 0.061193 seconds and 4 git commands to generate.