Added group by & refactored Iterable #129

This commit is contained in:
Sven Heidemann 2022-12-01 16:27:29 +01:00
parent ba1b5e49ae
commit d8f7e03815
5 changed files with 111 additions and 133 deletions

View File

@ -0,0 +1,2 @@
def default_lambda(x: object):
return x

View File

@ -155,6 +155,15 @@ class QueryableABC(ABC):
""" """
pass pass
def group_by(self, _func: Callable = None) -> 'QueryableABC':
r"""Groups by func
Returns
-------
Grouped list[list[any]]: any
"""
pass
@abstractmethod @abstractmethod
def last(self) -> any: def last(self) -> any:
r"""Returns last element r"""Returns last element

View File

@ -16,59 +16,36 @@ class Iterable(IterableABC):
IterableABC.__init__(self, t, values) IterableABC.__init__(self, t, values)
def all(self, _func: Callable = None) -> bool: def all(self, _func: Callable = None) -> bool:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
result = self.where(_func) return self.where(_func).count() == self.count()
return len(result) == len(self)
def any(self, _func: Callable = None) -> bool: def any(self, _func: Callable = None) -> bool:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
result = self.where(_func) return self.where(_func).count() > 0
return len(result) > 0
def average(self, _func: Callable = None) -> Union[int, float, complex]: def average(self, _func: Callable = None) -> Union[int, float, complex]:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None and not is_number(self.type): if _func is None and not is_number(self.type):
raise InvalidTypeException() raise InvalidTypeException()
if _func is None: return self.sum(_func) / self.count()
_func = _default_lambda
return float(self.sum(_func)) / float(self.count())
def contains(self, _value: object) -> bool: def contains(self, _value: object) -> bool:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _value is None: if _value is None:
raise ArgumentNoneException(ExceptionArgument.value) raise ArgumentNoneException(ExceptionArgument.value)
return self.where(lambda x: x == _value).count() > 0 return self.where(lambda x: x == _value).count() > 0
def count(self, _func: Callable = None) -> int: def count(self, _func: Callable = None) -> int:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
return len(self) return self.__len__()
return len(self.where(_func)) return self.where(_func).__len__()
def distinct(self, _func: Callable = None) -> IterableABC:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
def distinct(self, _func: Callable = None) -> 'Iterable':
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
@ -85,18 +62,12 @@ class Iterable(IterableABC):
return result return result
def element_at(self, _index: int) -> any: def element_at(self, _index: int) -> any:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _index is None: if _index is None:
raise ArgumentNoneException(ExceptionArgument.index) raise ArgumentNoneException(ExceptionArgument.index)
return self[_index] return self[_index]
def element_at_or_default(self, _index: int) -> any: def element_at_or_default(self, _index: int) -> any:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _index is None: if _index is None:
raise ArgumentNoneException(ExceptionArgument.index) raise ArgumentNoneException(ExceptionArgument.index)
@ -105,56 +76,52 @@ class Iterable(IterableABC):
except IndexError: except IndexError:
return None return None
def first(self: IterableABC) -> any: def first(self) -> any:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) == 0: if len(self) == 0:
raise IndexOutOfRangeException() raise IndexOutOfRangeException()
return self[0] return self[0]
def first_or_default(self: IterableABC) -> Optional[any]: def first_or_default(self) -> Optional[any]:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) == 0: if len(self) == 0:
return None return None
return self[0] return self[0]
def last(self: IterableABC) -> any: def group_by(self, _func: Callable = None) -> 'Iterable':
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) == 0:
raise IndexOutOfRangeException()
return self[len(self) - 1]
def last_or_default(self: IterableABC) -> Optional[any]:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) == 0:
return None
return self[len(self) - 1]
def for_each(self, _func: Callable = None):
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
groups = {}
for v in self:
value = _func(v)
if v not in groups:
groups[value] = Iterable(type(v))
groups[value].append(v)
return Iterable(Iterable).extend(groups.values())
def last(self) -> any:
if len(self) == 0:
raise IndexOutOfRangeException()
return self[len(self) - 1]
def last_or_default(self) -> Optional[any]:
if len(self) == 0:
return None
return self[len(self) - 1]
def for_each(self, _func: Callable = None) -> 'Iterable':
if _func is not None:
for element in self: for element in self:
_func(element) _func(element)
def max(self, _func: Callable = None) -> Union[int, float, complex]: return self
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
def max(self, _func: Callable = None) -> Union[int, float, complex]:
if _func is None and not is_number(self.type): if _func is None and not is_number(self.type):
raise InvalidTypeException() raise InvalidTypeException()
@ -166,6 +133,7 @@ class Iterable(IterableABC):
def median(self, _func=None) -> Union[int, float]: def median(self, _func=None) -> Union[int, float]:
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
result = self.order_by(_func).select(_func).to_list() result = self.order_by(_func).select(_func).to_list()
length = len(result) length = len(result)
i = int(length / 2) i = int(length / 2)
@ -176,9 +144,6 @@ class Iterable(IterableABC):
) )
def min(self, _func: Callable = None) -> Union[int, float, complex]: def min(self, _func: Callable = None) -> Union[int, float, complex]:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None and not is_number(self.type): if _func is None and not is_number(self.type):
raise InvalidTypeException() raise InvalidTypeException()
@ -188,9 +153,6 @@ class Iterable(IterableABC):
return _func(min(self, key=_func)) return _func(min(self, key=_func))
def order_by(self, _func: Callable = None) -> OrderedIterableABC: def order_by(self, _func: Callable = None) -> OrderedIterableABC:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
@ -198,22 +160,16 @@ class Iterable(IterableABC):
return OrderedIterable(self.type, _func, sorted(self, key=_func)) return OrderedIterable(self.type, _func, sorted(self, key=_func))
def order_by_descending(self, _func: Callable = None) -> OrderedIterableABC: def order_by_descending(self, _func: Callable = None) -> OrderedIterableABC:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
from cpl_query.iterable.ordered_iterable import OrderedIterable from cpl_query.iterable.ordered_iterable import OrderedIterable
return OrderedIterable(self.type, _func, sorted(self, key=_func, reverse=True)) return OrderedIterable(self.type, _func, sorted(self, key=_func, reverse=True))
def reverse(self: IterableABC) -> IterableABC: def reverse(self) -> 'Iterable':
if self is None: return Iterable().extend(reversed(self))
raise ArgumentNoneException(ExceptionArgument.list)
return Iterable().extend(reversed(self.to_list())) def select(self, _func: Callable = None) -> 'Iterable':
def select(self, _func: Callable = None) -> IterableABC:
if _func is None: if _func is None:
_func = _default_lambda _func = _default_lambda
@ -221,10 +177,7 @@ class Iterable(IterableABC):
result.extend(_func(_o) for _o in self) result.extend(_func(_o) for _o in self)
return result return result
def select_many(self, _func: Callable = None) -> IterableABC: def select_many(self, _func: Callable = None) -> 'Iterable':
if _func is None:
_func = _default_lambda
result = Iterable() result = Iterable()
# The line below is pain. I don't understand anything of it... # The line below is pain. I don't understand anything of it...
# written on 09.11.2022 by Sven Heidemann # written on 09.11.2022 by Sven Heidemann
@ -233,10 +186,7 @@ class Iterable(IterableABC):
result.extend(elements) result.extend(elements)
return result return result
def single(self: IterableABC) -> any: def single(self) -> any:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) > 1: if len(self) > 1:
raise Exception('Found more than one element') raise Exception('Found more than one element')
elif len(self) == 0: elif len(self) == 0:
@ -244,10 +194,7 @@ class Iterable(IterableABC):
return self[0] return self[0]
def single_or_default(self: IterableABC) -> Optional[any]: def single_or_default(self) -> Optional[any]:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if len(self) > 1: if len(self) > 1:
raise Exception('Index out of range') raise Exception('Index out of range')
elif len(self) == 0: elif len(self) == 0:
@ -255,19 +202,13 @@ class Iterable(IterableABC):
return self[0] return self[0]
def skip(self, _index: int) -> IterableABC: def skip(self, _index: int) -> 'Iterable':
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _index is None: if _index is None:
raise ArgumentNoneException(ExceptionArgument.index) raise ArgumentNoneException(ExceptionArgument.index)
return Iterable(self.type, values=self[_index:]) return Iterable(self.type, values=self[_index:])
def skip_last(self, _index: int) -> IterableABC: def skip_last(self, _index: int) -> 'Iterable':
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _index is None: if _index is None:
raise ArgumentNoneException(ExceptionArgument.index) raise ArgumentNoneException(ExceptionArgument.index)
@ -277,10 +218,20 @@ class Iterable(IterableABC):
result.extend(self[:index]) result.extend(self[:index])
return result return result
def take(self, _index: int) -> IterableABC: def sum(self, _func: Callable = None) -> Union[int, float, complex]:
if self is None: if _func is None and not is_number(self.type):
raise ArgumentNoneException(ExceptionArgument.list) raise InvalidTypeException()
if _func is None:
_func = _default_lambda
result = 0
for x in self:
result += _func(x)
return result
def take(self, _index: int) -> 'Iterable':
if _index is None: if _index is None:
raise ArgumentNoneException(ExceptionArgument.index) raise ArgumentNoneException(ExceptionArgument.index)
@ -288,10 +239,7 @@ class Iterable(IterableABC):
result.extend(self[:_index]) result.extend(self[:_index])
return result return result
def take_last(self, _index: int) -> IterableABC: def take_last(self, _index: int) -> 'Iterable':
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
index = len(self) - _index index = len(self) - _index
if index >= len(self) or index < 0: if index >= len(self) or index < 0:
@ -301,22 +249,7 @@ class Iterable(IterableABC):
result.extend(self[index:]) result.extend(self[index:])
return result return result
def sum(self, _func: Callable = None) -> Union[int, float, complex]: def where(self, _func: Callable = None) -> 'Iterable':
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None and not is_number(self.type):
raise InvalidTypeException()
if _func is None:
_func = _default_lambda
return sum([_func(x) for x in self])
def where(self, _func: Callable = None) -> IterableABC:
if self is None:
raise ArgumentNoneException(ExceptionArgument.list)
if _func is None: if _func is None:
raise ArgumentNoneException(ExceptionArgument.func) raise ArgumentNoneException(ExceptionArgument.func)

View File

@ -18,6 +18,6 @@ class Application(ApplicationABC):
def main(self): def main(self):
runner = unittest.TextTestRunner() runner = unittest.TextTestRunner()
runner.run(CLITestSuite()) # runner.run(CLITestSuite())
runner.run(QueryTestSuite()) runner.run(QueryTestSuite())
runner.run(TranslationTestSuite()) # runner.run(TranslationTestSuite())

View File

@ -5,6 +5,7 @@ from random import randint
from cpl_core.utils import String from cpl_core.utils import String
from cpl_query.exceptions import InvalidTypeException, ArgumentNoneException from cpl_query.exceptions import InvalidTypeException, ArgumentNoneException
from cpl_query.extension.list import List from cpl_query.extension.list import List
from cpl_query.iterable import Iterable
from unittests_query.models import User, Address from unittests_query.models import User, Address
@ -102,8 +103,18 @@ class IterableQueryTestCase(unittest.TestCase):
self.assertEqual(1, self._tests.count(lambda u: u == self._t_user)) self.assertEqual(1, self._tests.count(lambda u: u == self._t_user))
def test_distinct(self): def test_distinct(self):
res = self._tests.distinct(lambda u: u.address.nr).where(lambda u: u.address.nr == 5) res = self._tests.select(lambda u: u.address.nr).where(lambda a: a == 5).distinct()
self.assertEqual(1, len(res)) self.assertEqual(1, res.count())
addresses = []
for u in self._tests:
if u.address.nr in addresses:
continue
addresses.append(u.address.nr)
res2 = self._tests.distinct(lambda x: x.address.nr).select(lambda x: x.address.nr)
self.assertEqual(addresses, res2)
def test_element_at(self): def test_element_at(self):
index = randint(0, len(self._tests) - 1) index = randint(0, len(self._tests) - 1)
@ -168,6 +179,29 @@ class IterableQueryTestCase(unittest.TestCase):
self.assertEqual(res[0], s_res) self.assertEqual(res[0], s_res)
self.assertIsNone(sn_res) self.assertIsNone(sn_res)
def test_group_by(self):
def by_adr(u):
return u.address.nr
t = self._tests.select(by_adr).group_by()
res = self._tests.group_by(by_adr)
self.assertTrue(isinstance(res.first_or_default(), Iterable))
self.assertNotEqual(self._tests.count(), res.count())
self.assertEqual(self._tests.distinct(by_adr).count(), res.count())
elements = List(int)
groups = {}
for x in range(0, 1000):
v = randint(1, 100)
if v not in groups:
groups[v] = []
groups[v].append(v)
elements.append(v)
r1, r2 = list(groups.values()), elements.group_by()
self.assertEqual(r1, r2)
def test_for_each(self): def test_for_each(self):
users = [] users = []
self._tests.for_each(lambda user: ( self._tests.for_each(lambda user: (