# -*- python-mode -*- ## Copyright (C) 2012-2013 Daniel Pavel ## ## This program is free software; you can redistribute it and/or modify ## it under the terms of the GNU General Public License as published by ## the Free Software Foundation; either version 2 of the License, or ## (at your option) any later version. ## ## This program is distributed in the hope that it will be useful, ## but WITHOUT ANY WARRANTY; without even the implied warranty of ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ## GNU General Public License for more details. ## ## You should have received a copy of the GNU General Public License along ## with this program; if not, write to the Free Software Foundation, Inc., ## 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. # Some common functions and types. from binascii import hexlify as _hexlify from collections import namedtuple from struct import pack, unpack try: unicode # noqa: F821 # if Python2, unicode_literals will mess our first (un)pack() argument _pack_str = pack _unpack_str = unpack pack = lambda x, *args: _pack_str(str(x), *args) unpack = lambda x, *args: _unpack_str(str(x), *args) is_string = lambda d: isinstance(d, unicode) or isinstance(d, str) # noqa: F821 # no easy way to distinguish between b'' and '' :( # or (isinstance(d, str) \ # and not any((chr(k) in d for k in range(0x00, 0x1F))) \ # and not any((chr(k) in d for k in range(0x80, 0xFF))) \ # ) except Exception: # this is certainly Python 3 # In Py3, unicode and str are equal (the unicode object does not exist) is_string = lambda d: isinstance(d, str) # # # class NamedInt(int): """A regular Python integer with an attached name. Caution: comparison with strings will also match this NamedInt's name (case-insensitive).""" def __new__(cls, value, name): assert is_string(name) obj = int.__new__(cls, value) obj.name = str(name) return obj def bytes(self, count=2): return int2bytes(self, count) def __eq__(self, other): if isinstance(other, NamedInt): return int(self) == int(other) and self.name == other.name if isinstance(other, int): return int(self) == int(other) if is_string(other): return self.name.lower() == other.lower() # this should catch comparisons with bytes in Py3 if other is not None: raise TypeError('Unsupported type ' + str(type(other))) def __ne__(self, other): return not self.__eq__(other) def __hash__(self): return int(self) def __str__(self): return self.name __unicode__ = __str__ def __repr__(self): return 'NamedInt(%d, %r)' % (int(self), self.name) class NamedInts(object): """An ordered set of NamedInt values. Indexing can be made by int or string, and will return the corresponding NamedInt if it exists in this set, or `None`. Extracting slices will return all present NamedInts in the given interval (extended slices are not supported). Assigning a string to an indexed int will create a new NamedInt in this set; if the value already exists in the set (int or string), ValueError will be raised. """ __slots__ = ('__dict__', '_values', '_indexed', '_fallback') def __init__(self, **kwargs): def _readable_name(n): if not is_string(n): raise TypeError('expected (unicode) string, got ' + str(type(n))) return n.replace('__', '/').replace('_', ' ') # print (repr(kwargs)) values = {k: NamedInt(v, _readable_name(k)) for (k, v) in kwargs.items()} self.__dict__ = values self._values = sorted(list(values.values())) self._indexed = {int(v): v for v in self._values} # assert len(values) == len(self._indexed) # "(%d) %r\n=> (%d) %r" % (len(values), values, len(self._indexed), self._indexed) self._fallback = None @classmethod def list(cls, items, name_generator=lambda x: str(x)): values = {name_generator(x): x for x in items} return NamedInts(**values) @classmethod def range(cls, from_value, to_value, name_generator=lambda x: str(x), step=1): values = {name_generator(x): x for x in range(from_value, to_value + 1, step)} return NamedInts(**values) def flag_names(self, value): unknown_bits = value for k in self._indexed: assert bin(k).count('1') == 1 if k & value == k: unknown_bits &= ~k yield str(self._indexed[k]) if unknown_bits: yield 'unknown:%06X' % unknown_bits def __getitem__(self, index): if isinstance(index, int): if index in self._indexed: return self._indexed[int(index)] if self._fallback and isinstance(index, int): value = NamedInt(index, self._fallback(index)) self._indexed[index] = value self._values = sorted(self._values + [value]) return value elif is_string(index): if index in self.__dict__: return self.__dict__[index] return (next((x for x in self._values if str(x) == index), None)) elif isinstance(index, slice): if index.start is None and index.stop is None: return self._values[:] v_start = int(self._values[0]) if index.start is None else int(index.start) v_stop = (self._values[-1] + 1) if index.stop is None else int(index.stop) if v_start > v_stop or v_start > self._values[-1] or v_stop <= self._values[0]: return [] if v_start <= self._values[0] and v_stop > self._values[-1]: return self._values[:] start_index = 0 stop_index = len(self._values) for i, value in enumerate(self._values): if value < v_start: start_index = i + 1 elif index.stop is None: break if value >= v_stop: stop_index = i break return self._values[start_index:stop_index] def __setitem__(self, index, name): assert isinstance(index, int), type(index) if isinstance(name, NamedInt): assert int(index) == int(name), repr(index) + ' ' + repr(name) value = name elif is_string(name): value = NamedInt(index, name) else: raise TypeError('name must be a string') if str(value) in self.__dict__: raise ValueError('%s (%d) already known' % (value, int(value))) if int(value) in self._indexed: raise ValueError('%d (%s) already known' % (int(value), value)) self._values = sorted(self._values + [value]) self.__dict__[str(value)] = value self._indexed[int(value)] = value def __contains__(self, value): if isinstance(value, int): return value in self._indexed elif is_string(value): return value in self.__dict__ or value in self._values def __iter__(self): for v in self._values: yield v def __len__(self): return len(self._values) def __repr__(self): return 'NamedInts(%s)' % ', '.join(repr(v) for v in self._values) def strhex(x): assert x is not None """Produce a hex-string representation of a sequence of bytes.""" return _hexlify(x).decode('ascii').upper() def bytes2int(x): """Convert a bytes string to an int. The bytes are assumed to be in most-significant-first order. """ assert isinstance(x, bytes) assert len(x) < 9 qx = (b'\x00' * 8) + x result, = unpack('!Q', qx[-8:]) # assert x == int2bytes(result, len(x)) return result def int2bytes(x, count=None): """Convert an int to a bytes representation. The bytes are ordered in most-significant-first order. If 'count' is not given, the necessary number of bytes is computed. """ assert isinstance(x, int) result = pack('!Q', x) assert isinstance(result, bytes) # assert x == bytes2int(result) if count is None: return result.lstrip(b'\x00') assert isinstance(count, int) assert count > 0 assert x.bit_length() <= count * 8 return result[-count:] class KwException(Exception): """An exception that remembers all arguments passed to the constructor. They can be later accessed by simple member access. """ def __init__(self, **kwargs): super(KwException, self).__init__(kwargs) def __getattr__(self, k): try: return super(KwException, self).__getattr__(k) except AttributeError: return self.args[0][k] """Firmware information.""" FirmwareInfo = namedtuple('FirmwareInfo', ['kind', 'name', 'version', 'extras']) BATTERY_APPROX = NamedInts(empty=0, critical=5, low=20, good=50, full=90) del namedtuple