Skip to content

Lazy

Bases: Generic[T]

Object representing lazy evaluation of called methods.

Calling any method consumes the current Lazy object. Using the same object again may cause errors due to the draining of the generator.

Found in qwlist.Lazy

Examples:

>>> qlist = QList([1, 2, 3, 4])
>>> filtered = qlist.filter(lambda x: x < 3)
>>> mapped_str = filtered.map(str)
>>> mapped_float = filtered.map(float)
>>> print(mapped_float.qlist())  # prints [1.0, 2.0]
>>> print(mapped_str.qlist())    # prints []
Source code in src\qwlist\qwlist.py
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
class Lazy(Generic[T]):
    """
    Object representing lazy evaluation of called methods.

    Calling any method **consumes** the current `Lazy` object. **Using the same object
    again may cause errors** due to the draining of the generator.

    Found in `qwlist.Lazy`

    Examples:
        >>> qlist = QList([1, 2, 3, 4])
        >>> filtered = qlist.filter(lambda x: x < 3)
        >>> mapped_str = filtered.map(str)
        >>> mapped_float = filtered.map(float)
        >>> print(mapped_float.qlist())  # prints [1.0, 2.0]
        >>> print(mapped_str.qlist())    # prints []
    """
    def __init__(self, gen: Iterable[T]):
        """
        Args:
            gen (Iterable[T]): generator used to yield values on collecting items.
        """
        self.gen = gen

    def __repr__(self) -> str:
        return f'Lazy({repr(self.gen)})'

    def iter(self) -> Iterator[T]:
        """
        Changes `self` into `Iterator[T]` by calling the `iter()` function.

        Returns:
            iterator over the elements of `self`.
        """
        return iter(self.gen)

    def list(self) -> List[T]:
        """
        Goes through elements of `self` collecting values into standard Python `list`.

        Returns:
            Standard Python `list` of all elements from `self`.
        """
        return [elem for elem in self.gen]

    def qlist(self) -> "QList[T]":
        """
        Goes through elements of `self` collecting values into `QList`. Same as calling `collect()`.

        Returns:
            `QList` of all elements from `self`.
        """
        return QList(elem for elem in self.gen)

    def filter(self, pred: Callable[[T], bool]):
        """
        Returns a new `Lazy` object containing all values from `self` for which
        the predicate holds true.

        Args:
            pred: `function: (T) -> bool`

        Returns:
            New `Lazy[T]` with elements for which the predicate holds true.

        Examples:
            >>> Lazy([0, 1, 2, 3]).filter(lambda x: x < 2).collect()
            [0, 1]
        """
        def inner():
            for elem in self.gen:
                if pred(elem):
                    yield elem
        return Lazy(inner())

    def map(self, mapper: Callable[[T], K]):
        """
        Returns a new `Lazy` object containing all values from `self` with
        the mapping function applied on them.

        Args:
            mapper: `function: (T) -> K`

        Returns:
            New `Lazy[K]` with mapped elements from `self`.
        """
        def inner():
            for elem in self.gen:
                yield mapper(elem)
        return Lazy(inner())

    def fold(self, operation: Callable[[K, T], K], init: K) -> K:
        """
        Given the combination operator reduces `self` by processing
        its constituent parts, building up the final value.

        **Other names:** fold_left, reduce, accumulate, aggregate.

        Args:
            operation: `function: (K, T) -> K`. Given the initial value `init` applies the
                given combination operator on each element yielded by `self`,
                treating the result as the first argument in the next step.
            init (K): initial value for the combination operator.

        Returns:
            The final value created from calling the `operation` on consecutive elements of `self`.

        Examples:
            >>> Lazy([1, 2, 3]).fold(lambda acc, x: acc + x, 0)
            6
        """
        acc = init
        for elem in self.gen:
            acc = operation(acc, elem)
        return acc

    def scan(self, operation: Callable[[K, T], K], state: K) -> "Lazy[K]":
        """
        Given the combination operator creates a new `Lazy[K]` object by processing
        constituent parts of `self`, yielding intermediate steps and building up the final value.
        Scan is similar to fold but returns all intermediate states instead of just the final result.

        Args:
            operation: `function: (K, T) -> K`. Given the initial `state` applies the given
             combination operator on each element yielded by the `Lazy` object, yielding the result and
             then treating it as the first argument in the next step.
            state (K): initial value for the state.

        Returns:
            Lazy[K] with all intermediate steps of the `operation`.

        Examples:
            >>> Lazy([1, 2, 3]).scan(lambda acc, x: acc + x, 0).collect()
            [1, 3, 6]
        """
        def inner(s):
            for elem in self.gen:
                s = operation(s, elem)
                yield s
        return Lazy(inner(state))

    def foreach(self, action: Callable[[T], None]):
        """
        Applies the given function to each of yielded elements.

        Args:
            action: `function: (T) -> None`

        Returns:
            `None`
        """
        for elem in self.gen:
            action(elem)

    def flatmap(self, mapper: Callable[[T], Iterable[K]]) -> "Lazy[K]":
        """
        Applies the mapper function to each of the yielded elements and flattens the results.

        Args:
            mapper: `function: (T) -> Iterable[K]`.

        Returns:
            New `Lazy` with elements from `self` mapped to an iterable and then flattened.

        Examples:
            >>> Lazy([1, 2]).flatmap(lambda x: [x, x]).qlist()
            [1, 1, 2, 2]
        """
        def inner():
            for elem in self.gen:
                yield from mapper(elem)
        return Lazy(inner())

    def zip(self, other: Iterable[K]) -> "Lazy[Tuple[T, K]]":
        """
        Combines `self` with the given `Iterable` elementwise as tuples.
         The returned `Lazy` objects yields at most the number of elements of
         the shorter sequence (`self` or `other`).

        Args:
            other (Iterable[K]): iterable to zip with `self`.

        Returns:
            New Lazy with pairs of elements from `self` and `other`.

        Examples:
            >>> Lazy([1, 2, 3]).zip(['a', 'b', 'c']).collect()
            [(1, 'a'), (2, 'b'), (3, 'c')]
        """
        return Lazy(zip(self.gen, other))

    def collect(self) -> "QList[T]":
        """
        Goes through elements of `self` collecting values into `QList`.

        Returns:
            `QList` of all elements from `self`.
        """
        return QList(x for x in self.gen)

    def __iter__(self):
        return iter(self.gen)

    def skip(self, n: int) -> "Lazy[T]":
        """
        Skips `n` first elements of `self`.

        Args:
            n (int): numbers of elements to skip. Should be non-negative.

        Returns:
            New `Lazy` with `n` first elements of `self` skipped.

        Examples:
            >>> Lazy(range(10)).skip(2).collect()
            [2, 3, 4, 5, 6, 7, 8, 9]
        """
        def inner():
            for i, elem in enumerate(self.gen):
                if i >= n:
                    yield elem
        return Lazy(inner())

    def take(self, n: int) -> "Lazy[T]":
        """
        Takes `n` first elements of `self`.

        Args:
            n (int): numbers of elements to take. Should be non-negative.

        Returns:
            New `Lazy` with only `n` first elements from `self`.

        Examples:
            >>> Lazy(range(10)).take(2).collect()
            [0, 1]
        """
        def inner():
            for i, elem in enumerate(self.gen):
                if i >= n:
                    return None
                yield elem
        return Lazy(inner())

    def flatten(self) -> "Lazy[T]":
        """
        If `self` is a `Lazy` object of `Iterable[T]`, flatten concatenates all iterables into a
        single list and returns a `Lazy[T]` object.

        Returns:
            Concatenation of all elements from `self`.

        Raises:
            TypeError: when elements of Lazy are not iterables.

        Examples:
            >>> Lazy([[1, 2], [3, 4]]).flatten().collect()
            [1, 2, 3, 4]
        """

        def inner():
            for elem in self.gen:
                if not isinstance(elem, Iterable):
                    type_name = type(elem).__name__
                    raise TypeError(
                        f'could not flatten {self.__class__.__name__}[{type_name}] because {type_name} is not iterable.')
                yield from elem
        return Lazy(inner())

    def cycle(self) -> "Lazy[T]":
        """
        Returns a new `Lazy[T]` that cycles through the elements of `self`, which means
        on achieving the last element the iteration starts from the beginning. The
        returned `Lazy` object has no end (infinite iterator) unless `self` is empty
        in which case cycle returns an empty `Lazy` object (empty iterator).

        Returns:
            Infinite `Lazy` that loops through elements of `self`, or empty iterator if `self` is emtpy.

        Examples:
            >>> Lazy([1, 2, 3]).cycle().take(7).collect()
            [1, 2, 3, 1, 2, 3, 1]
        """
        def inner():
            saved = []
            for elem in self.gen:
                saved.append(elem)
                yield elem
            while saved:
                for elem in saved:
                    yield elem
        return Lazy(inner())

    def enumerate(self, start: int = 0) -> "Lazy[Tuple[int, T]]":
        """
        Returns a new `Lazy` object with index-value pairs as its elements. Index starts at
        the given position `start` (defaults to 0).

        Args:
            start (int): starting index. Defaults to 0.

        Returns:
            New `Lazy` with pairs of index and value.

        Examples:
            >>> Lazy(['a', 'b', 'c']).enumerate().collect()
            [(0, 'a'), (1, 'b'), (2, 'c')]
        """
        def inner():
            for i, elem in enumerate(self, start=start):
                yield i, elem
        return Lazy(inner())

    def batch(self, size: int) -> "Lazy[QList[T]]":
        """
        Groups elements into batches of given `size`. The last batch may have fewer elements.

        Args:
            size (int): size of one batch.

        Returns:
            New `Lazy` of batches (`QList`) of given `size`. Last batch may have fewer elements.

        Examples:
            >>> Lazy(range(5)).batch(2).collect()
            [[0, 1], [2, 3], [4]]
        """
        assert size > 0, f'batch size must be greater then 0 but got {size}'
        def inner():
            group = QList()
            for i, elem in enumerate(self.gen, start=1):
                group.append(elem)
                if i % size == 0:
                    yield group
                    group = QList()
            if group:
                yield group
        return Lazy(inner())

    def batch_by(self, grouper: Callable[[T], SupportsEq]) -> "Lazy[QList[T]]":
        """
        Batches elements of `self` based on the output of the grouper function. Elements are thrown
        to the same group as long as the grouper function returns the same key (keys must support equality checks).
        When a new key is returned a new batch (group) is created.

        Args:
            grouper (Callable[[T], SupportsEq]): `function: (T) -> SupportsEq` that provides the keys
             used to group elements, where the key type must support equality comparisons.

        Returns:
            New `Lazy[QList[T]]` with elements batched based on the `grouper` key.

        Examples:
            >>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[0]).collect()
            [['a1'], ['b1', 'b2'], ['a2', 'a3'], ['b3']]
            >>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[1]).collect()
            [['a1', 'b1'], ['b2', 'a2'], ['a3', 'b3']]
        """
        def inner():
            it = self.iter()
            try:
                first = next(it)
            except StopIteration:
                return
            batch = QList([first])
            key = grouper(first)
            for elem in it:
                new_key = grouper(elem)
                if new_key == key:
                    batch.append(elem)
                else:
                    yield batch
                    batch = QList([elem])
                    key = new_key
            yield batch
        return Lazy(inner())

    def chain(self, other: Iterable[T]) -> "Lazy[T]":
        """
        Chains `self` with `other`, returning a new Lazy[T] with all elements from both iterables.

        Args:
            other (Iterable[T]):  an iterable of elements to be "attached" after self is exhausted.

        Returns:
            New `Lazy` with elements from `self` and `other`.

        Examples:
            >>> Lazy(range(0, 3)).chain(range(3, 6)).collect()
            [0, 1, 2, 3, 4, 5]
        """
        def inner():
            yield from self.gen
            yield from other
        return Lazy(inner())

    def merge(self, other: Iterable[T], merger: Callable[[T, T], bool]) -> "Lazy[T]":
        """
        Merges `self` with `other`, maintaining the order of elements based on the merger function. It starts by
         taking the first elements from `self` and `other`, calling the merger function with these elements as arguments.
         If the output is True, the first element is yielded; otherwise, the second element is yielded. If `self` is
         empty, the remaining elements from `other` are yielded, and vice versa.

        Args:
            other (Iterable[T]): an iterable to be merged with `self`.
            merger (Callable[[T, T], bool]): `function: (T, T) -> bool` that takes two arguments (left and right).
             If the output is True, the left argument is yielded; otherwise, the right argument is yielded.

        Returns:
            New `Lazy` containing the merged elements.

        Examples:
            >>> QList([1, 3, 5]).merge([2, 4, 6], lambda left, right: left < right).collect()
            [1, 2, 3, 4, 5, 6]
        """
        it1 = iter(self)
        it2 = iter(other)

        try:
            elem1 = next(it1)
        except StopIteration:
            return Lazy(it2)
        try:
            elem2 = next(it2)
        except StopIteration:
            return Lazy([elem1]).chain(it1)

        def inner():
            left = elem1
            right = elem2
            while True:
                if merger(left, right):
                    yield left
                    try:
                        left = next(it1)
                    except StopIteration:
                        yield right
                        yield from it2
                        return
                else:
                    yield right
                    try:
                        right = next(it2)
                    except StopIteration:
                        yield left
                        yield from it1
                        return
        return Lazy(inner())

    def all(self, mapper: Optional[Callable[[T], Booly]] = None) -> bool:
        """
        Goes through the entire generator and checks if all elements are `Truthy`.
        `Booly` is a type that evaluates to something that is either `True` (`Truthy`) or `False` (`Falsy`).
        For example in Python an empty list evaluates to `False` (empty list is `Falsy`).

        Args:
            mapper (Optional[Callable[[T], Booly]]): `function: (T) -> Booly` that maps `T` to `Booly` which is a type that
             can be interpreted as either True or False. If not passed, identity function is used.

        Returns:
            `True` if all elements of `self` are `Truthy`. `False` otherwise.
        """
        def identity(x):
            return x
        mapper = identity if mapper is None else mapper
        for elem in self.gen:
            if not mapper(elem):
                return False
        return True

    def any(self, mapper: Optional[Callable[[T], Booly]] = None) -> bool:
        """
        Goes through the entire generator and checks if any element is `Truthy`.
        `Booly` is a type that evaluates to something that is either `True` (`Truthy`) or `False` (`Falsy`).
        For example in Python an empty list evaluates to `False` (empty list is `Falsy`).

        Args:
            mapper (Optional[Callable[[T], Booly]]): `function: (T) -> Booly` that maps `T` to `Booly` which is a type that
             can be interpreted as either True or False. If not passed, identity function is used.

        Returns:
            `True` if there is at least one element in `self` that is `Truthy`. `False` otherwise.
        """
        def identity(x):
            return x

        mapper = identity if mapper is None else mapper
        for elem in self.gen:
            if mapper(elem):
                return True
        return False

    def min(self, key: Optional[Callable[[T], SupportsLessThan]] = None) -> Optional[T]:
        """
        Returns the smallest element from `self`. If the key function is not passed, identity
        function is used in which case `T` must support `LessThan` operator.

        Args:
            key (Optional[Callable[[T], SupportsLessThan]): `function: (T) -> SupportsLessThan` that represents
             the relation of partial order between elements.

        Returns:
            the smallest element of `self` or `None` if `self` is empty.
        """
        def identity(x):
            return x
        key = identity if key is None else key

        it = self.iter()
        try:
            best = next(it)
        except StopIteration:
            return None
        for elem in it:
            if key(elem) < key(best):
                best = elem
        return best

    def max(self, key: Optional[Callable[[T], SupportsLessThan]] = None) -> Optional[T]:
        """
        Returns the biggest element from the iterable. If the key function is not passed, identity
        function is used in which case `T` must support `LessThan` operator.

        Args:
            key (Optional[Callable[[T], SupportsLessThan]): `function: (T) -> SupportsLessThan` that represents
             the relation of partial order between elements.

        Returns:
            the biggest element of `self` or `None` if `self` is empty.
        """

        def identity(x):
            return x
        key = identity if key is None else key

        it = self.iter()
        try:
            best = next(it)
        except StopIteration:
            return None
        for elem in it:
            if key(best) < key(elem):
                best = elem
        return best

    def full_flatten(self, break_str: bool = True, preserve_type: Optional[Type] = None) -> "Lazy[T]":
        """
        When `self` is an iterable of nested iterables, all the iterables are flattened to a single iterable.
        Recursive type annotation of `self` may be imagined to look like this: `Lazy[T | Iterable[T | Iterable[T | ...]]]`.


        Args:
            break_str (bool): If `True`, strings are flattened into individual characters. Defaults to `True`.
            preserve_type (Optional[Type]): Type to exclude from flattening (i.e., treated as non-iterable). For example,
             setting this to `str` makes `break_str` effectively `False`. Defaults to `None`.

        Returns:
            New `Lazy` with all nested iterables flattened to a single iterable.

        Examples:
            >>> Lazy(['abc', ['def', 'ghi']]).full_flatten().collect()
            ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i']

            >>> Lazy(['abc', ['def', 'ghi']]).full_flatten(break_str=False).collect()
            ['abc', 'def', 'ghi']

            >>> Lazy(['abc', ['def', 'ghi']]).full_flatten(preserve_type=list).collect()
            ['a', 'b', 'c', ['def', 'ghi']]
        """
        def inner():
            for elem in self.gen:
                if preserve_type is not None and isinstance(elem, preserve_type):
                    yield elem
                elif isinstance(elem, str):
                    if break_str:
                        if len(elem) == 1:
                            yield elem
                        else:
                            yield from Lazy(elem).full_flatten(break_str=break_str, preserve_type=preserve_type)
                    else:
                        yield elem
                elif isinstance(elem, Iterable):
                    yield from Lazy(elem).full_flatten(break_str=break_str, preserve_type=preserve_type)
                else:
                    yield elem
        return Lazy(inner())

    def sum(self) -> Optional[SupportsAdd]:
        """
        Sums all the elements and returns the sum. Returns `None` if `self` is empty.
        Elements of `self` must support addition.

        Returns:
            The sum of all elements of `self`.
        """
        it = self.iter()
        acc = None
        try:
            acc = next(it)
        except StopIteration:
            return acc
        for elem in it:
            acc = acc + elem
        return acc

    def take_while(self, pred: Callable[[T], bool]) -> "Lazy[T]":
        """
        Creates a new Lazy that yields elements based on a predicate. Takes a function as an argument.
        It will call this function on each element of `self`, and yield elements while the function
        returns `True`. After `False` is returned, iteration stops, and the rest of the elements is ignored.

        Args:
            pred (Callable[[T], bool]): `function: (T) -> bool`

        Returns:
            A new `Lazy` containing elements from the original sequence, stopping at the first element for which the
             predicate returns `False`.
        """
        def inner():
            for elem in self.gen:
                if not pred(elem):
                    return
                yield elem
        return Lazy(inner())

    def window(self, window_size: int) -> "Lazy[QList[T]]":
        """
        Creates a new `Lazy` of sliding windows of size `window_size`. If `window_size`
        is greater than the total length of `self` an empty iterator is returned.

        Args:
            window_size (int): the size of the sliding window. Must be greater than 0.

        Returns:
            New `Lazy` of all sliding windows.
        """
        assert window_size > 0, f'window size must be greater than 0 but got {window_size}.'
        def inner(n: int):
            window = deque(maxlen=n)
            it = self.iter()
            try:
                for _ in range(n):
                    window.append(next(it))
            except StopIteration:
                return
            yield QList(window)
            for elem in it:
                window.append(elem)
                yield QList(window)
        return Lazy(inner(n=window_size))

    def first(self) -> Optional[T]:
        """
        Tries to access the first element of `self`. Returns `None` if `self` is empty.

        Returns:
            First element of `self` or `None` if `self` is empty.
        """
        it = self.iter()
        try:
            return next(it)
        except StopIteration:
            pass
        return None

    def get(self, index: int, default: Optional[T] = None) -> Optional[T]:
        """
        Safely gets the element on the specified index. If the index is out of bounds `default` is returned.
        This is an `O(n)` operation that consumes the iterator!

        Args:
            index (int): index of the element to take
            default (Optional[T]): value to return if the index is out of bounds. Defaults to `None`

        Returns:
            Element at the specified index or `default` if index is out of bounds.
        """
        if index < 0:
            return default
        for i, elem in self.enumerate():
            if i == index:
                return elem
        return default

    def uncons(self) -> Optional[Tuple[T, "Lazy[T]"]]:
        """
        Splits `self` into head and tail. Returns `None` if `self` is emtpy.

        Returns:
            Head and tail of `self` or `None` is `self` is empty.

        """
        it = self.iter()
        try:
            head = next(it)
        except StopIteration:
            return None
        return head, Lazy(it)

    def split_when(self, pred: Callable[[T], bool]) -> Optional[Tuple["QList[T]", "Lazy[T]"]]:
        """
        Splits the lazy sequence into two parts at the first element satisfying the predicate.
        The element that satisfies the predicate is included in the left part, which is fully evaluated,
        while the right part remains lazy, allowing for operations on infinite sequences.

        Args:
            pred (Callable[[T], bool]): `function (T) -> bool` that returns `True`
                if the element is the split point.

        Returns:
            A tuple where the first element is a fully evaluated `QList` containing all elements up to and
             including the split point, and the second element is a lazily evaluated sequence of all
             elements after the split point. Returns `None` if `self` is empty. If no element satisfies the
             predicate, the left part contains all elements from `self` and the right part is an empty lazy sequence.
        """
        left = QList()
        it = self.iter()
        for elem in it:
            left.append(elem)
            if pred(elem):
                return left, Lazy(it)
        if not left:
            return None
        return left, Lazy([])

__init__(gen)

Parameters:

Name Type Description Default
gen Iterable[T]

generator used to yield values on collecting items.

required
Source code in src\qwlist\qwlist.py
def __init__(self, gen: Iterable[T]):
    """
    Args:
        gen (Iterable[T]): generator used to yield values on collecting items.
    """
    self.gen = gen

all(mapper=None)

Goes through the entire generator and checks if all elements are Truthy. Booly is a type that evaluates to something that is either True (Truthy) or False (Falsy). For example in Python an empty list evaluates to False (empty list is Falsy).

Parameters:

Name Type Description Default
mapper Optional[Callable[[T], Booly]]

function: (T) -> Booly that maps T to Booly which is a type that can be interpreted as either True or False. If not passed, identity function is used.

None

Returns:

Type Description
bool

True if all elements of self are Truthy. False otherwise.

Source code in src\qwlist\qwlist.py
def all(self, mapper: Optional[Callable[[T], Booly]] = None) -> bool:
    """
    Goes through the entire generator and checks if all elements are `Truthy`.
    `Booly` is a type that evaluates to something that is either `True` (`Truthy`) or `False` (`Falsy`).
    For example in Python an empty list evaluates to `False` (empty list is `Falsy`).

    Args:
        mapper (Optional[Callable[[T], Booly]]): `function: (T) -> Booly` that maps `T` to `Booly` which is a type that
         can be interpreted as either True or False. If not passed, identity function is used.

    Returns:
        `True` if all elements of `self` are `Truthy`. `False` otherwise.
    """
    def identity(x):
        return x
    mapper = identity if mapper is None else mapper
    for elem in self.gen:
        if not mapper(elem):
            return False
    return True

any(mapper=None)

Goes through the entire generator and checks if any element is Truthy. Booly is a type that evaluates to something that is either True (Truthy) or False (Falsy). For example in Python an empty list evaluates to False (empty list is Falsy).

Parameters:

Name Type Description Default
mapper Optional[Callable[[T], Booly]]

function: (T) -> Booly that maps T to Booly which is a type that can be interpreted as either True or False. If not passed, identity function is used.

None

Returns:

Type Description
bool

True if there is at least one element in self that is Truthy. False otherwise.

Source code in src\qwlist\qwlist.py
def any(self, mapper: Optional[Callable[[T], Booly]] = None) -> bool:
    """
    Goes through the entire generator and checks if any element is `Truthy`.
    `Booly` is a type that evaluates to something that is either `True` (`Truthy`) or `False` (`Falsy`).
    For example in Python an empty list evaluates to `False` (empty list is `Falsy`).

    Args:
        mapper (Optional[Callable[[T], Booly]]): `function: (T) -> Booly` that maps `T` to `Booly` which is a type that
         can be interpreted as either True or False. If not passed, identity function is used.

    Returns:
        `True` if there is at least one element in `self` that is `Truthy`. `False` otherwise.
    """
    def identity(x):
        return x

    mapper = identity if mapper is None else mapper
    for elem in self.gen:
        if mapper(elem):
            return True
    return False

batch(size)

Groups elements into batches of given size. The last batch may have fewer elements.

Parameters:

Name Type Description Default
size int

size of one batch.

required

Returns:

Type Description
Lazy[QList[T]]

New Lazy of batches (QList) of given size. Last batch may have fewer elements.

Examples:

>>> Lazy(range(5)).batch(2).collect()
[[0, 1], [2, 3], [4]]
Source code in src\qwlist\qwlist.py
def batch(self, size: int) -> "Lazy[QList[T]]":
    """
    Groups elements into batches of given `size`. The last batch may have fewer elements.

    Args:
        size (int): size of one batch.

    Returns:
        New `Lazy` of batches (`QList`) of given `size`. Last batch may have fewer elements.

    Examples:
        >>> Lazy(range(5)).batch(2).collect()
        [[0, 1], [2, 3], [4]]
    """
    assert size > 0, f'batch size must be greater then 0 but got {size}'
    def inner():
        group = QList()
        for i, elem in enumerate(self.gen, start=1):
            group.append(elem)
            if i % size == 0:
                yield group
                group = QList()
        if group:
            yield group
    return Lazy(inner())

batch_by(grouper)

Batches elements of self based on the output of the grouper function. Elements are thrown to the same group as long as the grouper function returns the same key (keys must support equality checks). When a new key is returned a new batch (group) is created.

Parameters:

Name Type Description Default
grouper Callable[[T], SupportsEq]

function: (T) -> SupportsEq that provides the keys used to group elements, where the key type must support equality comparisons.

required

Returns:

Type Description
Lazy[QList[T]]

New Lazy[QList[T]] with elements batched based on the grouper key.

Examples:

>>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[0]).collect()
[['a1'], ['b1', 'b2'], ['a2', 'a3'], ['b3']]
>>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[1]).collect()
[['a1', 'b1'], ['b2', 'a2'], ['a3', 'b3']]
Source code in src\qwlist\qwlist.py
def batch_by(self, grouper: Callable[[T], SupportsEq]) -> "Lazy[QList[T]]":
    """
    Batches elements of `self` based on the output of the grouper function. Elements are thrown
    to the same group as long as the grouper function returns the same key (keys must support equality checks).
    When a new key is returned a new batch (group) is created.

    Args:
        grouper (Callable[[T], SupportsEq]): `function: (T) -> SupportsEq` that provides the keys
         used to group elements, where the key type must support equality comparisons.

    Returns:
        New `Lazy[QList[T]]` with elements batched based on the `grouper` key.

    Examples:
        >>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[0]).collect()
        [['a1'], ['b1', 'b2'], ['a2', 'a3'], ['b3']]
        >>> Lazy(['a1', 'b1', 'b2', 'a2', 'a3', 'b3']).batch_by(lambda s: s[1]).collect()
        [['a1', 'b1'], ['b2', 'a2'], ['a3', 'b3']]
    """
    def inner():
        it = self.iter()
        try:
            first = next(it)
        except StopIteration:
            return
        batch = QList([first])
        key = grouper(first)
        for elem in it:
            new_key = grouper(elem)
            if new_key == key:
                batch.append(elem)
            else:
                yield batch
                batch = QList([elem])
                key = new_key
        yield batch
    return Lazy(inner())

chain(other)

Chains self with other, returning a new Lazy[T] with all elements from both iterables.

Parameters:

Name Type Description Default
other Iterable[T]

an iterable of elements to be "attached" after self is exhausted.

required

Returns:

Type Description
Lazy[T]

New Lazy with elements from self and other.

Examples:

>>> Lazy(range(0, 3)).chain(range(3, 6)).collect()
[0, 1, 2, 3, 4, 5]
Source code in src\qwlist\qwlist.py
def chain(self, other: Iterable[T]) -> "Lazy[T]":
    """
    Chains `self` with `other`, returning a new Lazy[T] with all elements from both iterables.

    Args:
        other (Iterable[T]):  an iterable of elements to be "attached" after self is exhausted.

    Returns:
        New `Lazy` with elements from `self` and `other`.

    Examples:
        >>> Lazy(range(0, 3)).chain(range(3, 6)).collect()
        [0, 1, 2, 3, 4, 5]
    """
    def inner():
        yield from self.gen
        yield from other
    return Lazy(inner())

collect()

Goes through elements of self collecting values into QList.

Returns:

Type Description
QList[T]

QList of all elements from self.

Source code in src\qwlist\qwlist.py
def collect(self) -> "QList[T]":
    """
    Goes through elements of `self` collecting values into `QList`.

    Returns:
        `QList` of all elements from `self`.
    """
    return QList(x for x in self.gen)

cycle()

Returns a new Lazy[T] that cycles through the elements of self, which means on achieving the last element the iteration starts from the beginning. The returned Lazy object has no end (infinite iterator) unless self is empty in which case cycle returns an empty Lazy object (empty iterator).

Returns:

Type Description
Lazy[T]

Infinite Lazy that loops through elements of self, or empty iterator if self is emtpy.

Examples:

>>> Lazy([1, 2, 3]).cycle().take(7).collect()
[1, 2, 3, 1, 2, 3, 1]
Source code in src\qwlist\qwlist.py
def cycle(self) -> "Lazy[T]":
    """
    Returns a new `Lazy[T]` that cycles through the elements of `self`, which means
    on achieving the last element the iteration starts from the beginning. The
    returned `Lazy` object has no end (infinite iterator) unless `self` is empty
    in which case cycle returns an empty `Lazy` object (empty iterator).

    Returns:
        Infinite `Lazy` that loops through elements of `self`, or empty iterator if `self` is emtpy.

    Examples:
        >>> Lazy([1, 2, 3]).cycle().take(7).collect()
        [1, 2, 3, 1, 2, 3, 1]
    """
    def inner():
        saved = []
        for elem in self.gen:
            saved.append(elem)
            yield elem
        while saved:
            for elem in saved:
                yield elem
    return Lazy(inner())

enumerate(start=0)

Returns a new Lazy object with index-value pairs as its elements. Index starts at the given position start (defaults to 0).

Parameters:

Name Type Description Default
start int

starting index. Defaults to 0.

0

Returns:

Type Description
Lazy[Tuple[int, T]]

New Lazy with pairs of index and value.

Examples:

>>> Lazy(['a', 'b', 'c']).enumerate().collect()
[(0, 'a'), (1, 'b'), (2, 'c')]
Source code in src\qwlist\qwlist.py
def enumerate(self, start: int = 0) -> "Lazy[Tuple[int, T]]":
    """
    Returns a new `Lazy` object with index-value pairs as its elements. Index starts at
    the given position `start` (defaults to 0).

    Args:
        start (int): starting index. Defaults to 0.

    Returns:
        New `Lazy` with pairs of index and value.

    Examples:
        >>> Lazy(['a', 'b', 'c']).enumerate().collect()
        [(0, 'a'), (1, 'b'), (2, 'c')]
    """
    def inner():
        for i, elem in enumerate(self, start=start):
            yield i, elem
    return Lazy(inner())

filter(pred)

Returns a new Lazy object containing all values from self for which the predicate holds true.

Parameters:

Name Type Description Default
pred Callable[[T], bool]

function: (T) -> bool

required

Returns:

Type Description

New Lazy[T] with elements for which the predicate holds true.

Examples:

>>> Lazy([0, 1, 2, 3]).filter(lambda x: x < 2).collect()
[0, 1]
Source code in src\qwlist\qwlist.py
def filter(self, pred: Callable[[T], bool]):
    """
    Returns a new `Lazy` object containing all values from `self` for which
    the predicate holds true.

    Args:
        pred: `function: (T) -> bool`

    Returns:
        New `Lazy[T]` with elements for which the predicate holds true.

    Examples:
        >>> Lazy([0, 1, 2, 3]).filter(lambda x: x < 2).collect()
        [0, 1]
    """
    def inner():
        for elem in self.gen:
            if pred(elem):
                yield elem
    return Lazy(inner())

first()

Tries to access the first element of self. Returns None if self is empty.

Returns:

Type Description
Optional[T]

First element of self or None if self is empty.

Source code in src\qwlist\qwlist.py
def first(self) -> Optional[T]:
    """
    Tries to access the first element of `self`. Returns `None` if `self` is empty.

    Returns:
        First element of `self` or `None` if `self` is empty.
    """
    it = self.iter()
    try:
        return next(it)
    except StopIteration:
        pass
    return None

flatmap(mapper)

Applies the mapper function to each of the yielded elements and flattens the results.

Parameters:

Name Type Description Default
mapper Callable[[T], Iterable[K]]

function: (T) -> Iterable[K].

required

Returns:

Type Description
Lazy[K]

New Lazy with elements from self mapped to an iterable and then flattened.

Examples:

>>> Lazy([1, 2]).flatmap(lambda x: [x, x]).qlist()
[1, 1, 2, 2]
Source code in src\qwlist\qwlist.py
def flatmap(self, mapper: Callable[[T], Iterable[K]]) -> "Lazy[K]":
    """
    Applies the mapper function to each of the yielded elements and flattens the results.

    Args:
        mapper: `function: (T) -> Iterable[K]`.

    Returns:
        New `Lazy` with elements from `self` mapped to an iterable and then flattened.

    Examples:
        >>> Lazy([1, 2]).flatmap(lambda x: [x, x]).qlist()
        [1, 1, 2, 2]
    """
    def inner():
        for elem in self.gen:
            yield from mapper(elem)
    return Lazy(inner())

flatten()

If self is a Lazy object of Iterable[T], flatten concatenates all iterables into a single list and returns a Lazy[T] object.

Returns:

Type Description
Lazy[T]

Concatenation of all elements from self.

Raises:

Type Description
TypeError

when elements of Lazy are not iterables.

Examples:

>>> Lazy([[1, 2], [3, 4]]).flatten().collect()
[1, 2, 3, 4]
Source code in src\qwlist\qwlist.py
def flatten(self) -> "Lazy[T]":
    """
    If `self` is a `Lazy` object of `Iterable[T]`, flatten concatenates all iterables into a
    single list and returns a `Lazy[T]` object.

    Returns:
        Concatenation of all elements from `self`.

    Raises:
        TypeError: when elements of Lazy are not iterables.

    Examples:
        >>> Lazy([[1, 2], [3, 4]]).flatten().collect()
        [1, 2, 3, 4]
    """

    def inner():
        for elem in self.gen:
            if not isinstance(elem, Iterable):
                type_name = type(elem).__name__
                raise TypeError(
                    f'could not flatten {self.__class__.__name__}[{type_name}] because {type_name} is not iterable.')
            yield from elem
    return Lazy(inner())

fold(operation, init)

Given the combination operator reduces self by processing its constituent parts, building up the final value.

Other names: fold_left, reduce, accumulate, aggregate.

Parameters:

Name Type Description Default
operation Callable[[K, T], K]

function: (K, T) -> K. Given the initial value init applies the given combination operator on each element yielded by self, treating the result as the first argument in the next step.

required
init K

initial value for the combination operator.

required

Returns:

Type Description
K

The final value created from calling the operation on consecutive elements of self.

Examples:

>>> Lazy([1, 2, 3]).fold(lambda acc, x: acc + x, 0)
6
Source code in src\qwlist\qwlist.py
def fold(self, operation: Callable[[K, T], K], init: K) -> K:
    """
    Given the combination operator reduces `self` by processing
    its constituent parts, building up the final value.

    **Other names:** fold_left, reduce, accumulate, aggregate.

    Args:
        operation: `function: (K, T) -> K`. Given the initial value `init` applies the
            given combination operator on each element yielded by `self`,
            treating the result as the first argument in the next step.
        init (K): initial value for the combination operator.

    Returns:
        The final value created from calling the `operation` on consecutive elements of `self`.

    Examples:
        >>> Lazy([1, 2, 3]).fold(lambda acc, x: acc + x, 0)
        6
    """
    acc = init
    for elem in self.gen:
        acc = operation(acc, elem)
    return acc

foreach(action)

Applies the given function to each of yielded elements.

Parameters:

Name Type Description Default
action Callable[[T], None]

function: (T) -> None

required

Returns:

Type Description

None

Source code in src\qwlist\qwlist.py
def foreach(self, action: Callable[[T], None]):
    """
    Applies the given function to each of yielded elements.

    Args:
        action: `function: (T) -> None`

    Returns:
        `None`
    """
    for elem in self.gen:
        action(elem)

full_flatten(break_str=True, preserve_type=None)

When self is an iterable of nested iterables, all the iterables are flattened to a single iterable. Recursive type annotation of self may be imagined to look like this: Lazy[T | Iterable[T | Iterable[T | ...]]].

Parameters:

Name Type Description Default
break_str bool

If True, strings are flattened into individual characters. Defaults to True.

True
preserve_type Optional[Type]

Type to exclude from flattening (i.e., treated as non-iterable). For example, setting this to str makes break_str effectively False. Defaults to None.

None

Returns:

Type Description
Lazy[T]

New Lazy with all nested iterables flattened to a single iterable.

Examples:

>>> Lazy(['abc', ['def', 'ghi']]).full_flatten().collect()
['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i']
>>> Lazy(['abc', ['def', 'ghi']]).full_flatten(break_str=False).collect()
['abc', 'def', 'ghi']
>>> Lazy(['abc', ['def', 'ghi']]).full_flatten(preserve_type=list).collect()
['a', 'b', 'c', ['def', 'ghi']]
Source code in src\qwlist\qwlist.py
def full_flatten(self, break_str: bool = True, preserve_type: Optional[Type] = None) -> "Lazy[T]":
    """
    When `self` is an iterable of nested iterables, all the iterables are flattened to a single iterable.
    Recursive type annotation of `self` may be imagined to look like this: `Lazy[T | Iterable[T | Iterable[T | ...]]]`.


    Args:
        break_str (bool): If `True`, strings are flattened into individual characters. Defaults to `True`.
        preserve_type (Optional[Type]): Type to exclude from flattening (i.e., treated as non-iterable). For example,
         setting this to `str` makes `break_str` effectively `False`. Defaults to `None`.

    Returns:
        New `Lazy` with all nested iterables flattened to a single iterable.

    Examples:
        >>> Lazy(['abc', ['def', 'ghi']]).full_flatten().collect()
        ['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i']

        >>> Lazy(['abc', ['def', 'ghi']]).full_flatten(break_str=False).collect()
        ['abc', 'def', 'ghi']

        >>> Lazy(['abc', ['def', 'ghi']]).full_flatten(preserve_type=list).collect()
        ['a', 'b', 'c', ['def', 'ghi']]
    """
    def inner():
        for elem in self.gen:
            if preserve_type is not None and isinstance(elem, preserve_type):
                yield elem
            elif isinstance(elem, str):
                if break_str:
                    if len(elem) == 1:
                        yield elem
                    else:
                        yield from Lazy(elem).full_flatten(break_str=break_str, preserve_type=preserve_type)
                else:
                    yield elem
            elif isinstance(elem, Iterable):
                yield from Lazy(elem).full_flatten(break_str=break_str, preserve_type=preserve_type)
            else:
                yield elem
    return Lazy(inner())

get(index, default=None)

Safely gets the element on the specified index. If the index is out of bounds default is returned. This is an O(n) operation that consumes the iterator!

Parameters:

Name Type Description Default
index int

index of the element to take

required
default Optional[T]

value to return if the index is out of bounds. Defaults to None

None

Returns:

Type Description
Optional[T]

Element at the specified index or default if index is out of bounds.

Source code in src\qwlist\qwlist.py
def get(self, index: int, default: Optional[T] = None) -> Optional[T]:
    """
    Safely gets the element on the specified index. If the index is out of bounds `default` is returned.
    This is an `O(n)` operation that consumes the iterator!

    Args:
        index (int): index of the element to take
        default (Optional[T]): value to return if the index is out of bounds. Defaults to `None`

    Returns:
        Element at the specified index or `default` if index is out of bounds.
    """
    if index < 0:
        return default
    for i, elem in self.enumerate():
        if i == index:
            return elem
    return default

iter()

Changes self into Iterator[T] by calling the iter() function.

Returns:

Type Description
Iterator[T]

iterator over the elements of self.

Source code in src\qwlist\qwlist.py
def iter(self) -> Iterator[T]:
    """
    Changes `self` into `Iterator[T]` by calling the `iter()` function.

    Returns:
        iterator over the elements of `self`.
    """
    return iter(self.gen)

list()

Goes through elements of self collecting values into standard Python list.

Returns:

Type Description
List[T]

Standard Python list of all elements from self.

Source code in src\qwlist\qwlist.py
def list(self) -> List[T]:
    """
    Goes through elements of `self` collecting values into standard Python `list`.

    Returns:
        Standard Python `list` of all elements from `self`.
    """
    return [elem for elem in self.gen]

map(mapper)

Returns a new Lazy object containing all values from self with the mapping function applied on them.

Parameters:

Name Type Description Default
mapper Callable[[T], K]

function: (T) -> K

required

Returns:

Type Description

New Lazy[K] with mapped elements from self.

Source code in src\qwlist\qwlist.py
def map(self, mapper: Callable[[T], K]):
    """
    Returns a new `Lazy` object containing all values from `self` with
    the mapping function applied on them.

    Args:
        mapper: `function: (T) -> K`

    Returns:
        New `Lazy[K]` with mapped elements from `self`.
    """
    def inner():
        for elem in self.gen:
            yield mapper(elem)
    return Lazy(inner())

max(key=None)

Returns the biggest element from the iterable. If the key function is not passed, identity function is used in which case T must support LessThan operator.

Parameters:

Name Type Description Default
key Optional[Callable[[T], SupportsLessThan]

function: (T) -> SupportsLessThan that represents the relation of partial order between elements.

None

Returns:

Type Description
Optional[T]

the biggest element of self or None if self is empty.

Source code in src\qwlist\qwlist.py
def max(self, key: Optional[Callable[[T], SupportsLessThan]] = None) -> Optional[T]:
    """
    Returns the biggest element from the iterable. If the key function is not passed, identity
    function is used in which case `T` must support `LessThan` operator.

    Args:
        key (Optional[Callable[[T], SupportsLessThan]): `function: (T) -> SupportsLessThan` that represents
         the relation of partial order between elements.

    Returns:
        the biggest element of `self` or `None` if `self` is empty.
    """

    def identity(x):
        return x
    key = identity if key is None else key

    it = self.iter()
    try:
        best = next(it)
    except StopIteration:
        return None
    for elem in it:
        if key(best) < key(elem):
            best = elem
    return best

merge(other, merger)

Merges self with other, maintaining the order of elements based on the merger function. It starts by taking the first elements from self and other, calling the merger function with these elements as arguments. If the output is True, the first element is yielded; otherwise, the second element is yielded. If self is empty, the remaining elements from other are yielded, and vice versa.

Parameters:

Name Type Description Default
other Iterable[T]

an iterable to be merged with self.

required
merger Callable[[T, T], bool]

function: (T, T) -> bool that takes two arguments (left and right). If the output is True, the left argument is yielded; otherwise, the right argument is yielded.

required

Returns:

Type Description
Lazy[T]

New Lazy containing the merged elements.

Examples:

>>> QList([1, 3, 5]).merge([2, 4, 6], lambda left, right: left < right).collect()
[1, 2, 3, 4, 5, 6]
Source code in src\qwlist\qwlist.py
def merge(self, other: Iterable[T], merger: Callable[[T, T], bool]) -> "Lazy[T]":
    """
    Merges `self` with `other`, maintaining the order of elements based on the merger function. It starts by
     taking the first elements from `self` and `other`, calling the merger function with these elements as arguments.
     If the output is True, the first element is yielded; otherwise, the second element is yielded. If `self` is
     empty, the remaining elements from `other` are yielded, and vice versa.

    Args:
        other (Iterable[T]): an iterable to be merged with `self`.
        merger (Callable[[T, T], bool]): `function: (T, T) -> bool` that takes two arguments (left and right).
         If the output is True, the left argument is yielded; otherwise, the right argument is yielded.

    Returns:
        New `Lazy` containing the merged elements.

    Examples:
        >>> QList([1, 3, 5]).merge([2, 4, 6], lambda left, right: left < right).collect()
        [1, 2, 3, 4, 5, 6]
    """
    it1 = iter(self)
    it2 = iter(other)

    try:
        elem1 = next(it1)
    except StopIteration:
        return Lazy(it2)
    try:
        elem2 = next(it2)
    except StopIteration:
        return Lazy([elem1]).chain(it1)

    def inner():
        left = elem1
        right = elem2
        while True:
            if merger(left, right):
                yield left
                try:
                    left = next(it1)
                except StopIteration:
                    yield right
                    yield from it2
                    return
            else:
                yield right
                try:
                    right = next(it2)
                except StopIteration:
                    yield left
                    yield from it1
                    return
    return Lazy(inner())

min(key=None)

Returns the smallest element from self. If the key function is not passed, identity function is used in which case T must support LessThan operator.

Parameters:

Name Type Description Default
key Optional[Callable[[T], SupportsLessThan]

function: (T) -> SupportsLessThan that represents the relation of partial order between elements.

None

Returns:

Type Description
Optional[T]

the smallest element of self or None if self is empty.

Source code in src\qwlist\qwlist.py
def min(self, key: Optional[Callable[[T], SupportsLessThan]] = None) -> Optional[T]:
    """
    Returns the smallest element from `self`. If the key function is not passed, identity
    function is used in which case `T` must support `LessThan` operator.

    Args:
        key (Optional[Callable[[T], SupportsLessThan]): `function: (T) -> SupportsLessThan` that represents
         the relation of partial order between elements.

    Returns:
        the smallest element of `self` or `None` if `self` is empty.
    """
    def identity(x):
        return x
    key = identity if key is None else key

    it = self.iter()
    try:
        best = next(it)
    except StopIteration:
        return None
    for elem in it:
        if key(elem) < key(best):
            best = elem
    return best

qlist()

Goes through elements of self collecting values into QList. Same as calling collect().

Returns:

Type Description
QList[T]

QList of all elements from self.

Source code in src\qwlist\qwlist.py
def qlist(self) -> "QList[T]":
    """
    Goes through elements of `self` collecting values into `QList`. Same as calling `collect()`.

    Returns:
        `QList` of all elements from `self`.
    """
    return QList(elem for elem in self.gen)

scan(operation, state)

Given the combination operator creates a new Lazy[K] object by processing constituent parts of self, yielding intermediate steps and building up the final value. Scan is similar to fold but returns all intermediate states instead of just the final result.

Parameters:

Name Type Description Default
operation Callable[[K, T], K]

function: (K, T) -> K. Given the initial state applies the given combination operator on each element yielded by the Lazy object, yielding the result and then treating it as the first argument in the next step.

required
state K

initial value for the state.

required

Returns:

Type Description
Lazy[K]

Lazy[K] with all intermediate steps of the operation.

Examples:

>>> Lazy([1, 2, 3]).scan(lambda acc, x: acc + x, 0).collect()
[1, 3, 6]
Source code in src\qwlist\qwlist.py
def scan(self, operation: Callable[[K, T], K], state: K) -> "Lazy[K]":
    """
    Given the combination operator creates a new `Lazy[K]` object by processing
    constituent parts of `self`, yielding intermediate steps and building up the final value.
    Scan is similar to fold but returns all intermediate states instead of just the final result.

    Args:
        operation: `function: (K, T) -> K`. Given the initial `state` applies the given
         combination operator on each element yielded by the `Lazy` object, yielding the result and
         then treating it as the first argument in the next step.
        state (K): initial value for the state.

    Returns:
        Lazy[K] with all intermediate steps of the `operation`.

    Examples:
        >>> Lazy([1, 2, 3]).scan(lambda acc, x: acc + x, 0).collect()
        [1, 3, 6]
    """
    def inner(s):
        for elem in self.gen:
            s = operation(s, elem)
            yield s
    return Lazy(inner(state))

skip(n)

Skips n first elements of self.

Parameters:

Name Type Description Default
n int

numbers of elements to skip. Should be non-negative.

required

Returns:

Type Description
Lazy[T]

New Lazy with n first elements of self skipped.

Examples:

>>> Lazy(range(10)).skip(2).collect()
[2, 3, 4, 5, 6, 7, 8, 9]
Source code in src\qwlist\qwlist.py
def skip(self, n: int) -> "Lazy[T]":
    """
    Skips `n` first elements of `self`.

    Args:
        n (int): numbers of elements to skip. Should be non-negative.

    Returns:
        New `Lazy` with `n` first elements of `self` skipped.

    Examples:
        >>> Lazy(range(10)).skip(2).collect()
        [2, 3, 4, 5, 6, 7, 8, 9]
    """
    def inner():
        for i, elem in enumerate(self.gen):
            if i >= n:
                yield elem
    return Lazy(inner())

split_when(pred)

Splits the lazy sequence into two parts at the first element satisfying the predicate. The element that satisfies the predicate is included in the left part, which is fully evaluated, while the right part remains lazy, allowing for operations on infinite sequences.

Parameters:

Name Type Description Default
pred Callable[[T], bool]

function (T) -> bool that returns True if the element is the split point.

required

Returns:

Type Description
Optional[Tuple[QList[T], Lazy[T]]]

A tuple where the first element is a fully evaluated QList containing all elements up to and including the split point, and the second element is a lazily evaluated sequence of all elements after the split point. Returns None if self is empty. If no element satisfies the predicate, the left part contains all elements from self and the right part is an empty lazy sequence.

Source code in src\qwlist\qwlist.py
def split_when(self, pred: Callable[[T], bool]) -> Optional[Tuple["QList[T]", "Lazy[T]"]]:
    """
    Splits the lazy sequence into two parts at the first element satisfying the predicate.
    The element that satisfies the predicate is included in the left part, which is fully evaluated,
    while the right part remains lazy, allowing for operations on infinite sequences.

    Args:
        pred (Callable[[T], bool]): `function (T) -> bool` that returns `True`
            if the element is the split point.

    Returns:
        A tuple where the first element is a fully evaluated `QList` containing all elements up to and
         including the split point, and the second element is a lazily evaluated sequence of all
         elements after the split point. Returns `None` if `self` is empty. If no element satisfies the
         predicate, the left part contains all elements from `self` and the right part is an empty lazy sequence.
    """
    left = QList()
    it = self.iter()
    for elem in it:
        left.append(elem)
        if pred(elem):
            return left, Lazy(it)
    if not left:
        return None
    return left, Lazy([])

sum()

Sums all the elements and returns the sum. Returns None if self is empty. Elements of self must support addition.

Returns:

Type Description
Optional[SupportsAdd]

The sum of all elements of self.

Source code in src\qwlist\qwlist.py
def sum(self) -> Optional[SupportsAdd]:
    """
    Sums all the elements and returns the sum. Returns `None` if `self` is empty.
    Elements of `self` must support addition.

    Returns:
        The sum of all elements of `self`.
    """
    it = self.iter()
    acc = None
    try:
        acc = next(it)
    except StopIteration:
        return acc
    for elem in it:
        acc = acc + elem
    return acc

take(n)

Takes n first elements of self.

Parameters:

Name Type Description Default
n int

numbers of elements to take. Should be non-negative.

required

Returns:

Type Description
Lazy[T]

New Lazy with only n first elements from self.

Examples:

>>> Lazy(range(10)).take(2).collect()
[0, 1]
Source code in src\qwlist\qwlist.py
def take(self, n: int) -> "Lazy[T]":
    """
    Takes `n` first elements of `self`.

    Args:
        n (int): numbers of elements to take. Should be non-negative.

    Returns:
        New `Lazy` with only `n` first elements from `self`.

    Examples:
        >>> Lazy(range(10)).take(2).collect()
        [0, 1]
    """
    def inner():
        for i, elem in enumerate(self.gen):
            if i >= n:
                return None
            yield elem
    return Lazy(inner())

take_while(pred)

Creates a new Lazy that yields elements based on a predicate. Takes a function as an argument. It will call this function on each element of self, and yield elements while the function returns True. After False is returned, iteration stops, and the rest of the elements is ignored.

Parameters:

Name Type Description Default
pred Callable[[T], bool]

function: (T) -> bool

required

Returns:

Type Description
Lazy[T]

A new Lazy containing elements from the original sequence, stopping at the first element for which the predicate returns False.

Source code in src\qwlist\qwlist.py
def take_while(self, pred: Callable[[T], bool]) -> "Lazy[T]":
    """
    Creates a new Lazy that yields elements based on a predicate. Takes a function as an argument.
    It will call this function on each element of `self`, and yield elements while the function
    returns `True`. After `False` is returned, iteration stops, and the rest of the elements is ignored.

    Args:
        pred (Callable[[T], bool]): `function: (T) -> bool`

    Returns:
        A new `Lazy` containing elements from the original sequence, stopping at the first element for which the
         predicate returns `False`.
    """
    def inner():
        for elem in self.gen:
            if not pred(elem):
                return
            yield elem
    return Lazy(inner())

uncons()

Splits self into head and tail. Returns None if self is emtpy.

Returns:

Type Description
Optional[Tuple[T, Lazy[T]]]

Head and tail of self or None is self is empty.

Source code in src\qwlist\qwlist.py
def uncons(self) -> Optional[Tuple[T, "Lazy[T]"]]:
    """
    Splits `self` into head and tail. Returns `None` if `self` is emtpy.

    Returns:
        Head and tail of `self` or `None` is `self` is empty.

    """
    it = self.iter()
    try:
        head = next(it)
    except StopIteration:
        return None
    return head, Lazy(it)

window(window_size)

Creates a new Lazy of sliding windows of size window_size. If window_size is greater than the total length of self an empty iterator is returned.

Parameters:

Name Type Description Default
window_size int

the size of the sliding window. Must be greater than 0.

required

Returns:

Type Description
Lazy[QList[T]]

New Lazy of all sliding windows.

Source code in src\qwlist\qwlist.py
def window(self, window_size: int) -> "Lazy[QList[T]]":
    """
    Creates a new `Lazy` of sliding windows of size `window_size`. If `window_size`
    is greater than the total length of `self` an empty iterator is returned.

    Args:
        window_size (int): the size of the sliding window. Must be greater than 0.

    Returns:
        New `Lazy` of all sliding windows.
    """
    assert window_size > 0, f'window size must be greater than 0 but got {window_size}.'
    def inner(n: int):
        window = deque(maxlen=n)
        it = self.iter()
        try:
            for _ in range(n):
                window.append(next(it))
        except StopIteration:
            return
        yield QList(window)
        for elem in it:
            window.append(elem)
            yield QList(window)
    return Lazy(inner(n=window_size))

zip(other)

Combines self with the given Iterable elementwise as tuples. The returned Lazy objects yields at most the number of elements of the shorter sequence (self or other).

Parameters:

Name Type Description Default
other Iterable[K]

iterable to zip with self.

required

Returns:

Type Description
Lazy[Tuple[T, K]]

New Lazy with pairs of elements from self and other.

Examples:

>>> Lazy([1, 2, 3]).zip(['a', 'b', 'c']).collect()
[(1, 'a'), (2, 'b'), (3, 'c')]
Source code in src\qwlist\qwlist.py
def zip(self, other: Iterable[K]) -> "Lazy[Tuple[T, K]]":
    """
    Combines `self` with the given `Iterable` elementwise as tuples.
     The returned `Lazy` objects yields at most the number of elements of
     the shorter sequence (`self` or `other`).

    Args:
        other (Iterable[K]): iterable to zip with `self`.

    Returns:
        New Lazy with pairs of elements from `self` and `other`.

    Examples:
        >>> Lazy([1, 2, 3]).zip(['a', 'b', 'c']).collect()
        [(1, 'a'), (2, 'b'), (3, 'c')]
    """
    return Lazy(zip(self.gen, other))