mirror of
				https://github.com/python/cpython.git
				synced 2025-11-03 23:21:29 +00:00 
			
		
		
		
	
		
			
	
	
		
			247 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			247 lines
		
	
	
	
		
			7.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| 
								 | 
							
								# Test iterators.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import unittest
							 | 
						||
| 
								 | 
							
								from test_support import run_unittest, TESTFN, unlink
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								# Test result of triple loop (too big to inline)
							 | 
						||
| 
								 | 
							
								TRIPLETS = [(0, 0, 0), (0, 0, 1), (0, 0, 2),
							 | 
						||
| 
								 | 
							
								            (0, 1, 0), (0, 1, 1), (0, 1, 2),
							 | 
						||
| 
								 | 
							
								            (0, 2, 0), (0, 2, 1), (0, 2, 2),
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            (1, 0, 0), (1, 0, 1), (1, 0, 2),
							 | 
						||
| 
								 | 
							
								            (1, 1, 0), (1, 1, 1), (1, 1, 2),
							 | 
						||
| 
								 | 
							
								            (1, 2, 0), (1, 2, 1), (1, 2, 2),
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            (2, 0, 0), (2, 0, 1), (2, 0, 2),
							 | 
						||
| 
								 | 
							
								            (2, 1, 0), (2, 1, 1), (2, 1, 2),
							 | 
						||
| 
								 | 
							
								            (2, 2, 0), (2, 2, 1), (2, 2, 2)]
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								# Helper classes
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class BasicIterClass:
							 | 
						||
| 
								 | 
							
								    def __init__(self, n):
							 | 
						||
| 
								 | 
							
								        self.n = n
							 | 
						||
| 
								 | 
							
								        self.i = 0
							 | 
						||
| 
								 | 
							
								    def next(self):
							 | 
						||
| 
								 | 
							
								        res = self.i
							 | 
						||
| 
								 | 
							
								        if res >= self.n:
							 | 
						||
| 
								 | 
							
								            raise StopIteration
							 | 
						||
| 
								 | 
							
								        self.i = res + 1
							 | 
						||
| 
								 | 
							
								        return res
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class IteratingSequenceClass:
							 | 
						||
| 
								 | 
							
								    def __init__(self, n):
							 | 
						||
| 
								 | 
							
								        self.n = n
							 | 
						||
| 
								 | 
							
								    def __iter__(self):
							 | 
						||
| 
								 | 
							
								        return BasicIterClass(self.n)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class SequenceClass:
							 | 
						||
| 
								 | 
							
								    def __init__(self, n):
							 | 
						||
| 
								 | 
							
								        self.n = n
							 | 
						||
| 
								 | 
							
								    def __getitem__(self, i):
							 | 
						||
| 
								 | 
							
								        if 0 <= i < self.n:
							 | 
						||
| 
								 | 
							
								            return i
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            raise IndexError
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								# Main test suite
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class TestCase(unittest.TestCase):
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Helper to check that an iterator returns a given sequence
							 | 
						||
| 
								 | 
							
								    def check_iterator(self, it, seq):
							 | 
						||
| 
								 | 
							
								        res = []
							 | 
						||
| 
								 | 
							
								        while 1:
							 | 
						||
| 
								 | 
							
								            try:
							 | 
						||
| 
								 | 
							
								                val = it.next()
							 | 
						||
| 
								 | 
							
								            except StopIteration:
							 | 
						||
| 
								 | 
							
								                break
							 | 
						||
| 
								 | 
							
								            res.append(val)
							 | 
						||
| 
								 | 
							
								        self.assertEqual(res, seq)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Helper to check that a for loop generates a given sequence
							 | 
						||
| 
								 | 
							
								    def check_for_loop(self, expr, seq):
							 | 
						||
| 
								 | 
							
								        res = []
							 | 
						||
| 
								 | 
							
								        for val in expr:
							 | 
						||
| 
								 | 
							
								            res.append(val)
							 | 
						||
| 
								 | 
							
								        self.assertEqual(res, seq)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test basic use of iter() function
							 | 
						||
| 
								 | 
							
								    def test_iter_basic(self):
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(range(10)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test that iter(iter(x)) is the same as iter(x)
							 | 
						||
| 
								 | 
							
								    def test_iter_idempotency(self):
							 | 
						||
| 
								 | 
							
								        seq = range(10)
							 | 
						||
| 
								 | 
							
								        it = iter(seq)
							 | 
						||
| 
								 | 
							
								        it2 = iter(it)
							 | 
						||
| 
								 | 
							
								        self.assert_(it is it2)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test that for loops over iterators work
							 | 
						||
| 
								 | 
							
								    def test_iter_for_loop(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter(range(10)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test several independent iterators over the same list
							 | 
						||
| 
								 | 
							
								    def test_iter_independence(self):
							 | 
						||
| 
								 | 
							
								        seq = range(3)
							 | 
						||
| 
								 | 
							
								        res = []
							 | 
						||
| 
								 | 
							
								        for i in iter(seq):
							 | 
						||
| 
								 | 
							
								            for j in iter(seq):
							 | 
						||
| 
								 | 
							
								                for k in iter(seq):
							 | 
						||
| 
								 | 
							
								                    res.append((i, j, k))
							 | 
						||
| 
								 | 
							
								        self.assertEqual(res, TRIPLETS)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test triple list comprehension using iterators
							 | 
						||
| 
								 | 
							
								    def test_nested_comprehensions_iter(self):
							 | 
						||
| 
								 | 
							
								        seq = range(3)
							 | 
						||
| 
								 | 
							
								        res = [(i, j, k)
							 | 
						||
| 
								 | 
							
								               for i in iter(seq) for j in iter(seq) for k in iter(seq)]
							 | 
						||
| 
								 | 
							
								        self.assertEqual(res, TRIPLETS)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test triple list comprehension without iterators
							 | 
						||
| 
								 | 
							
								    def test_nested_comprehensions_for(self):
							 | 
						||
| 
								 | 
							
								        seq = range(3)
							 | 
						||
| 
								 | 
							
								        res = [(i, j, k) for i in seq for j in seq for k in seq]
							 | 
						||
| 
								 | 
							
								        self.assertEqual(res, TRIPLETS)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a class with __iter__ in a for loop
							 | 
						||
| 
								 | 
							
								    def test_iter_class_for(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(IteratingSequenceClass(10), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a class with __iter__ with explicit iter()
							 | 
						||
| 
								 | 
							
								    def test_iter_class_iter(self):
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(IteratingSequenceClass(10)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test for loop on a sequence class without __iter__
							 | 
						||
| 
								 | 
							
								    def test_seq_class_for(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(SequenceClass(10), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test iter() on a sequence class without __iter__
							 | 
						||
| 
								 | 
							
								    def test_seq_class_iter(self):
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(SequenceClass(10)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test two-argument iter() with callable instance
							 | 
						||
| 
								 | 
							
								    def test_iter_callable(self):
							 | 
						||
| 
								 | 
							
								        class C:
							 | 
						||
| 
								 | 
							
								            def __init__(self):
							 | 
						||
| 
								 | 
							
								                self.i = 0
							 | 
						||
| 
								 | 
							
								            def __call__(self):
							 | 
						||
| 
								 | 
							
								                i = self.i
							 | 
						||
| 
								 | 
							
								                self.i = i + 1
							 | 
						||
| 
								 | 
							
								                if i > 100:
							 | 
						||
| 
								 | 
							
								                    raise IndexError # Emergency stop
							 | 
						||
| 
								 | 
							
								                return i
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(C(), 10), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test two-argument iter() with function
							 | 
						||
| 
								 | 
							
								    def test_iter_function(self):
							 | 
						||
| 
								 | 
							
								        def spam(state=[0]):
							 | 
						||
| 
								 | 
							
								            i = state[0]
							 | 
						||
| 
								 | 
							
								            state[0] = i+1
							 | 
						||
| 
								 | 
							
								            return i
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(spam, 10), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test two-argument iter() with function that raises StopIteration
							 | 
						||
| 
								 | 
							
								    def test_iter_function_stop(self):
							 | 
						||
| 
								 | 
							
								        def spam(state=[0]):
							 | 
						||
| 
								 | 
							
								            i = state[0]
							 | 
						||
| 
								 | 
							
								            if i == 10:
							 | 
						||
| 
								 | 
							
								                raise StopIteration
							 | 
						||
| 
								 | 
							
								            state[0] = i+1
							 | 
						||
| 
								 | 
							
								            return i
							 | 
						||
| 
								 | 
							
								        self.check_iterator(iter(spam, 20), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test exception propagation through function iterator
							 | 
						||
| 
								 | 
							
								    def test_exception_function(self):
							 | 
						||
| 
								 | 
							
								        def spam(state=[0]):
							 | 
						||
| 
								 | 
							
								            i = state[0]
							 | 
						||
| 
								 | 
							
								            state[0] = i+1
							 | 
						||
| 
								 | 
							
								            if i == 10:
							 | 
						||
| 
								 | 
							
								                raise RuntimeError
							 | 
						||
| 
								 | 
							
								            return i
							 | 
						||
| 
								 | 
							
								        res = []
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            for x in iter(spam, 20):
							 | 
						||
| 
								 | 
							
								                res.append(x)
							 | 
						||
| 
								 | 
							
								        except RuntimeError:
							 | 
						||
| 
								 | 
							
								            self.assertEqual(res, range(10))
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            self.fail("should have raised RuntimeError")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test exception propagation through sequence iterator
							 | 
						||
| 
								 | 
							
								    def test_exception_sequence(self):
							 | 
						||
| 
								 | 
							
								        class MySequenceClass(SequenceClass):
							 | 
						||
| 
								 | 
							
								            def __getitem__(self, i):
							 | 
						||
| 
								 | 
							
								                if i == 10:
							 | 
						||
| 
								 | 
							
								                    raise RuntimeError
							 | 
						||
| 
								 | 
							
								                return SequenceClass.__getitem__(self, i)
							 | 
						||
| 
								 | 
							
								        res = []
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            for x in MySequenceClass(20):
							 | 
						||
| 
								 | 
							
								                res.append(x)
							 | 
						||
| 
								 | 
							
								        except RuntimeError:
							 | 
						||
| 
								 | 
							
								            self.assertEqual(res, range(10))
							 | 
						||
| 
								 | 
							
								        else:
							 | 
						||
| 
								 | 
							
								            self.fail("should have raised RuntimeError")
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test for StopIteration from __getitem__
							 | 
						||
| 
								 | 
							
								    def test_stop_sequence(self):
							 | 
						||
| 
								 | 
							
								        class MySequenceClass(SequenceClass):
							 | 
						||
| 
								 | 
							
								            def __getitem__(self, i):
							 | 
						||
| 
								 | 
							
								                if i == 10:
							 | 
						||
| 
								 | 
							
								                    raise StopIteration
							 | 
						||
| 
								 | 
							
								                return SequenceClass.__getitem__(self, i)
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(MySequenceClass(20), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a big range
							 | 
						||
| 
								 | 
							
								    def test_iter_big_range(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter(range(10000)), range(10000))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test an empty list
							 | 
						||
| 
								 | 
							
								    def test_iter_empty(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter([]), [])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a tuple
							 | 
						||
| 
								 | 
							
								    def test_iter_tuple(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter((0,1,2,3,4,5,6,7,8,9)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test an xrange
							 | 
						||
| 
								 | 
							
								    def test_iter_xrange(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter(xrange(10)), range(10))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a string
							 | 
						||
| 
								 | 
							
								    def test_iter_string(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter("abcde"), ["a", "b", "c", "d", "e"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a Unicode string
							 | 
						||
| 
								 | 
							
								    def test_iter_unicode(self):
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(iter(u"abcde"), [u"a", u"b", u"c", u"d", u"e"])
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a directory
							 | 
						||
| 
								 | 
							
								    def test_iter_dict(self):
							 | 
						||
| 
								 | 
							
								        dict = {}
							 | 
						||
| 
								 | 
							
								        for i in range(10):
							 | 
						||
| 
								 | 
							
								            dict[i] = None
							 | 
						||
| 
								 | 
							
								        self.check_for_loop(dict, dict.keys())
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    # Test a file
							 | 
						||
| 
								 | 
							
								    def test_iter_file(self):
							 | 
						||
| 
								 | 
							
								        f = open(TESTFN, "w")
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            for i in range(5):
							 | 
						||
| 
								 | 
							
								                f.write("%d\n" % i)
							 | 
						||
| 
								 | 
							
								        finally:
							 | 
						||
| 
								 | 
							
								            f.close()
							 | 
						||
| 
								 | 
							
								        f = open(TESTFN, "r")
							 | 
						||
| 
								 | 
							
								        try:
							 | 
						||
| 
								 | 
							
								            self.check_for_loop(f, ["0\n", "1\n", "2\n", "3\n", "4\n"])
							 | 
						||
| 
								 | 
							
								            self.check_for_loop(f, [])
							 | 
						||
| 
								 | 
							
								        finally:
							 | 
						||
| 
								 | 
							
								            f.close()
							 | 
						||
| 
								 | 
							
								            try:
							 | 
						||
| 
								 | 
							
								                unlink(TESTFN)
							 | 
						||
| 
								 | 
							
								            except OSError:
							 | 
						||
| 
								 | 
							
								                pass
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								run_unittest(TestCase)
							 |