| doctests = """ | 
 |  | 
 | Test simple loop with conditional | 
 |  | 
 |     >>> sum(i*i for i in range(100) if i&1 == 1) | 
 |     166650 | 
 |  | 
 | Test simple nesting | 
 |  | 
 |     >>> list((i,j) for i in range(3) for j in range(4) ) | 
 |     [(0, 0), (0, 1), (0, 2), (0, 3), (1, 0), (1, 1), (1, 2), (1, 3), (2, 0), (2, 1), (2, 2), (2, 3)] | 
 |  | 
 | Test nesting with the inner expression dependent on the outer | 
 |  | 
 |     >>> list((i,j) for i in range(4) for j in range(i) ) | 
 |     [(1, 0), (2, 0), (2, 1), (3, 0), (3, 1), (3, 2)] | 
 |  | 
 | Make sure the induction variable is not exposed | 
 |  | 
 |     >>> i = 20 | 
 |     >>> sum(i*i for i in range(100)) | 
 |     328350 | 
 |     >>> i | 
 |     20 | 
 |  | 
 | Test first class | 
 |  | 
 |     >>> g = (i*i for i in range(4)) | 
 |     >>> type(g) | 
 |     <type 'generator'> | 
 |     >>> list(g) | 
 |     [0, 1, 4, 9] | 
 |  | 
 | Test direct calls to next() | 
 |  | 
 |     >>> g = (i*i for i in range(3)) | 
 |     >>> g.next() | 
 |     0 | 
 |     >>> g.next() | 
 |     1 | 
 |     >>> g.next() | 
 |     4 | 
 |     >>> g.next() | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#21>", line 1, in -toplevel- | 
 |         g.next() | 
 |     StopIteration | 
 |  | 
 | Does it stay stopped? | 
 |  | 
 |     >>> g.next() | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#21>", line 1, in -toplevel- | 
 |         g.next() | 
 |     StopIteration | 
 |     >>> list(g) | 
 |     [] | 
 |  | 
 | Test running gen when defining function is out of scope | 
 |  | 
 |     >>> def f(n): | 
 |     ...     return (i*i for i in xrange(n)) | 
 |     >>> list(f(10)) | 
 |     [0, 1, 4, 9, 16, 25, 36, 49, 64, 81] | 
 |  | 
 |     >>> def f(n): | 
 |     ...     return ((i,j) for i in xrange(3) for j in xrange(n)) | 
 |     >>> list(f(4)) | 
 |     [(0, 0), (0, 1), (0, 2), (0, 3), (1, 0), (1, 1), (1, 2), (1, 3), (2, 0), (2, 1), (2, 2), (2, 3)] | 
 |     >>> def f(n): | 
 |     ...     return ((i,j) for i in xrange(3) for j in xrange(4) if j in xrange(n)) | 
 |     >>> list(f(4)) | 
 |     [(0, 0), (0, 1), (0, 2), (0, 3), (1, 0), (1, 1), (1, 2), (1, 3), (2, 0), (2, 1), (2, 2), (2, 3)] | 
 |     >>> list(f(2)) | 
 |     [(0, 0), (0, 1), (1, 0), (1, 1), (2, 0), (2, 1)] | 
 |  | 
 | Verify that parenthesis are required in a statement | 
 |  | 
 |     >>> def f(n): | 
 |     ...     return i*i for i in xrange(n) | 
 |     Traceback (most recent call last): | 
 |        ... | 
 |     SyntaxError: invalid syntax | 
 |  | 
 | Verify that parenthesis are required when used as a keyword argument value | 
 |  | 
 |     >>> dict(a = i for i in xrange(10)) | 
 |     Traceback (most recent call last): | 
 |        ... | 
 |     SyntaxError: invalid syntax | 
 |  | 
 | Verify that parenthesis are required when used as a keyword argument value | 
 |  | 
 |     >>> dict(a = (i for i in xrange(10))) #doctest: +ELLIPSIS | 
 |     {'a': <generator object <genexpr> at ...>} | 
 |  | 
 | Verify early binding for the outermost for-expression | 
 |  | 
 |     >>> x=10 | 
 |     >>> g = (i*i for i in range(x)) | 
 |     >>> x = 5 | 
 |     >>> list(g) | 
 |     [0, 1, 4, 9, 16, 25, 36, 49, 64, 81] | 
 |  | 
 | Verify that the outermost for-expression makes an immediate check | 
 | for iterability | 
 |  | 
 |     >>> (i for i in 6) | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#4>", line 1, in -toplevel- | 
 |         (i for i in 6) | 
 |     TypeError: 'int' object is not iterable | 
 |  | 
 | Verify late binding for the outermost if-expression | 
 |  | 
 |     >>> include = (2,4,6,8) | 
 |     >>> g = (i*i for i in range(10) if i in include) | 
 |     >>> include = (1,3,5,7,9) | 
 |     >>> list(g) | 
 |     [1, 9, 25, 49, 81] | 
 |  | 
 | Verify late binding for the innermost for-expression | 
 |  | 
 |     >>> g = ((i,j) for i in range(3) for j in range(x)) | 
 |     >>> x = 4 | 
 |     >>> list(g) | 
 |     [(0, 0), (0, 1), (0, 2), (0, 3), (1, 0), (1, 1), (1, 2), (1, 3), (2, 0), (2, 1), (2, 2), (2, 3)] | 
 |  | 
 | Verify re-use of tuples (a side benefit of using genexps over listcomps) | 
 |  | 
 |     >>> tupleids = map(id, ((i,i) for i in xrange(10))) | 
 |     >>> int(max(tupleids) - min(tupleids)) | 
 |     0 | 
 |  | 
 | Verify that syntax error's are raised for genexps used as lvalues | 
 |  | 
 |     >>> (y for y in (1,2)) = 10 | 
 |     Traceback (most recent call last): | 
 |        ... | 
 |       File "<doctest test.test_genexps.__test__.doctests[40]>", line 1 | 
 |     SyntaxError: can't assign to generator expression | 
 |  | 
 |     >>> (y for y in (1,2)) += 10 | 
 |     Traceback (most recent call last): | 
 |        ... | 
 |       File "<doctest test.test_genexps.__test__.doctests[41]>", line 1 | 
 |     SyntaxError: can't assign to generator expression | 
 |  | 
 |  | 
 | ########### Tests borrowed from or inspired by test_generators.py ############ | 
 |  | 
 | Make a generator that acts like range() | 
 |  | 
 |     >>> yrange = lambda n:  (i for i in xrange(n)) | 
 |     >>> list(yrange(10)) | 
 |     [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] | 
 |  | 
 | Generators always return to the most recent caller: | 
 |  | 
 |     >>> def creator(): | 
 |     ...     r = yrange(5) | 
 |     ...     print "creator", r.next() | 
 |     ...     return r | 
 |     >>> def caller(): | 
 |     ...     r = creator() | 
 |     ...     for i in r: | 
 |     ...             print "caller", i | 
 |     >>> caller() | 
 |     creator 0 | 
 |     caller 1 | 
 |     caller 2 | 
 |     caller 3 | 
 |     caller 4 | 
 |  | 
 | Generators can call other generators: | 
 |  | 
 |     >>> def zrange(n): | 
 |     ...     for i in yrange(n): | 
 |     ...         yield i | 
 |     >>> list(zrange(5)) | 
 |     [0, 1, 2, 3, 4] | 
 |  | 
 |  | 
 | Verify that a gen exp cannot be resumed while it is actively running: | 
 |  | 
 |     >>> g = (me.next() for i in xrange(10)) | 
 |     >>> me = g | 
 |     >>> me.next() | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#30>", line 1, in -toplevel- | 
 |         me.next() | 
 |       File "<pyshell#28>", line 1, in <generator expression> | 
 |         g = (me.next() for i in xrange(10)) | 
 |     ValueError: generator already executing | 
 |  | 
 | Verify exception propagation | 
 |  | 
 |     >>> g = (10 // i for i in (5, 0, 2)) | 
 |     >>> g.next() | 
 |     2 | 
 |     >>> g.next() | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#37>", line 1, in -toplevel- | 
 |         g.next() | 
 |       File "<pyshell#35>", line 1, in <generator expression> | 
 |         g = (10 // i for i in (5, 0, 2)) | 
 |     ZeroDivisionError: integer division or modulo by zero | 
 |     >>> g.next() | 
 |     Traceback (most recent call last): | 
 |       File "<pyshell#38>", line 1, in -toplevel- | 
 |         g.next() | 
 |     StopIteration | 
 |  | 
 | Make sure that None is a valid return value | 
 |  | 
 |     >>> list(None for i in xrange(10)) | 
 |     [None, None, None, None, None, None, None, None, None, None] | 
 |  | 
 | Check that generator attributes are present | 
 |  | 
 |     >>> g = (i*i for i in range(3)) | 
 |     >>> expected = set(['gi_frame', 'gi_running', 'next']) | 
 |     >>> set(attr for attr in dir(g) if not attr.startswith('__')) >= expected | 
 |     True | 
 |  | 
 |     >>> print g.next.__doc__ | 
 |     x.next() -> the next value, or raise StopIteration | 
 |     >>> import types | 
 |     >>> isinstance(g, types.GeneratorType) | 
 |     True | 
 |  | 
 | Check the __iter__ slot is defined to return self | 
 |  | 
 |     >>> iter(g) is g | 
 |     True | 
 |  | 
 | Verify that the running flag is set properly | 
 |  | 
 |     >>> g = (me.gi_running for i in (0,1)) | 
 |     >>> me = g | 
 |     >>> me.gi_running | 
 |     0 | 
 |     >>> me.next() | 
 |     1 | 
 |     >>> me.gi_running | 
 |     0 | 
 |  | 
 | Verify that genexps are weakly referencable | 
 |  | 
 |     >>> import weakref | 
 |     >>> g = (i*i for i in range(4)) | 
 |     >>> wr = weakref.ref(g) | 
 |     >>> wr() is g | 
 |     True | 
 |     >>> p = weakref.proxy(g) | 
 |     >>> list(p) | 
 |     [0, 1, 4, 9] | 
 |  | 
 |  | 
 | """ | 
 |  | 
 |  | 
 | __test__ = {'doctests' : doctests} | 
 |  | 
 | def test_main(verbose=None): | 
 |     import sys | 
 |     from test import test_support | 
 |     from test import test_genexps | 
 |     test_support.run_doctest(test_genexps, verbose) | 
 |  | 
 |     # verify reference counting | 
 |     if verbose and hasattr(sys, "gettotalrefcount"): | 
 |         import gc | 
 |         counts = [None] * 5 | 
 |         for i in xrange(len(counts)): | 
 |             test_support.run_doctest(test_genexps, verbose) | 
 |             gc.collect() | 
 |             counts[i] = sys.gettotalrefcount() | 
 |         print counts | 
 |  | 
 | if __name__ == "__main__": | 
 |     test_main(verbose=True) |