bt2: add `IntegerRangeSet` const classes
[babeltrace.git] / src / bindings / python / bt2 / bt2 / integer_range_set.py
1 # The MIT License (MIT)
2 #
3 # Copyright (c) 2017 Philippe Proulx <pproulx@efficios.com>
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
24 import collections.abc
25 import bt2
26
27
28 class _IntegerRangeConst:
29 def __init__(self, lower, upper=None):
30 self._check_type(lower)
31
32 if upper is None:
33 upper = lower
34
35 self._check_type(upper)
36
37 if lower > upper:
38 raise ValueError(
39 "range's lower bound ({}) is greater than its upper bound ({})".format(
40 lower, upper
41 )
42 )
43
44 self._lower = lower
45 self._upper = upper
46
47 @property
48 def lower(self):
49 return self._lower
50
51 @property
52 def upper(self):
53 return self._upper
54
55 def contains(self, value):
56 self._check_type(value)
57 return value >= self._lower and value <= self._upper
58
59 def __eq__(self, other):
60 if not isinstance(other, _IntegerRangeConst):
61 return False
62
63 return self.lower == other.lower and self.upper == other.upper
64
65
66 class _IntegerRange(_IntegerRangeConst):
67 def __init__(self, lower, upper=None):
68 super().__init__(lower, upper)
69
70
71 class _SignedIntegerRangeConst(_IntegerRangeConst):
72 _is_type = staticmethod(utils._is_int64)
73 _check_type = staticmethod(utils._check_int64)
74
75
76 class SignedIntegerRange(_SignedIntegerRangeConst, _IntegerRange):
77 pass
78
79
80 class _UnsignedIntegerRangeConst(_IntegerRangeConst):
81 _is_type = staticmethod(utils._is_uint64)
82 _check_type = staticmethod(utils._check_uint64)
83
84
85 class UnsignedIntegerRange(_UnsignedIntegerRangeConst, _IntegerRange):
86 pass
87
88
89 class _IntegerRangeSetConst(object._SharedObject, collections.abc.Set):
90 def __len__(self):
91 range_set_ptr = self._as_range_set_ptr(self._ptr)
92 count = native_bt.integer_range_set_get_range_count(range_set_ptr)
93 assert count >= 0
94 return count
95
96 def __contains__(self, other_range):
97 for rg in self:
98 if rg == other_range:
99 return True
100
101 return False
102
103 def __iter__(self):
104 for idx in range(len(self)):
105 rg_ptr = self._borrow_range_ptr_by_index(self._ptr, idx)
106 assert rg_ptr is not None
107 lower = self._range_get_lower(rg_ptr)
108 upper = self._range_get_upper(rg_ptr)
109 yield self._range_pycls(lower, upper)
110
111 def __eq__(self, other):
112
113 if not isinstance(other, _IntegerRangeSetConst):
114 return False
115
116 return self._compare(self._ptr, other._ptr)
117
118 def contains_value(self, value):
119 for rg in self:
120 if rg.contains(value):
121 return True
122
123 return False
124
125
126 class _IntegerRangeSet(_IntegerRangeSetConst, collections.abc.MutableSet):
127 def __init__(self, ranges=None):
128 ptr = self._create_range_set()
129
130 if ptr is None:
131 raise bt2._MemoryError('cannot create range set object')
132
133 super().__init__(ptr)
134
135 if ranges is not None:
136 # will raise if not iterable
137 for rg in ranges:
138 self.add(rg)
139
140 def add(self, rg):
141 if type(rg) is not self._range_pycls:
142 if self._range_pycls._is_type(rg):
143 rg = self._range_pycls(rg)
144 else:
145 # assume it's a simple pair (will raise if it's not)
146 rg = self._range_pycls(rg[0], rg[1])
147
148 status = self._add_range(self._ptr, rg.lower, rg.upper)
149 utils._handle_func_status(status, 'cannot add range to range set object')
150
151 def discard(self, rg):
152 raise NotImplementedError
153
154
155 class _SignedIntegerRangeSetConst(_IntegerRangeSetConst):
156 _get_ref = staticmethod(native_bt.integer_range_set_signed_get_ref)
157 _put_ref = staticmethod(native_bt.integer_range_set_signed_put_ref)
158 _as_range_set_ptr = staticmethod(
159 native_bt.integer_range_set_signed_as_range_set_const
160 )
161 _borrow_range_ptr_by_index = staticmethod(
162 native_bt.integer_range_set_signed_borrow_range_by_index_const
163 )
164 _range_get_lower = staticmethod(native_bt.integer_range_signed_get_lower)
165 _range_get_upper = staticmethod(native_bt.integer_range_signed_get_upper)
166 _compare = staticmethod(native_bt.integer_range_set_signed_compare)
167 _range_pycls = _SignedIntegerRangeConst
168
169
170 class SignedIntegerRangeSet(_SignedIntegerRangeSetConst, _IntegerRangeSet):
171 _create_range_set = staticmethod(native_bt.integer_range_set_signed_create)
172 _add_range = staticmethod(native_bt.integer_range_set_signed_add_range)
173 _range_pycls = SignedIntegerRange
174
175
176 class _UnsignedIntegerRangeSetConst(_IntegerRangeSetConst):
177 _get_ref = staticmethod(native_bt.integer_range_set_unsigned_get_ref)
178 _put_ref = staticmethod(native_bt.integer_range_set_unsigned_put_ref)
179 _as_range_set_ptr = staticmethod(
180 native_bt.integer_range_set_unsigned_as_range_set_const
181 )
182 _borrow_range_ptr_by_index = staticmethod(
183 native_bt.integer_range_set_unsigned_borrow_range_by_index_const
184 )
185 _range_get_lower = staticmethod(native_bt.integer_range_unsigned_get_lower)
186 _range_get_upper = staticmethod(native_bt.integer_range_unsigned_get_upper)
187 _compare = staticmethod(native_bt.integer_range_set_unsigned_compare)
188 _range_pycls = _UnsignedIntegerRangeConst
189
190
191 class UnsignedIntegerRangeSet(_UnsignedIntegerRangeSetConst, _IntegerRangeSet):
192 _create_range_set = staticmethod(native_bt.integer_range_set_unsigned_create)
193 _add_range = staticmethod(native_bt.integer_range_set_unsigned_add_range)
194 _range_pycls = UnsignedIntegerRange
This page took 0.03709 seconds and 5 git commands to generate.