2005-01-27 07:59:08 +01:00
|
|
|
###
|
|
|
|
# Copyright (c) 2002-2005, Jeremiah Fincher
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# * Neither the name of the author of this software nor the name of
|
|
|
|
# contributors to this software may be used to endorse or promote products
|
|
|
|
# derived from this software without specific prior written consent.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
###
|
|
|
|
|
2005-02-16 03:15:51 +01:00
|
|
|
from __future__ import division
|
|
|
|
|
2005-02-04 06:55:04 +01:00
|
|
|
import sys
|
2005-01-27 07:59:08 +01:00
|
|
|
import random
|
|
|
|
|
|
|
|
from itertools import *
|
|
|
|
|
2013-08-24 11:28:29 +02:00
|
|
|
# For old plugins
|
|
|
|
ifilter = filter
|
|
|
|
def ifilterfalse(p, L):
|
|
|
|
return ifilter(lambda x:not p(x), L)
|
|
|
|
imap = map
|
|
|
|
|
2005-01-27 07:59:08 +01:00
|
|
|
def len(iterable):
|
|
|
|
"""Returns the length of an iterator."""
|
|
|
|
i = 0
|
|
|
|
for _ in iterable:
|
|
|
|
i += 1
|
|
|
|
return i
|
|
|
|
|
|
|
|
def trueCycle(iterable):
|
2014-01-20 15:13:01 +01:00
|
|
|
while True:
|
2005-01-27 07:59:08 +01:00
|
|
|
yielded = False
|
|
|
|
for x in iterable:
|
|
|
|
yield x
|
|
|
|
yielded = True
|
|
|
|
if not yielded:
|
|
|
|
raise StopIteration
|
|
|
|
|
|
|
|
def partition(p, iterable):
|
|
|
|
"""Partitions an iterable based on a predicate p.
|
|
|
|
Returns a (yes,no) tuple"""
|
|
|
|
no = []
|
|
|
|
yes = []
|
|
|
|
for elt in iterable:
|
|
|
|
if p(elt):
|
|
|
|
yes.append(elt)
|
|
|
|
else:
|
|
|
|
no.append(elt)
|
|
|
|
return (yes, no)
|
|
|
|
|
|
|
|
def any(p, iterable):
|
|
|
|
"""Returns true if any element in iterable satisfies predicate p."""
|
|
|
|
for elt in ifilter(p, iterable):
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def all(p, iterable):
|
|
|
|
"""Returns true if all elements in iterable satisfy predicate p."""
|
|
|
|
for elt in ifilterfalse(p, iterable):
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
|
|
|
|
def choice(iterable):
|
|
|
|
if isinstance(iterable, (list, tuple)):
|
|
|
|
return random.choice(iterable)
|
|
|
|
else:
|
|
|
|
n = 1
|
2012-08-05 09:59:42 +02:00
|
|
|
found = False
|
2005-01-27 07:59:08 +01:00
|
|
|
for x in iterable:
|
|
|
|
if random.random() < 1/n:
|
|
|
|
ret = x
|
2012-08-05 09:59:42 +02:00
|
|
|
found = True
|
2005-01-27 07:59:08 +01:00
|
|
|
n += 1
|
2012-08-05 09:59:42 +02:00
|
|
|
if not found:
|
2005-01-27 07:59:08 +01:00
|
|
|
raise IndexError
|
|
|
|
return ret
|
|
|
|
|
|
|
|
def flatten(iterable, strings=False):
|
|
|
|
"""Flattens a list of lists into a single list. See the test for examples.
|
|
|
|
"""
|
|
|
|
for elt in iterable:
|
|
|
|
if not strings and isinstance(elt, basestring):
|
|
|
|
yield elt
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
for x in flatten(elt):
|
|
|
|
yield x
|
|
|
|
except TypeError:
|
|
|
|
yield elt
|
|
|
|
|
|
|
|
def split(isSeparator, iterable, maxsplit=-1, yieldEmpty=False):
|
|
|
|
"""split(isSeparator, iterable, maxsplit=-1, yieldEmpty=False)
|
|
|
|
|
|
|
|
Splits an iterator based on a predicate isSeparator."""
|
|
|
|
if isinstance(isSeparator, basestring):
|
|
|
|
f = lambda s: s == isSeparator
|
|
|
|
else:
|
|
|
|
f = isSeparator
|
|
|
|
acc = []
|
|
|
|
for element in iterable:
|
|
|
|
if maxsplit == 0 or not f(element):
|
|
|
|
acc.append(element)
|
|
|
|
else:
|
|
|
|
maxsplit -= 1
|
|
|
|
if acc or yieldEmpty:
|
|
|
|
yield acc
|
|
|
|
acc = []
|
|
|
|
if acc or yieldEmpty:
|
|
|
|
yield acc
|
|
|
|
|
|
|
|
def ilen(iterable):
|
|
|
|
i = 0
|
|
|
|
for _ in iterable:
|
|
|
|
i += 1
|
|
|
|
return i
|
|
|
|
|
2012-08-04 17:26:59 +02:00
|
|
|
def startswith(long_, short):
|
|
|
|
longI = iter(long_)
|
2005-02-18 00:30:54 +01:00
|
|
|
shortI = iter(short)
|
|
|
|
try:
|
|
|
|
while True:
|
2014-01-20 15:13:01 +01:00
|
|
|
if next(shortI) != next(longI):
|
2005-02-18 00:30:54 +01:00
|
|
|
return False
|
|
|
|
except StopIteration:
|
|
|
|
return True
|
|
|
|
|
2005-05-30 21:19:11 +02:00
|
|
|
def limited(iterable, limit):
|
|
|
|
i = limit
|
|
|
|
iterable = iter(iterable)
|
|
|
|
try:
|
|
|
|
while i:
|
2014-01-20 15:13:01 +01:00
|
|
|
yield next(iterable)
|
2005-05-30 21:19:11 +02:00
|
|
|
i -= 1
|
|
|
|
except StopIteration:
|
|
|
|
raise ValueError, 'Expected %s elements in iterable (%r), got %s.' % \
|
|
|
|
(limit, iterable, limit-i)
|
|
|
|
|
2006-02-11 16:52:51 +01:00
|
|
|
# vim:set shiftwidth=4 softtabstop=4 expandtab textwidth=79:
|