Home | History | Annotate | Download | only in test
      1 import sys
      2 import unittest
      3 from test import test_support
      4 from UserList import UserList
      5 
      6 # We do a bit of trickery here to be able to test both the C implementation
      7 # and the Python implementation of the module.
      8 
      9 # Make it impossible to import the C implementation anymore.
     10 sys.modules['_bisect'] = 0
     11 # We must also handle the case that bisect was imported before.
     12 if 'bisect' in sys.modules:
     13     del sys.modules['bisect']
     14 
     15 # Now we can import the module and get the pure Python implementation.
     16 import bisect as py_bisect
     17 
     18 # Restore everything to normal.
     19 del sys.modules['_bisect']
     20 del sys.modules['bisect']
     21 
     22 # This is now the module with the C implementation.
     23 import bisect as c_bisect
     24 
     25 
     26 class Range(object):
     27     """A trivial xrange()-like object without any integer width limitations."""
     28     def __init__(self, start, stop):
     29         self.start = start
     30         self.stop = stop
     31         self.last_insert = None
     32 
     33     def __len__(self):
     34         return self.stop - self.start
     35 
     36     def __getitem__(self, idx):
     37         n = self.stop - self.start
     38         if idx < 0:
     39             idx += n
     40         if idx >= n:
     41             raise IndexError(idx)
     42         return self.start + idx
     43 
     44     def insert(self, idx, item):
     45         self.last_insert = idx, item
     46 
     47 
     48 class TestBisect(unittest.TestCase):
     49     module = None
     50 
     51     def setUp(self):
     52         self.precomputedCases = [
     53             (self.module.bisect_right, [], 1, 0),
     54             (self.module.bisect_right, [1], 0, 0),
     55             (self.module.bisect_right, [1], 1, 1),
     56             (self.module.bisect_right, [1], 2, 1),
     57             (self.module.bisect_right, [1, 1], 0, 0),
     58             (self.module.bisect_right, [1, 1], 1, 2),
     59             (self.module.bisect_right, [1, 1], 2, 2),
     60             (self.module.bisect_right, [1, 1, 1], 0, 0),
     61             (self.module.bisect_right, [1, 1, 1], 1, 3),
     62             (self.module.bisect_right, [1, 1, 1], 2, 3),
     63             (self.module.bisect_right, [1, 1, 1, 1], 0, 0),
     64             (self.module.bisect_right, [1, 1, 1, 1], 1, 4),
     65             (self.module.bisect_right, [1, 1, 1, 1], 2, 4),
     66             (self.module.bisect_right, [1, 2], 0, 0),
     67             (self.module.bisect_right, [1, 2], 1, 1),
     68             (self.module.bisect_right, [1, 2], 1.5, 1),
     69             (self.module.bisect_right, [1, 2], 2, 2),
     70             (self.module.bisect_right, [1, 2], 3, 2),
     71             (self.module.bisect_right, [1, 1, 2, 2], 0, 0),
     72             (self.module.bisect_right, [1, 1, 2, 2], 1, 2),
     73             (self.module.bisect_right, [1, 1, 2, 2], 1.5, 2),
     74             (self.module.bisect_right, [1, 1, 2, 2], 2, 4),
     75             (self.module.bisect_right, [1, 1, 2, 2], 3, 4),
     76             (self.module.bisect_right, [1, 2, 3], 0, 0),
     77             (self.module.bisect_right, [1, 2, 3], 1, 1),
     78             (self.module.bisect_right, [1, 2, 3], 1.5, 1),
     79             (self.module.bisect_right, [1, 2, 3], 2, 2),
     80             (self.module.bisect_right, [1, 2, 3], 2.5, 2),
     81             (self.module.bisect_right, [1, 2, 3], 3, 3),
     82             (self.module.bisect_right, [1, 2, 3], 4, 3),
     83             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 0, 0),
     84             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 1, 1),
     85             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 1.5, 1),
     86             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 2, 3),
     87             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 2.5, 3),
     88             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 3, 6),
     89             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 3.5, 6),
     90             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 4, 10),
     91             (self.module.bisect_right, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 5, 10),
     92 
     93             (self.module.bisect_left, [], 1, 0),
     94             (self.module.bisect_left, [1], 0, 0),
     95             (self.module.bisect_left, [1], 1, 0),
     96             (self.module.bisect_left, [1], 2, 1),
     97             (self.module.bisect_left, [1, 1], 0, 0),
     98             (self.module.bisect_left, [1, 1], 1, 0),
     99             (self.module.bisect_left, [1, 1], 2, 2),
    100             (self.module.bisect_left, [1, 1, 1], 0, 0),
    101             (self.module.bisect_left, [1, 1, 1], 1, 0),
    102             (self.module.bisect_left, [1, 1, 1], 2, 3),
    103             (self.module.bisect_left, [1, 1, 1, 1], 0, 0),
    104             (self.module.bisect_left, [1, 1, 1, 1], 1, 0),
    105             (self.module.bisect_left, [1, 1, 1, 1], 2, 4),
    106             (self.module.bisect_left, [1, 2], 0, 0),
    107             (self.module.bisect_left, [1, 2], 1, 0),
    108             (self.module.bisect_left, [1, 2], 1.5, 1),
    109             (self.module.bisect_left, [1, 2], 2, 1),
    110             (self.module.bisect_left, [1, 2], 3, 2),
    111             (self.module.bisect_left, [1, 1, 2, 2], 0, 0),
    112             (self.module.bisect_left, [1, 1, 2, 2], 1, 0),
    113             (self.module.bisect_left, [1, 1, 2, 2], 1.5, 2),
    114             (self.module.bisect_left, [1, 1, 2, 2], 2, 2),
    115             (self.module.bisect_left, [1, 1, 2, 2], 3, 4),
    116             (self.module.bisect_left, [1, 2, 3], 0, 0),
    117             (self.module.bisect_left, [1, 2, 3], 1, 0),
    118             (self.module.bisect_left, [1, 2, 3], 1.5, 1),
    119             (self.module.bisect_left, [1, 2, 3], 2, 1),
    120             (self.module.bisect_left, [1, 2, 3], 2.5, 2),
    121             (self.module.bisect_left, [1, 2, 3], 3, 2),
    122             (self.module.bisect_left, [1, 2, 3], 4, 3),
    123             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 0, 0),
    124             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 1, 0),
    125             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 1.5, 1),
    126             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 2, 1),
    127             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 2.5, 3),
    128             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 3, 3),
    129             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 3.5, 6),
    130             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 4, 6),
    131             (self.module.bisect_left, [1, 2, 2, 3, 3, 3, 4, 4, 4, 4], 5, 10)
    132         ]
    133 
    134     def test_precomputed(self):
    135         for func, data, elem, expected in self.precomputedCases:
    136             self.assertEqual(func(data, elem), expected)
    137             self.assertEqual(func(UserList(data), elem), expected)
    138 
    139     def test_negative_lo(self):
    140         # Issue 3301
    141         mod = self.module
    142         self.assertRaises(ValueError, mod.bisect_left, [1, 2, 3], 5, -1, 3),
    143         self.assertRaises(ValueError, mod.bisect_right, [1, 2, 3], 5, -1, 3),
    144         self.assertRaises(ValueError, mod.insort_left, [1, 2, 3], 5, -1, 3),
    145         self.assertRaises(ValueError, mod.insort_right, [1, 2, 3], 5, -1, 3),
    146 
    147     def test_large_range(self):
    148         # Issue 13496
    149         mod = self.module
    150         n = sys.maxsize
    151         try:
    152             data = xrange(n-1)
    153         except OverflowError:
    154             self.skipTest("can't create a xrange() object of size `sys.maxsize`")
    155         self.assertEqual(mod.bisect_left(data, n-3), n-3)
    156         self.assertEqual(mod.bisect_right(data, n-3), n-2)
    157         self.assertEqual(mod.bisect_left(data, n-3, n-10, n), n-3)
    158         self.assertEqual(mod.bisect_right(data, n-3, n-10, n), n-2)
    159 
    160     def test_large_pyrange(self):
    161         # Same as above, but without C-imposed limits on range() parameters
    162         mod = self.module
    163         n = sys.maxsize
    164         data = Range(0, n-1)
    165         self.assertEqual(mod.bisect_left(data, n-3), n-3)
    166         self.assertEqual(mod.bisect_right(data, n-3), n-2)
    167         self.assertEqual(mod.bisect_left(data, n-3, n-10, n), n-3)
    168         self.assertEqual(mod.bisect_right(data, n-3, n-10, n), n-2)
    169         x = n - 100
    170         mod.insort_left(data, x, x - 50, x + 50)
    171         self.assertEqual(data.last_insert, (x, x))
    172         x = n - 200
    173         mod.insort_right(data, x, x - 50, x + 50)
    174         self.assertEqual(data.last_insert, (x + 1, x))
    175 
    176     def test_random(self, n=25):
    177         from random import randrange
    178         for i in xrange(n):
    179             data = [randrange(0, n, 2) for j in xrange(i)]
    180             data.sort()
    181             elem = randrange(-1, n+1)
    182             ip = self.module.bisect_left(data, elem)
    183             if ip < len(data):
    184                 self.assertTrue(elem <= data[ip])
    185             if ip > 0:
    186                 self.assertTrue(data[ip-1] < elem)
    187             ip = self.module.bisect_right(data, elem)
    188             if ip < len(data):
    189                 self.assertTrue(elem < data[ip])
    190             if ip > 0:
    191                 self.assertTrue(data[ip-1] <= elem)
    192 
    193     def test_optionalSlicing(self):
    194         for func, data, elem, expected in self.precomputedCases:
    195             for lo in xrange(4):
    196                 lo = min(len(data), lo)
    197                 for hi in xrange(3,8):
    198                     hi = min(len(data), hi)
    199                     ip = func(data, elem, lo, hi)
    200                     self.assertTrue(lo <= ip <= hi)
    201                     if func is self.module.bisect_left and ip < hi:
    202                         self.assertTrue(elem <= data[ip])
    203                     if func is self.module.bisect_left and ip > lo:
    204                         self.assertTrue(data[ip-1] < elem)
    205                     if func is self.module.bisect_right and ip < hi:
    206                         self.assertTrue(elem < data[ip])
    207                     if func is self.module.bisect_right and ip > lo:
    208                         self.assertTrue(data[ip-1] <= elem)
    209                     self.assertEqual(ip, max(lo, min(hi, expected)))
    210 
    211     def test_backcompatibility(self):
    212         self.assertEqual(self.module.bisect, self.module.bisect_right)
    213 
    214     def test_keyword_args(self):
    215         data = [10, 20, 30, 40, 50]
    216         self.assertEqual(self.module.bisect_left(a=data, x=25, lo=1, hi=3), 2)
    217         self.assertEqual(self.module.bisect_right(a=data, x=25, lo=1, hi=3), 2)
    218         self.assertEqual(self.module.bisect(a=data, x=25, lo=1, hi=3), 2)
    219         self.module.insort_left(a=data, x=25, lo=1, hi=3)
    220         self.module.insort_right(a=data, x=25, lo=1, hi=3)
    221         self.module.insort(a=data, x=25, lo=1, hi=3)
    222         self.assertEqual(data, [10, 20, 25, 25, 25, 30, 40, 50])
    223 
    224 class TestBisectPython(TestBisect):
    225     module = py_bisect
    226 
    227 class TestBisectC(TestBisect):
    228     module = c_bisect
    229 
    230 #==============================================================================
    231 
    232 class TestInsort(unittest.TestCase):
    233     module = None
    234 
    235     def test_vsBuiltinSort(self, n=500):
    236         from random import choice
    237         for insorted in (list(), UserList()):
    238             for i in xrange(n):
    239                 digit = choice("0123456789")
    240                 if digit in "02468":
    241                     f = self.module.insort_left
    242                 else:
    243                     f = self.module.insort_right
    244                 f(insorted, digit)
    245             self.assertEqual(sorted(insorted), insorted)
    246 
    247     def test_backcompatibility(self):
    248         self.assertEqual(self.module.insort, self.module.insort_right)
    249 
    250     def test_listDerived(self):
    251         class List(list):
    252             data = []
    253             def insert(self, index, item):
    254                 self.data.insert(index, item)
    255 
    256         lst = List()
    257         self.module.insort_left(lst, 10)
    258         self.module.insort_right(lst, 5)
    259         self.assertEqual([5, 10], lst.data)
    260 
    261 class TestInsortPython(TestInsort):
    262     module = py_bisect
    263 
    264 class TestInsortC(TestInsort):
    265     module = c_bisect
    266 
    267 #==============================================================================
    268 
    269 
    270 class LenOnly:
    271     "Dummy sequence class defining __len__ but not __getitem__."
    272     def __len__(self):
    273         return 10
    274 
    275 class GetOnly:
    276     "Dummy sequence class defining __getitem__ but not __len__."
    277     def __getitem__(self, ndx):
    278         return 10
    279 
    280 class CmpErr:
    281     "Dummy element that always raises an error during comparison"
    282     def __cmp__(self, other):
    283         raise ZeroDivisionError
    284 
    285 class TestErrorHandling(unittest.TestCase):
    286     module = None
    287 
    288     def test_non_sequence(self):
    289         for f in (self.module.bisect_left, self.module.bisect_right,
    290                   self.module.insort_left, self.module.insort_right):
    291             self.assertRaises(TypeError, f, 10, 10)
    292 
    293     def test_len_only(self):
    294         for f in (self.module.bisect_left, self.module.bisect_right,
    295                   self.module.insort_left, self.module.insort_right):
    296             self.assertRaises(AttributeError, f, LenOnly(), 10)
    297 
    298     def test_get_only(self):
    299         for f in (self.module.bisect_left, self.module.bisect_right,
    300                   self.module.insort_left, self.module.insort_right):
    301             self.assertRaises(AttributeError, f, GetOnly(), 10)
    302 
    303     def test_cmp_err(self):
    304         seq = [CmpErr(), CmpErr(), CmpErr()]
    305         for f in (self.module.bisect_left, self.module.bisect_right,
    306                   self.module.insort_left, self.module.insort_right):
    307             self.assertRaises(ZeroDivisionError, f, seq, 10)
    308 
    309     def test_arg_parsing(self):
    310         for f in (self.module.bisect_left, self.module.bisect_right,
    311                   self.module.insort_left, self.module.insort_right):
    312             self.assertRaises(TypeError, f, 10)
    313 
    314 class TestErrorHandlingPython(TestErrorHandling):
    315     module = py_bisect
    316 
    317 class TestErrorHandlingC(TestErrorHandling):
    318     module = c_bisect
    319 
    320 #==============================================================================
    321 
    322 libreftest = """
    323 Example from the Library Reference:  Doc/library/bisect.rst
    324 
    325 The bisect() function is generally useful for categorizing numeric data.
    326 This example uses bisect() to look up a letter grade for an exam total
    327 (say) based on a set of ordered numeric breakpoints: 85 and up is an `A',
    328 75..84 is a `B', etc.
    329 
    330     >>> grades = "FEDCBA"
    331     >>> breakpoints = [30, 44, 66, 75, 85]
    332     >>> from bisect import bisect
    333     >>> def grade(total):
    334     ...           return grades[bisect(breakpoints, total)]
    335     ...
    336     >>> grade(66)
    337     'C'
    338     >>> map(grade, [33, 99, 77, 44, 12, 88])
    339     ['E', 'A', 'B', 'D', 'F', 'A']
    340 
    341 """
    342 
    343 #------------------------------------------------------------------------------
    344 
    345 __test__ = {'libreftest' : libreftest}
    346 
    347 def test_main(verbose=None):
    348     from test import test_bisect
    349 
    350     test_classes = [TestBisectPython, TestBisectC,
    351                     TestInsortPython, TestInsortC,
    352                     TestErrorHandlingPython, TestErrorHandlingC]
    353 
    354     test_support.run_unittest(*test_classes)
    355     test_support.run_doctest(test_bisect, verbose)
    356 
    357     # verify reference counting
    358     if verbose and hasattr(sys, "gettotalrefcount"):
    359         import gc
    360         counts = [None] * 5
    361         for i in xrange(len(counts)):
    362             test_support.run_unittest(*test_classes)
    363             gc.collect()
    364             counts[i] = sys.gettotalrefcount()
    365         print counts
    366 
    367 if __name__ == "__main__":
    368     test_main(verbose=True)
    369