Commit | Line | Data |
---|---|---|
81447b5b PP |
1 | # The MIT License (MIT) |
2 | # | |
f6a5e476 | 3 | # Copyright (c) 2017 Philippe Proulx <pproulx@efficios.com> |
81447b5b PP |
4 | # |
5 | # Permission is hereby granted, free of charge, to any person obtaining a copy | |
6 | # of this software and associated documentation files (the "Software"), to deal | |
7 | # in the Software without restriction, including without limitation the rights | |
8 | # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | |
9 | # copies of the Software, and to permit persons to whom the Software is | |
10 | # furnished to do so, subject to the following conditions: | |
11 | # | |
12 | # The above copyright notice and this permission notice shall be included in | |
13 | # all copies or substantial portions of the Software. | |
14 | # | |
15 | # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
16 | # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
17 | # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
18 | # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
19 | # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | |
20 | # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | |
21 | # THE SOFTWARE. | |
22 | ||
23 | from bt2 import native_bt, object, utils | |
81447b5b | 24 | import collections.abc |
c946c9de PP |
25 | from bt2 import value as bt2_value |
26 | from bt2 import stream as bt2_stream | |
c946c9de | 27 | from bt2 import stream_class as bt2_stream_class |
81447b5b | 28 | import bt2 |
f377a958 | 29 | import functools |
cd03c43c PP |
30 | import uuid as uuidp |
31 | ||
32 | ||
7868a02e SM |
33 | def _bt2_trace_class(): |
34 | from bt2 import trace_class as bt2_trace_class | |
35 | ||
36 | return bt2_trace_class | |
37 | ||
38 | ||
9cbe0c59 FD |
39 | class _TraceEnvironmentConst(collections.abc.Mapping): |
40 | _create_value_from_ptr_and_get_ref = staticmethod( | |
41 | bt2_value._create_from_const_ptr_and_get_ref | |
42 | ) | |
43 | ||
cd03c43c PP |
44 | def __init__(self, trace): |
45 | self._trace = trace | |
46 | ||
47 | def __getitem__(self, key): | |
48 | utils._check_str(key) | |
49 | ||
50 | borrow_entry_fn = native_bt.trace_borrow_environment_entry_value_by_name_const | |
51 | value_ptr = borrow_entry_fn(self._trace._ptr, key) | |
52 | ||
53 | if value_ptr is None: | |
54 | raise KeyError(key) | |
55 | ||
9cbe0c59 | 56 | return self._create_value_from_ptr_and_get_ref(value_ptr) |
cd03c43c PP |
57 | |
58 | def __len__(self): | |
59 | count = native_bt.trace_get_environment_entry_count(self._trace._ptr) | |
60 | assert count >= 0 | |
61 | return count | |
62 | ||
63 | def __iter__(self): | |
9cbe0c59 | 64 | trace_ptr = self._trace._ptr |
cd03c43c PP |
65 | |
66 | for idx in range(len(self)): | |
67 | borrow_entry_fn = native_bt.trace_borrow_environment_entry_by_index_const | |
68 | entry_name, _ = borrow_entry_fn(trace_ptr, idx) | |
69 | assert entry_name is not None | |
70 | yield entry_name | |
81447b5b PP |
71 | |
72 | ||
9cbe0c59 FD |
73 | class _TraceEnvironment(_TraceEnvironmentConst, collections.abc.MutableMapping): |
74 | _create_value_from_ptr_and_get_ref = staticmethod( | |
75 | bt2_value._create_from_ptr_and_get_ref | |
76 | ) | |
77 | ||
78 | def __setitem__(self, key, value): | |
79 | if isinstance(value, str): | |
80 | set_env_entry_fn = native_bt.trace_set_environment_entry_string | |
81 | elif isinstance(value, int): | |
82 | set_env_entry_fn = native_bt.trace_set_environment_entry_integer | |
83 | else: | |
84 | raise TypeError('expected str or int, got {}'.format(type(value))) | |
81447b5b | 85 | |
9cbe0c59 FD |
86 | status = set_env_entry_fn(self._trace._ptr, key, value) |
87 | utils._handle_func_status(status, "cannot set trace object's environment entry") | |
81447b5b | 88 | |
9cbe0c59 FD |
89 | def __delitem__(self, key): |
90 | raise NotImplementedError | |
91 | ||
92 | ||
93 | class _TraceConst(object._SharedObject, collections.abc.Mapping): | |
f377a958 SM |
94 | _get_ref = staticmethod(native_bt.trace_get_ref) |
95 | _put_ref = staticmethod(native_bt.trace_put_ref) | |
9cbe0c59 FD |
96 | _borrow_stream_ptr_by_id = staticmethod(native_bt.trace_borrow_stream_by_id_const) |
97 | _borrow_stream_ptr_by_index = staticmethod( | |
98 | native_bt.trace_borrow_stream_by_index_const | |
99 | ) | |
100 | _borrow_class_ptr = staticmethod(native_bt.trace_borrow_class_const) | |
101 | _borrow_user_attributes_ptr = staticmethod( | |
102 | native_bt.trace_borrow_user_attributes_const | |
103 | ) | |
104 | _create_value_from_ptr_and_get_ref = staticmethod( | |
105 | bt2_value._create_from_const_ptr_and_get_ref | |
106 | ) | |
107 | _stream_pycls = property(lambda _: bt2_stream._StreamConst) | |
7868a02e | 108 | _trace_class_pycls = property(lambda _: _bt2_trace_class()._TraceClassConst) |
9cbe0c59 | 109 | _trace_env_pycls = property(lambda _: _TraceEnvironmentConst) |
f6a5e476 PP |
110 | |
111 | def __len__(self): | |
f377a958 SM |
112 | count = native_bt.trace_get_stream_count(self._ptr) |
113 | assert count >= 0 | |
f6a5e476 PP |
114 | return count |
115 | ||
f377a958 SM |
116 | def __getitem__(self, id): |
117 | utils._check_uint64(id) | |
81447b5b | 118 | |
9cbe0c59 | 119 | stream_ptr = self._borrow_stream_ptr_by_id(self._ptr, id) |
81447b5b | 120 | |
f377a958 SM |
121 | if stream_ptr is None: |
122 | raise KeyError(id) | |
81447b5b | 123 | |
9cbe0c59 | 124 | return self._stream_pycls._create_from_ptr_and_get_ref(stream_ptr) |
81447b5b PP |
125 | |
126 | def __iter__(self): | |
f377a958 | 127 | for idx in range(len(self)): |
9cbe0c59 | 128 | stream_ptr = self._borrow_stream_ptr_by_index(self._ptr, idx) |
f377a958 | 129 | assert stream_ptr is not None |
81447b5b | 130 | |
f377a958 SM |
131 | id = native_bt.stream_get_id(stream_ptr) |
132 | assert id >= 0 | |
81447b5b | 133 | |
f377a958 | 134 | yield id |
81447b5b | 135 | |
a7f0580d PP |
136 | @property |
137 | def cls(self): | |
9cbe0c59 | 138 | trace_class_ptr = self._borrow_class_ptr(self._ptr) |
a7f0580d | 139 | assert trace_class_ptr is not None |
9cbe0c59 | 140 | return self._trace_class_pycls._create_from_ptr_and_get_ref(trace_class_ptr) |
a7f0580d | 141 | |
b2df5780 PP |
142 | @property |
143 | def user_attributes(self): | |
9cbe0c59 | 144 | ptr = self._borrow_user_attributes_ptr(self._ptr) |
b2df5780 | 145 | assert ptr is not None |
9cbe0c59 | 146 | return self._create_value_from_ptr_and_get_ref(ptr) |
b2df5780 | 147 | |
81447b5b PP |
148 | @property |
149 | def name(self): | |
839d52a5 | 150 | return native_bt.trace_get_name(self._ptr) |
81447b5b | 151 | |
9cbe0c59 FD |
152 | @property |
153 | def uuid(self): | |
154 | uuid_bytes = native_bt.trace_get_uuid(self._ptr) | |
155 | if uuid_bytes is None: | |
156 | return | |
157 | ||
158 | return uuidp.UUID(bytes=uuid_bytes) | |
159 | ||
160 | @property | |
161 | def environment(self): | |
162 | return self._trace_env_pycls(self) | |
163 | ||
164 | def add_destruction_listener(self, listener): | |
165 | '''Add a listener to be called when the trace is destroyed.''' | |
166 | if not callable(listener): | |
167 | raise TypeError("'listener' parameter is not callable") | |
168 | ||
169 | fn = native_bt.bt2_trace_add_destruction_listener | |
170 | listener_from_native = functools.partial( | |
171 | _trace_destruction_listener_from_native, listener | |
172 | ) | |
173 | ||
174 | status, listener_id = fn(self._ptr, listener_from_native) | |
175 | utils._handle_func_status( | |
176 | status, 'cannot add destruction listener to trace object' | |
177 | ) | |
178 | ||
179 | return utils._ListenerHandle(listener_id, self) | |
180 | ||
ab778b5c SM |
181 | def remove_destruction_listener(self, listener_handle): |
182 | utils._check_type(listener_handle, utils._ListenerHandle) | |
183 | ||
184 | if listener_handle._obj.addr != self.addr: | |
185 | raise ValueError( | |
186 | 'This trace destruction listener does not match the trace object.' | |
187 | ) | |
188 | ||
189 | if listener_handle._listener_id is None: | |
190 | raise ValueError('This trace destruction listener was already removed.') | |
191 | ||
192 | status = native_bt.trace_remove_destruction_listener( | |
193 | self._ptr, listener_handle._listener_id | |
194 | ) | |
195 | utils._handle_func_status(status) | |
196 | listener_handle._listener_id = None | |
197 | ||
9cbe0c59 FD |
198 | |
199 | class _Trace(_TraceConst): | |
200 | _borrow_stream_ptr_by_id = staticmethod(native_bt.trace_borrow_stream_by_id) | |
201 | _borrow_stream_ptr_by_index = staticmethod(native_bt.trace_borrow_stream_by_index) | |
202 | _borrow_class_ptr = staticmethod(native_bt.trace_borrow_class) | |
203 | _borrow_user_attributes_ptr = staticmethod(native_bt.trace_borrow_user_attributes) | |
204 | _create_value_from_ptr_and_get_ref = staticmethod( | |
205 | bt2_value._create_from_ptr_and_get_ref | |
206 | ) | |
207 | _stream_pycls = property(lambda _: bt2_stream._Stream) | |
7868a02e | 208 | _trace_class_pycls = property(lambda _: _bt2_trace_class()._TraceClass) |
9cbe0c59 FD |
209 | _trace_env_pycls = property(lambda _: _TraceEnvironment) |
210 | ||
f377a958 | 211 | def _name(self, name): |
81447b5b | 212 | utils._check_str(name) |
fb25b9e3 | 213 | status = native_bt.trace_set_name(self._ptr, name) |
61d96b89 | 214 | utils._handle_func_status(status, "cannot set trace class object's name") |
81447b5b | 215 | |
f377a958 | 216 | _name = property(fset=_name) |
f6a5e476 | 217 | |
9cbe0c59 FD |
218 | def _user_attributes(self, user_attributes): |
219 | value = bt2_value.create_value(user_attributes) | |
220 | utils._check_type(value, bt2_value.MapValue) | |
221 | native_bt.trace_set_user_attributes(self._ptr, value._ptr) | |
cd03c43c | 222 | |
9cbe0c59 | 223 | _user_attributes = property(fset=_user_attributes) |
cd03c43c PP |
224 | |
225 | def _uuid(self, uuid): | |
226 | utils._check_type(uuid, uuidp.UUID) | |
227 | native_bt.trace_set_uuid(self._ptr, uuid.bytes) | |
228 | ||
229 | _uuid = property(fset=_uuid) | |
230 | ||
b2df5780 | 231 | def create_stream(self, stream_class, id=None, name=None, user_attributes=None): |
c946c9de | 232 | utils._check_type(stream_class, bt2_stream_class._StreamClass) |
81447b5b | 233 | |
f377a958 SM |
234 | if stream_class.assigns_automatic_stream_id: |
235 | if id is not None: | |
61d96b89 FD |
236 | raise ValueError( |
237 | "id provided, but stream class assigns automatic stream ids" | |
238 | ) | |
81447b5b | 239 | |
f377a958 SM |
240 | stream_ptr = native_bt.stream_create(stream_class._ptr, self._ptr) |
241 | else: | |
242 | if id is None: | |
61d96b89 FD |
243 | raise ValueError( |
244 | "id not provided, but stream class does not assign automatic stream ids" | |
245 | ) | |
81447b5b | 246 | |
f377a958 | 247 | utils._check_uint64(id) |
61d96b89 FD |
248 | stream_ptr = native_bt.stream_create_with_id( |
249 | stream_class._ptr, self._ptr, id | |
250 | ) | |
81447b5b | 251 | |
f377a958 | 252 | if stream_ptr is None: |
614743a5 | 253 | raise bt2._MemoryError('cannot create stream object') |
81447b5b | 254 | |
c946c9de | 255 | stream = bt2_stream._Stream._create_from_ptr(stream_ptr) |
81447b5b | 256 | |
f377a958 SM |
257 | if name is not None: |
258 | stream._name = name | |
81447b5b | 259 | |
b2df5780 PP |
260 | if user_attributes is not None: |
261 | stream._user_attributes = user_attributes | |
262 | ||
f377a958 | 263 | return stream |
81447b5b | 264 | |
81447b5b | 265 | |
9cbe0c59 FD |
266 | def _trace_destruction_listener_from_native(user_listener, trace_ptr): |
267 | trace = _TraceConst._create_from_ptr_and_get_ref(trace_ptr) | |
268 | user_listener(trace) |