U
    -cG                      @   s  d Z ddlZddlmZ ddlmZmZmZmZm	Z	m
Z
mZmZmZmZ ddlZddlmZmZmZ dddd	d
dddddddddddddddddddddd d!d"d#d$d%g Zd&d" ZdJd'd Zd(d! ZdKd)dZdLd*dZd+d Zefd,dZd-d ZeZd.d Zd/d
 Z d0d Z!dMd1dZ"d2d3 Z#zdd4lm$Z% W n e&k
rJ   e#Z$Y nX d5d Z$e#j e$_ dNd6dZ'd7d Z(d8d Z)d9d Z*dOd:d$Z+dPd;d%Z,dQd<dZ-dRd=dZ.d>d?d@dZ/dSdAdZ0dBd Z1dCd Z2dDd Z3dEd Z4dFd	 Z5dGd Z6dHd# Z7dId Z8dS )Ta  Imported from the recipes section of the itertools documentation.

All functions taken from the recipes section of the itertools library docs
[1]_.
Some backward-compatible usability improvements have been made.

.. [1] http://docs.python.org/library/itertools.html#recipes

    N)deque)
chaincombinationscountcyclegroupbyislicerepeatstarmapteezip_longest)	randrangesamplechoice	all_equalbefore_and_afterconsumeconvolve
dotproduct
first_trueflattengrouperiter_exceptncyclesnthnth_combinationpadnonepad_nonepairwise	partitionpowersetprependquantify#random_combination_with_replacementrandom_combinationrandom_permutationrandom_product
repeatfunc
roundrobinsliding_windowtabulatetailtake
triplewiseunique_everseenunique_justseenc                 C   s   t t|| S )zReturn first *n* items of the iterable as a list.

        >>> take(3, range(10))
        [0, 1, 2]

    If there are fewer than *n* items in the iterable, all of them are
    returned.

        >>> take(10, range(3))
        [0, 1, 2]

    )listr   niterable r4   P/tmp/pip-unpacked-wheel-izj_87as/pkg_resources/_vendor/more_itertools/recipes.pyr,   ?   s    c                 C   s   t | t|S )a  Return an iterator over the results of ``func(start)``,
    ``func(start + 1)``, ``func(start + 2)``...

    *func* should be a function that accepts one integer argument.

    If *start* is not specified it defaults to 0. It will be incremented each
    time the iterator is advanced.

        >>> square = lambda x: x ** 2
        >>> iterator = tabulate(square, -3)
        >>> take(4, iterator)
        [9, 4, 1, 0]

    )mapr   )functionstartr4   r4   r5   r*   O   s    c                 C   s   t t|| dS )zReturn an iterator over the last *n* items of *iterable*.

    >>> t = tail(3, 'ABCDEFG')
    >>> list(t)
    ['E', 'F', 'G']

    maxlen)iterr   r1   r4   r4   r5   r+   a   s    c                 C   s,   |dkrt | dd ntt| ||d dS )aX  Advance *iterable* by *n* steps. If *n* is ``None``, consume it
    entirely.

    Efficiently exhausts an iterator without returning values. Defaults to
    consuming the whole iterator, but an optional second argument may be
    provided to limit consumption.

        >>> i = (x for x in range(10))
        >>> next(i)
        0
        >>> consume(i, 3)
        >>> next(i)
        4
        >>> consume(i)
        >>> next(i)
        Traceback (most recent call last):
          File "<stdin>", line 1, in <module>
        StopIteration

    If the iterator has fewer items remaining than the provided limit, the
    whole iterator will be consumed.

        >>> i = (x for x in range(3))
        >>> consume(i, 5)
        >>> next(i)
        Traceback (most recent call last):
          File "<stdin>", line 1, in <module>
        StopIteration

    Nr   r9   )r   nextr   )iteratorr2   r4   r4   r5   r   l   s     c                 C   s   t t| |d|S )zReturns the nth item or a default value.

    >>> l = range(10)
    >>> nth(l, 3)
    3
    >>> nth(l, 20, "zebra")
    'zebra'

    N)r<   r   )r3   r2   defaultr4   r4   r5   r      s    
c                 C   s   t | }t|dot|d S )z
    Returns ``True`` if all the elements are equal to each other.

        >>> all_equal('aaaa')
        True
        >>> all_equal('aaab')
        False

    TF)r   r<   )r3   gr4   r4   r5   r      s    
c                 C   s   t t|| S )zcReturn the how many times the predicate is true.

    >>> quantify([True, False, True])
    2

    )sumr6   )r3   predr4   r4   r5   r"      s    c                 C   s   t | tdS )a   Returns the sequence of elements and then returns ``None`` indefinitely.

        >>> take(5, pad_none(range(3)))
        [0, 1, 2, None, None]

    Useful for emulating the behavior of the built-in :func:`map` function.

    See also :func:`padded`.

    N)r   r	   r3   r4   r4   r5   r      s    c                 C   s   t tt| |S )zvReturns the sequence elements *n* times

    >>> list(ncycles(["a", "b"], 3))
    ['a', 'b', 'a', 'b', 'a', 'b']

    )r   from_iterabler	   tuple)r3   r2   r4   r4   r5   r      s    c                 C   s   t ttj| |S )zcReturns the dot product of the two iterables.

    >>> dotproduct([10, 10], [20, 20])
    400

    )r@   r6   operatormul)Zvec1Zvec2r4   r4   r5   r      s    c                 C   s
   t | S )zReturn an iterator flattening one level of nesting in a list of lists.

        >>> list(flatten([[0, 1], [2, 3]]))
        [0, 1, 2, 3]

    See also :func:`collapse`, which can flatten multiple levels of nesting.

    )r   rC   )ZlistOfListsr4   r4   r5   r      s    	c                 G   s&   |dkrt | t|S t | t||S )aG  Call *func* with *args* repeatedly, returning an iterable over the
    results.

    If *times* is specified, the iterable will terminate after that many
    repetitions:

        >>> from operator import add
        >>> times = 4
        >>> args = 3, 5
        >>> list(repeatfunc(add, times, *args))
        [8, 8, 8, 8]

    If *times* is ``None`` the iterable will not terminate:

        >>> from random import randrange
        >>> times = None
        >>> args = 1, 11
        >>> take(6, repeatfunc(randrange, times, *args))  # doctest:+SKIP
        [2, 4, 8, 1, 8, 4]

    N)r
   r	   )functimesargsr4   r4   r5   r'      s    c                 c   s*   t | \}}t|d t||E dH  dS )zReturns an iterator of paired items, overlapping, from the original

    >>> take(4, pairwise(count()))
    [(0, 1), (1, 2), (2, 3), (3, 4)]

    On Python 3.10 and above, this is an alias for :func:`itertools.pairwise`.

    N)r   r<   zip)r3   abr4   r4   r5   	_pairwise  s    	
rM   r   c                 c   s   t | E d H  d S N)itertools_pairwiserB   r4   r4   r5   r     s    c                 C   s<   t | tr tdt | | }} t| g| }t|d|iS )zCollect data into fixed-length chunks or blocks.

    >>> list(grouper('ABCDEFG', 3, 'x'))
    [('A', 'B', 'C'), ('D', 'E', 'F'), ('G', 'x', 'x')]

    z+grouper expects iterable as first parameter	fillvalue)
isinstanceintwarningswarnDeprecationWarningr;   r   )r3   r2   rQ   rI   r4   r4   r5   r     s    
 
c                  g   sf   t | }tdd | D }|rbz|D ]}| V  q$W q tk
r^   |d8 }tt||}Y qX qdS )aJ  Yields an item from each iterable, alternating between them.

        >>> list(roundrobin('ABC', 'D', 'EF'))
        ['A', 'D', 'E', 'B', 'F', 'C']

    This function produces the same output as :func:`interleave_longest`, but
    may perform better for some inputs (in particular when the number of
    iterables is small).

    c                 s   s   | ]}t |jV  qd S rO   )r;   __next__).0itr4   r4   r5   	<genexpr><  s     zroundrobin.<locals>.<genexpr>   N)lenr   StopIterationr   )	iterablespendingZnextsr<   r4   r4   r5   r(   /  s    c                    sF    dkrt   fdd|D }t|\}}dd |D dd |D fS )a  
    Returns a 2-tuple of iterables derived from the input iterable.
    The first yields the items that have ``pred(item) == False``.
    The second yields the items that have ``pred(item) == True``.

        >>> is_odd = lambda x: x % 2 != 0
        >>> iterable = range(10)
        >>> even_items, odd_items = partition(is_odd, iterable)
        >>> list(even_items), list(odd_items)
        ([0, 2, 4, 6, 8], [1, 3, 5, 7, 9])

    If *pred* is None, :func:`bool` is used.

        >>> iterable = [0, 1, False, True, '', ' ']
        >>> false_items, true_items = partition(None, iterable)
        >>> list(false_items), list(true_items)
        ([0, False, ''], [1, True, ' '])

    Nc                 3   s   | ]} ||fV  qd S rO   r4   )rX   xrA   r4   r5   rZ   ]  s     zpartition.<locals>.<genexpr>c                 s   s   | ]\}}|s|V  qd S rO   r4   rX   Zcondr`   r4   r4   r5   rZ   `  s      c                 s   s   | ]\}}|r|V  qd S rO   r4   rb   r4   r4   r5   rZ   a  s      )boolr   )rA   r3   Zevaluationst1t2r4   ra   r5   r   F  s    c                    s,   t |  t fddtt d D S )a  Yields all possible subsets of the iterable.

        >>> list(powerset([1, 2, 3]))
        [(), (1,), (2,), (3,), (1, 2), (1, 3), (2, 3), (1, 2, 3)]

    :func:`powerset` will operate on iterables that aren't :class:`set`
    instances, so repeated elements in the input will produce repeated elements
    in the output. Use :func:`unique_everseen` on the input to avoid generating
    duplicates:

        >>> seq = [1, 1, 0]
        >>> list(powerset(seq))
        [(), (1,), (1,), (0,), (1, 1), (1, 0), (1, 0), (1, 1, 0)]
        >>> from more_itertools import unique_everseen
        >>> list(powerset(unique_everseen(seq)))
        [(), (1,), (0,), (1, 0)]

    c                 3   s   | ]}t  |V  qd S rO   )r   )rX   rsr4   r5   rZ   y  s     zpowerset.<locals>.<genexpr>r[   )r0   r   rC   ranger\   rB   r4   rg   r5   r    e  s    c           	   	   c   s   t  }|j}g }|j}|dk	}| D ]Z}|r2||n|}z||krN|| |V  W q" tk
rz   ||krv|| |V  Y q"X q"dS )a  
    Yield unique elements, preserving order.

        >>> list(unique_everseen('AAAABBBCCDAABBB'))
        ['A', 'B', 'C', 'D']
        >>> list(unique_everseen('ABBCcAD', str.lower))
        ['A', 'B', 'C', 'D']

    Sequences with a mix of hashable and unhashable items can be used.
    The function will be slower (i.e., `O(n^2)`) for unhashable items.

    Remember that ``list`` objects are unhashable - you can use the *key*
    parameter to transform the list to a tuple (which is hashable) to
    avoid a slowdown.

        >>> iterable = ([1, 2], [2, 3], [1, 2])
        >>> list(unique_everseen(iterable))  # Slow
        [[1, 2], [2, 3]]
        >>> list(unique_everseen(iterable, key=tuple))  # Faster
        [[1, 2], [2, 3]]

    Similary, you may want to convert unhashable ``set`` objects with
    ``key=frozenset``. For ``dict`` objects,
    ``key=lambda x: frozenset(x.items())`` can be used.

    N)setaddappend	TypeError)	r3   keyZseensetZseenset_addZseenlistZseenlist_addZuse_keyelementkr4   r4   r5   r.   |  s    
c                 C   s   t tt tdt| |S )zYields elements in order, ignoring serial duplicates

    >>> list(unique_justseen('AAAABBBCCDAABBB'))
    ['A', 'B', 'C', 'D', 'A', 'B']
    >>> list(unique_justseen('ABBCcAD', str.lower))
    ['A', 'B', 'C', 'A', 'D']

    r[   )r6   r<   rE   
itemgetterr   )r3   rn   r4   r4   r5   r/     s    	c                 c   s8   z|dk	r| V  |  V  qW n |k
r2   Y nX dS )a  Yields results from a function repeatedly until an exception is raised.

    Converts a call-until-exception interface to an iterator interface.
    Like ``iter(func, sentinel)``, but uses an exception instead of a sentinel
    to end the loop.

        >>> l = [0, 1, 2]
        >>> list(iter_except(l.pop, IndexError))
        [2, 1, 0]

    Multiple exceptions can be specified as a stopping condition:

        >>> l = [1, 2, 3, '...', 4, 5, 6]
        >>> list(iter_except(lambda: 1 + l.pop(), (IndexError, TypeError)))
        [7, 6, 5]
        >>> list(iter_except(lambda: 1 + l.pop(), (IndexError, TypeError)))
        [4, 3, 2]
        >>> list(iter_except(lambda: 1 + l.pop(), (IndexError, TypeError)))
        []

    Nr4   )rG   	exceptionfirstr4   r4   r5   r     s    c                 C   s   t t|| |S )a  
    Returns the first true value in the iterable.

    If no true value is found, returns *default*

    If *pred* is not None, returns the first item for which
    ``pred(item) == True`` .

        >>> first_true(range(10))
        1
        >>> first_true(range(10), pred=lambda x: x > 5)
        6
        >>> first_true(range(10), default='missing', pred=lambda x: x > 9)
        'missing'

    )r<   filter)r3   r>   rA   r4   r4   r5   r     s    r[   )r	   c                 G   s$   dd |D |  }t dd |D S )a  Draw an item at random from each of the input iterables.

        >>> random_product('abc', range(4), 'XYZ')  # doctest:+SKIP
        ('c', 3, 'Z')

    If *repeat* is provided as a keyword argument, that many items will be
    drawn from each iterable.

        >>> random_product('abcd', range(4), repeat=2)  # doctest:+SKIP
        ('a', 2, 'd', 3)

    This equivalent to taking a random selection from
    ``itertools.product(*args, **kwarg)``.

    c                 S   s   g | ]}t |qS r4   rD   rX   poolr4   r4   r5   
<listcomp>  s     z"random_product.<locals>.<listcomp>c                 s   s   | ]}t |V  qd S rO   )r   rv   r4   r4   r5   rZ     s     z!random_product.<locals>.<genexpr>ru   )r	   rI   poolsr4   r4   r5   r&     s    c                 C   s*   t | }|dkrt|n|}t t||S )ab  Return a random *r* length permutation of the elements in *iterable*.

    If *r* is not specified or is ``None``, then *r* defaults to the length of
    *iterable*.

        >>> random_permutation(range(5))  # doctest:+SKIP
        (3, 4, 0, 1, 2)

    This equivalent to taking a random selection from
    ``itertools.permutations(iterable, r)``.

    N)rD   r\   r   )r3   rf   rw   r4   r4   r5   r%     s    c                    s8   t |  t }ttt||}t  fdd|D S )zReturn a random *r* length subsequence of the elements in *iterable*.

        >>> random_combination(range(5), 3)  # doctest:+SKIP
        (2, 3, 4)

    This equivalent to taking a random selection from
    ``itertools.combinations(iterable, r)``.

    c                 3   s   | ]} | V  qd S rO   r4   rX   irw   r4   r5   rZ     s     z%random_combination.<locals>.<genexpr>)rD   r\   sortedr   ri   )r3   rf   r2   indicesr4   r|   r5   r$     s    
c                    s@   t | t t fddt|D }t fdd|D S )aS  Return a random *r* length subsequence of elements in *iterable*,
    allowing individual elements to be repeated.

        >>> random_combination_with_replacement(range(3), 5) # doctest:+SKIP
        (0, 0, 1, 2, 2)

    This equivalent to taking a random selection from
    ``itertools.combinations_with_replacement(iterable, r)``.

    c                 3   s   | ]}t  V  qd S rO   )r   rz   )r2   r4   r5   rZ   +  s     z6random_combination_with_replacement.<locals>.<genexpr>c                 3   s   | ]} | V  qd S rO   r4   rz   r|   r4   r5   rZ   ,  s     )rD   r\   r}   ri   )r3   rf   r~   r4   )r2   rw   r5   r#     s    c           	      C   s   t | }t|}|dk s ||kr$td}t||| }td|d D ]}||| |  | }qD|dk rn||7 }|dk s~||krtg }|r|| | |d |d   }}}||kr||8 }|||  | |d  }}q||d|   qt |S )a  Equivalent to ``list(combinations(iterable, r))[index]``.

    The subsequences of *iterable* that are of length *r* can be ordered
    lexicographically. :func:`nth_combination` computes the subsequence at
    sort position *index* directly, without computing the previous
    subsequences.

        >>> nth_combination(range(5), 3, 5)
        (0, 3, 4)

    ``ValueError`` will be raised If *r* is negative or greater than the length
    of *iterable*.
    ``IndexError`` will be raised if the given *index* is invalid.
    r   r[   )rD   r\   
ValueErrorminri   
IndexErrorrl   )	r3   rf   indexrw   r2   crp   r{   resultr4   r4   r5   r   /  s(     c                 C   s   t | g|S )a  Yield *value*, followed by the elements in *iterator*.

        >>> value = '0'
        >>> iterator = ['1', '2', '3']
        >>> list(prepend(value, iterator))
        ['0', '1', '2', '3']

    To prepend multiple values, see :func:`itertools.chain`
    or :func:`value_chain`.

    )r   )valuer=   r4   r4   r5   r!   Y  s    c                 c   sh   t |ddd }t|}tdg|d| }t| td|d D ]"}|| tttj	||V  q@dS )aB  Convolve the iterable *signal* with the iterable *kernel*.

        >>> signal = (1, 2, 3, 4, 5)
        >>> kernel = [3, 2, 1]
        >>> list(convolve(signal, kernel))
        [3, 8, 14, 20, 26, 14, 5]

    Note: the input arguments are not interchangeable, as the *kernel*
    is immediately consumed and stored.

    Nr   r   r9   r[   )
rD   r\   r   r   r	   rl   r@   r6   rE   rF   )signalkernelr2   windowr`   r4   r4   r5   r   h  s    
c                    s6   t   g  fdd} fdd}| | fS )a  A variant of :func:`takewhile` that allows complete access to the
    remainder of the iterator.

         >>> it = iter('ABCdEfGhI')
         >>> all_upper, remainder = before_and_after(str.isupper, it)
         >>> ''.join(all_upper)
         'ABC'
         >>> ''.join(remainder) # takewhile() would lose the 'd'
         'dEfGhI'

    Note that the first iterator must be fully consumed before the second
    iterator can generate valid results.
    c                  3   s.    D ]$} | r| V  q |   d S qd S rO   )rl   )elemrY   	predicate
transitionr4   r5   true_iterator  s
    
z'before_and_after.<locals>.true_iteratorc                   3   s   E d H   E d H  d S rO   r4   r4   )rY   r   r4   r5   remainder_iterator  s    
z,before_and_after.<locals>.remainder_iterator)r;   )r   rY   r   r   r4   r   r5   r   |  s
    c                 c   s.   t t | D ]\\}}\}}|||fV  qdS )zReturn overlapping triplets from *iterable*.

    >>> list(triplewise('ABCDE'))
    [('A', 'B', 'C'), ('B', 'C', 'D'), ('C', 'D', 'E')]

    NrN   )r3   rK   _rL   r   r4   r4   r5   r-     s    c                 c   sR   t | }tt|||d}t||kr0t|V  |D ]}|| t|V  q4dS )aY  Return a sliding window of width *n* over *iterable*.

        >>> list(sliding_window(range(6), 4))
        [(0, 1, 2, 3), (1, 2, 3, 4), (2, 3, 4, 5)]

    If *iterable* has fewer than *n* items, then nothing is yielded:

        >>> list(sliding_window(range(3), 4))
        []

    For a variant with more features, see :func:`windowed`.
    r9   N)r;   r   r   r\   rD   rl   )r3   r2   rY   r   r`   r4   r4   r5   r)     s    

)r   )N)N)N)N)N)N)N)NN)N)9__doc__rT   collectionsr   	itertoolsr   r   r   r   r   r   r	   r
   r   r   rE   randomr   r   r   __all__r,   r*   r+   r   r   r   rc   r"   r   r   r   r   r   r'   rM   r   rP   ImportErrorr   r(   r   r    r.   r/   r   r   r&   r%   r$   r#   r   r!   r   r   r-   r)   r4   r4   r4   r5   <module>   s   	0$

(







-



* 