mirror of
				https://github.com/python/cpython.git
				synced 2025-10-24 18:33:49 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			222 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
	
		
			4.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import doctest
 | |
| import unittest
 | |
| 
 | |
| 
 | |
| doctests = """
 | |
| 
 | |
| Unpack tuple
 | |
| 
 | |
|     >>> t = (1, 2, 3)
 | |
|     >>> a, b, c = t
 | |
|     >>> a == 1 and b == 2 and c == 3
 | |
|     True
 | |
| 
 | |
| Unpack list
 | |
| 
 | |
|     >>> l = [4, 5, 6]
 | |
|     >>> a, b, c = l
 | |
|     >>> a == 4 and b == 5 and c == 6
 | |
|     True
 | |
| 
 | |
| Unpack dict
 | |
| 
 | |
|     >>> d = {4: 'four', 5: 'five', 6: 'six'}
 | |
|     >>> a, b, c = d
 | |
|     >>> a == 4 and b == 5 and c == 6
 | |
|     True
 | |
| 
 | |
| Unpack implied tuple
 | |
| 
 | |
|     >>> a, b, c = 7, 8, 9
 | |
|     >>> a == 7 and b == 8 and c == 9
 | |
|     True
 | |
| 
 | |
| Unpack string... fun!
 | |
| 
 | |
|     >>> a, b, c = 'one'
 | |
|     >>> a == 'o' and b == 'n' and c == 'e'
 | |
|     True
 | |
| 
 | |
| Unpack generic sequence
 | |
| 
 | |
|     >>> class Seq:
 | |
|     ...     def __getitem__(self, i):
 | |
|     ...         if i >= 0 and i < 3: return i
 | |
|     ...         raise IndexError
 | |
|     ...
 | |
|     >>> a, b, c = Seq()
 | |
|     >>> a == 0 and b == 1 and c == 2
 | |
|     True
 | |
| 
 | |
| Single element unpacking, with extra syntax
 | |
| 
 | |
|     >>> st = (99,)
 | |
|     >>> sl = [100]
 | |
|     >>> a, = st
 | |
|     >>> a
 | |
|     99
 | |
|     >>> b, = sl
 | |
|     >>> b
 | |
|     100
 | |
| 
 | |
| Now for some failures
 | |
| 
 | |
| Unpacking non-sequence
 | |
| 
 | |
|     >>> a, b, c = 7
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     TypeError: cannot unpack non-iterable int object
 | |
| 
 | |
| Unpacking tuple of wrong size
 | |
| 
 | |
|     >>> a, b = t
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 2, got 3)
 | |
| 
 | |
| Unpacking tuple of wrong size
 | |
| 
 | |
|     >>> a, b = l
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 2, got 3)
 | |
| 
 | |
| Unpacking sequence too short
 | |
| 
 | |
|     >>> a, b, c, d = Seq()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: not enough values to unpack (expected 4, got 3)
 | |
| 
 | |
| Unpacking sequence too long
 | |
| 
 | |
|     >>> a, b = Seq()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 2)
 | |
| 
 | |
| Unpacking a sequence where the test for too long raises a different kind of
 | |
| error
 | |
| 
 | |
|     >>> class BozoError(Exception):
 | |
|     ...     pass
 | |
|     ...
 | |
|     >>> class BadSeq:
 | |
|     ...     def __getitem__(self, i):
 | |
|     ...         if i >= 0 and i < 3:
 | |
|     ...             return i
 | |
|     ...         elif i == 3:
 | |
|     ...             raise BozoError
 | |
|     ...         else:
 | |
|     ...             raise IndexError
 | |
|     ...
 | |
| 
 | |
| Trigger code while not expecting an IndexError (unpack sequence too long, wrong
 | |
| error)
 | |
| 
 | |
|     >>> a, b, c, d, e = BadSeq()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     test.test_unpack.BozoError
 | |
| 
 | |
| Trigger code while expecting an IndexError (unpack sequence too short, wrong
 | |
| error)
 | |
| 
 | |
|     >>> a, b, c = BadSeq()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     test.test_unpack.BozoError
 | |
| 
 | |
| Allow unpacking empty iterables
 | |
| 
 | |
|     >>> () = []
 | |
|     >>> [] = ()
 | |
|     >>> [] = []
 | |
|     >>> () = ()
 | |
| 
 | |
| Unpacking non-iterables should raise TypeError
 | |
| 
 | |
|     >>> () = 42
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     TypeError: cannot unpack non-iterable int object
 | |
| 
 | |
| Unpacking to an empty iterable should raise ValueError
 | |
| 
 | |
|     >>> () = [42]
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 0, got 1)
 | |
| 
 | |
| Unpacking a larger iterable should raise ValuleError, but it
 | |
| should not entirely consume the iterable
 | |
| 
 | |
|     >>> it = iter(range(100))
 | |
|     >>> x, y, z = it
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 3)
 | |
|     >>> next(it)
 | |
|     4
 | |
| 
 | |
| Unpacking unbalanced dict
 | |
| 
 | |
|     >>> d = {4: 'four', 5: 'five', 6: 'six', 7: 'seven'}
 | |
|     >>> a, b, c = d
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 3, got 4)
 | |
| 
 | |
| Ensure that custom `__len__()` is NOT called when showing the error message
 | |
| 
 | |
|     >>> class LengthTooLong:
 | |
|     ...     def __len__(self):
 | |
|     ...         return 5
 | |
|     ...     def __getitem__(self, i):
 | |
|     ...         return i*2
 | |
|     ...
 | |
|     >>> x, y, z = LengthTooLong()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 3)
 | |
| 
 | |
| For evil cases like these as well, no actual count to be shown
 | |
| 
 | |
|     >>> class BadLength:
 | |
|     ...     def __len__(self):
 | |
|     ...         return 1
 | |
|     ...     def __getitem__(self, i):
 | |
|     ...         return i*2
 | |
|     ...
 | |
|     >>> x, y, z = BadLength()
 | |
|     Traceback (most recent call last):
 | |
|       ...
 | |
|     ValueError: too many values to unpack (expected 3)
 | |
| """
 | |
| 
 | |
| __test__ = {'doctests' : doctests}
 | |
| 
 | |
| def load_tests(loader, tests, pattern):
 | |
|     tests.addTest(doctest.DocTestSuite())
 | |
|     return tests
 | |
| 
 | |
| 
 | |
| class TestCornerCases(unittest.TestCase):
 | |
|     def test_extended_oparg_not_ignored(self):
 | |
|         # https://github.com/python/cpython/issues/91625
 | |
|         target = "(" + "y,"*400 + ")"
 | |
|         code = f"""def unpack_400(x):
 | |
|             {target} = x
 | |
|             return y
 | |
|         """
 | |
|         ns = {}
 | |
|         exec(code, ns)
 | |
|         unpack_400 = ns["unpack_400"]
 | |
|         # Warm up the function for quickening (PEP 659)
 | |
|         for _ in range(30):
 | |
|             y = unpack_400(range(400))
 | |
|             self.assertEqual(y, 399)
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     unittest.main()
 | 
