# Copyright (C) 2015 Chintalagiri Shashank
#
# This file is part of Tendril.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as published by
# the Free Software Foundation, either version 3 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
Base Unit Types (:mod:`tendril.utils.types.unitbase`)
=====================================================
The Types provided in this module are not intended for direct use. Instead,
they provide reusable primitives which can be sub-classed to provide
functional Types.
Ideally, all Unit classes should derive from the :class:`UnitBase` class
provided here. In practice, only the newer Types follow this inheritance,
while the older ones still need to be migrated to this form.
.. rubric:: Module Contents
.. autosummary::
UnitBase
NumericalUnitBase
DummyUnit
parse_none
parse_percent
Percentage
.. seealso:: :mod:`tendril.utils.types`, for an overview applicable to
most types defined in Tendril.
"""
import re
import six
import numbers
from math import log10
from math import floor
from decimal import Decimal
from decimal import InvalidOperation
from fractions import Fraction
from . import ParseException
[docs]def round_to_n(x, n):
if x:
return round(x, -int(floor(log10(x))) + (n - 1))
return 0
[docs]def remove_exponent(num):
return num.to_integral() if num == num.to_integral() else num.normalize()
[docs]class TypedComparisonMixin(object):
"""
This mixin allows implementing comparison operators in a Python 3
compatible way.
Two instances of a class are compared using their :func:`_cmpkey`
methods. If the instances have a different ``__class__``, the
comparison is not implemented. A single exception is implemented,
for when the other instance is of a numerical type, with value 0.
"""
[docs] def _compare(self, other, method):
if self.__class__ != other.__class__:
if other is None:
return method(self._cmpkey(), 0)
if not isinstance(other, numbers.Number) or other != 0:
raise TypeError(
"Comparison of : " + repr(self) + ", " + repr(other)
)
else:
return method(self._cmpkey(), other)
return method(self._cmpkey(), other._cmpkey())
def __lt__(self, other):
return self._compare(other, lambda s, o: s < o)
def __le__(self, other):
return self._compare(other, lambda s, o: s <= o)
[docs] def __eq__(self, other):
return self._compare(other, lambda s, o: s == o)
def __ge__(self, other):
return self._compare(other, lambda s, o: s >= o)
def __gt__(self, other):
return self._compare(other, lambda s, o: s > o)
def __ne__(self, other):
return self._compare(other, lambda s, o: s != o)
[docs] def _cmpkey(self):
raise NotImplementedError
[docs]class UnitBase(object):
"""
The base class for all :mod:`tendril.utils.types` units.
When instantiated, the `value` param is processed as follows :
- :class:`str` value is passed though `_parse_func`, and whatever it
returns is stored.
- :class:`numbers.Number` values are first converted into
:class:`decimal.Decimal` and then stored.
- All other `value` types are simply stored as is.
:param value: The core value to be stored.
:param _dostr: The canonical unit / order string to use.
:param _parse_func: The function used to parse string values into an
actual value in the canonical unit.
.. rubric:: Arithmetic Operations
This class supports no arithmetic operations.
"""
_dostr = None
_parse_func = None
def __init__(self, value):
if isinstance(value, (six.text_type, six.string_types)):
try:
value = self._parse_func(value)
except Exception as e:
raise ParseException(value, e)
if isinstance(value, numbers.Number):
if isinstance(value, Fraction):
value = float(value)
if not isinstance(value, Decimal):
value = Decimal(value)
self._value = value
# def __float__(self):
# return float(self._value)
#
# def __int__(self):
# return int(self._value)
@property
def value(self):
"""
:return: The core value of the Unit instance, in it's canonical unit.
"""
return self._value
def __repr__(self):
return str(self._value) + self._dostr
[docs]class NumericalUnitBase(TypedComparisonMixin, UnitBase):
"""
The base class for all :mod:`tendril.utils.types` numerical units.
Provides the patterns used by the various Numerical Units to provide
their functionality. This class represents and implements the core
ideas that remain valid across Units (for the most part). The various
methods and functions implemented here establish the minimum required
functionality and behaviour expected of all numerical units.
Specific numerical unit classes may override the methods present here
to tweak the implementation and/or the interface as per the
requirements of the quantity they represent, as long as they stay true
to the spirit of the architecture.
:param value: The core value to be stored.
.. seealso:: :class:`UnitBase`
.. rubric:: The `_orders` / `_ostrs` Class Variables
Supported units can be provided as a list of strings (`_ostrs`) or a
list of tuples (`_orders`).
- In case `_ostrs` is provided, it is assumed that each string
represents a unit value 1000 times smaller than the next.
- In case `_orders` is provided, it is assumed that the first
element of each tuple is the string representation of the order,
and the second element is the multiplicative factor relative to
the default order string.
- In both cases, note that first order within which the unit value's
representation lies between 1 and 1000 is used to produce the unit's
string representation. As such, you should place higher priority or
more 'standard' units towards the beginning of the list.
.. rubric:: Arithmetic Operations
.. autosummary::
__add__
__sub__
__mul__
__div__
_cmpkey
"""
_orders = None
_ostrs = None
_osuffix = None
_parse_func = None
_regex_std = None
_allow_nOr = True
_order_type = None
_order_dict = None
_has_bare_order = False
_rorders = None
_pluralize_ostr = False
_separate_unit = False
def __init__(self, value):
if isinstance(value, bytes):
value = value.decode()
if not self._orders:
doidx = self._ostrs.index(self._dostr)
self._orders = [(ostr, (3 * (idx - doidx)))
for idx, ostr in enumerate(self._ostrs)]
self._order_type = 0
elif not self._ostrs:
self._ostrs = [order[0] for order in self._orders]
self._order_type = 1
if not self._order_dict:
self._order_dict = {o[0]: o[1] for o in self._orders}
if self._parse_func is None:
self._parse_func = self._standard_parser
if not self._has_bare_order:
try:
value = Decimal(value)
except InvalidOperation:
pass
super(NumericalUnitBase, self).__init__(value)
[docs] def _standard_parser(self, value):
if not self._regex_std:
raise Exception("Standard parser requires a defined regex")
match = self._regex_std.match(value)
n = Decimal(match.group('numerical'))
r = match.group('residual')
if r:
if not self._allow_nOr:
raise ValueError("nOr structure not allowed for this unit")
r = Decimal(r)
n += r/10
o = match.group('order')
if self._osuffix and not o.endswith(self._osuffix):
o += self._osuffix
try:
if self._order_type:
f = self._order_dict[o]
if isinstance(f, numbers.Number):
return n * f
else:
return f(n)
else:
exp = self._order_dict[o]
if exp >= 0:
return n * (10 ** exp)
else:
# TODO Significant error in precision here
return n / (10 ** abs(exp))
except KeyError:
raise ValueError('Order unrecognized : {0} for {1}'
''.format(o, self.__class__))
def __float__(self):
return float(self._value)
[docs] def __add__(self, other):
"""
Addition of two Unit class instances of the same type returns a
Unit class instance of the same type, with the sum of the two
operands as it's value.
If the other operand is a numerical type and evaluates to 0, this
object is simply returned unchanged.
Addition with all other Types / Classes is not supported.
"""
if isinstance(other, numbers.Number) and other == 0:
return self
elif self.__class__ == other.__class__:
return self.__class__(self._value + other._value)
else:
return NotImplemented
def __radd__(self, other):
if other == 0:
return self
else:
return self.__add__(other)
[docs] def __mul__(self, other):
"""
Multiplication of one Unit type class instance with a numerical type
results in a Unit type class instance of the same type, whose value
is the Unit type operand's value multiplied by the numerical operand's
value.
Multiplication with all other Types / Classes is not supported.
"""
if isinstance(other, numbers.Number):
if isinstance(self, FactorBase):
return self.__class__(other * self.value)
if isinstance(other, Decimal):
return self.__class__(self.value * other)
return self.__class__(self.value * Decimal(other))
if isinstance(other, Percentage):
return self.__class__(self.value * other.value / 100)
if isinstance(other, GainBase):
if isinstance(self, GainBase):
if self._gtype != other._gtype:
raise TypeError("Gain is of a different type.")
return self.__class__(self.value * other.value)
if other._gtype and not isinstance(self, other._gtype[1]):
raise TypeError("Gain is of a different type.")
if isinstance(self, other._gtype[0]):
return self.__class__(self.value * other.value)
else:
return other._gtype[0](self.value * other.value)
else:
return NotImplemented
def __rmul__(self, other):
return self.__mul__(other)
[docs] def __div__(self, other):
"""
Division of one Unit type class instance with a numerical type
results in a Unit type class instance of the same type, whose value
is the Unit type operand's value divided by the numerical operand's
value.
In this case, the first operand must be a Unit type class instance,
and not the reverse.
Division of one Unit type class instance by another of the same type
returns a numerical value, which is obtained by performing the
division with the operands' value.
Division with all other Types / Classes is not supported.
"""
if isinstance(other, numbers.Number):
if isinstance(other, Decimal):
return self.__class__(self.value / other)
else:
return self.__class__(self.value / Decimal(other))
elif isinstance(other, Percentage):
return self.__class__(self.value / other.value * 100)
elif isinstance(other, self.__class__):
return self.value / other.value
else:
return NotImplemented
def __rdiv__(self, other):
return NotImplemented
def __truediv__(self, other):
return self.__div__(other)
def __rtruediv__(self, other):
return self.__rdiv__(other)
[docs] def __sub__(self, other):
"""
Subtraction of two Unit class instances of the same type returns a
Unit class instance of the same type, with the difference of the two
operands as it's value.
If the other operand is a numerical type and evaluates to 0, this
object is simply returned unchanged.
Subtraction with all other Types / Classes is not supported.
"""
if isinstance(other, numbers.Number) and other == 0:
return self
else:
return self.__add__(other.__mul__(-1))
def __rsub__(self, other):
return other.__sub__(self)
def __abs__(self):
if self._value < 0:
return self.__class__(self._value * -1)
else:
return self
[docs] def _cmpkey(self):
"""
The comparison of two Unit type class instances of the
same type behaves identically to the comparison of the
operands' values.
Comparison with all other Types / Classes is not supported.
"""
return self._value
[docs] def _pluralize(self, value, ostr):
if value != 1 and self._pluralize_ostr:
return value, ostr + 's'
else:
return value, ostr
@property
def natural_repr(self):
if self._order_type == 1:
return self._pluralize(self._value, self._dostr)
ostr = self._dostr
value = self._value
done = False
while not done:
ostri = self._ostrs.index(ostr)
if 1 <= abs(value) < 1000:
done = True
elif abs(value) >= 1000:
if ostri < len(self._ostrs) - 1:
ostr = self._ostrs[ostri + 1]
value /= Decimal(1000)
else:
done = True
elif abs(value) < 1:
if ostri > 0:
ostr = self._ostrs[ostri - 1]
value *= Decimal(1000)
else:
done = True
return self._pluralize(value, ostr)
[docs] def _pack_str(self, num, unit):
if not isinstance(num, six.string_types):
num = str(num)
if self._separate_unit:
return ' '.join([num, unit])
else:
return num + unit
@property
def quantized_repr(self):
num, unit = self.natural_repr
neg = False
if num < 0:
neg = True
num = str(round_to_n(float(abs(num)), 5))
if neg is True:
num = '-' + num
return self._pack_str(num, unit)
@property
def integral_repr(self):
num, unit = self.natural_repr
neg = False
if num < 0:
neg = True
if num == 0:
return str(num) + unit
num = str(round_to_n(float(abs(num)), 2))
if neg is True:
num = '-' + num
return self._pack_str(num, unit)
@property
def integral_autoscaled_repr(self):
value = self._value
rval = None
for unit, factor in self._rorders:
uv, r = divmod(value, factor)
value = r
if not uv:
continue
part = self._pack_str(*self._pluralize(uv, unit))
if rval:
rval = ', '.join([rval, part])
else:
rval = part
if not rval:
unit, factor = self._rorders[-1]
return self._pack_str(*self._pluralize(0, unit))
return rval
[docs] def fmt_repr(self, fmt):
num, unit = self.natural_repr
parts = []
idx = 0
if '%v' in fmt:
fmt = fmt.replace('%v', '{' + str(idx) + '}')
# TODO Consider applying remove_exponent across the board
parts.append(remove_exponent(num))
idx += 1
if '%u' in fmt:
fmt = fmt.replace('%u', '{' + str(idx) + '}')
parts.append(unit)
idx += 1
return fmt.format(*parts)
def __repr__(self):
if self._rorders:
return self.integral_autoscaled_repr
return self._pack_str(*self.natural_repr)
[docs]class DummyUnit(UnitBase):
"""
This class provides a type for placeholder objects. The original
use case is for handling wave boundaries in streaming protocols.
Does not support any arithmetic operations.
"""
def __repr__(self):
return "Dummy Unit"
[docs]def parse_none(value):
"""
A placeholder parse function which can be used if the Unit
requires / supports no parsing.
"""
return value
[docs]class FactorBase(NumericalUnitBase):
pass
[docs]class Percentage(FactorBase):
"""
A base Unit class which provides support for Types that are essentially
percentages.
The contribution this base class makes is to be able to parse percentage
strings so that the Descendant class need not.
Only the standard :class:`NumericalUnitBase` Arithmetic is supported by
this class at this time.
"""
_has_bare_order = True
_orders = [('%', Decimal('1')), ('pc', Decimal('1')), ('', 100)]
_dostr = '%'
_allow_nOr = False
_regex_std = re.compile(r"^(?P<numerical>[\d]+\.?[\d]*)\s?(?P<order>(pc)?%?)(?P<residual>)$") # noqa
[docs]class Tolerance(Percentage):
pass
[docs]class GainBase(FactorBase):
_inverse_class = None
_gtype = None
def __rdiv__(self, other):
"""
Division of one Unit type class instance with a numerical type
results in a Unit type class instance of the same type, whose value
is the numerical operand's value divided by the unit type operand's
value.
In this case, the second operand must be a Unit type class instance,
and not the reverse.
"""
if isinstance(other, numbers.Number):
if not self._inverse_class:
if not self._gtype[0] == self._gtype[1]:
raise TypeError
inv_cls = self.__class__
else:
inv_cls = self._inverse_class
if isinstance(other, Decimal):
return inv_cls(other / self.value)
else:
return inv_cls(Decimal(other) / self.value)
else:
return NotImplemented
[docs] def in_db(self):
return 20 * log10(self._value)