From 942b45d639c57c1049355cb66ad8cf795bfa0728 Mon Sep 17 00:00:00 2001 From: Scott Sanderson Date: Tue, 17 Nov 2015 21:53:16 -0500 Subject: [PATCH] ENH: Implement support for with-blocks. The basic idea here is rewrite the ast of the with-block into two statements and try-except, which we already know how to oneline-ify In general, we can rewrite a with block of the form: with as : as __anonymous = = __anonymous.__enter__() try: except: if not __anonymous.__exit__(*sys.exc_info()): raise else: __anonymous.__exit__(None, None, None) --- main.py | 134 ++++++++++++++++++++++++++++++++- tests/with.py | 16 ++++ tests/with_raise.py | 15 ++++ tests/with_suppressed_raise.py | 19 +++++ 4 files changed, 181 insertions(+), 3 deletions(-) create mode 100644 tests/with.py create mode 100644 tests/with_raise.py create mode 100644 tests/with_suppressed_raise.py diff --git a/main.py b/main.py index 8cba1ef..9b6e744 100644 --- a/main.py +++ b/main.py @@ -165,7 +165,10 @@ def mangle(self, name): def var(self, name): name = self.mangle(name) - sym = self.table.lookup(name) + try: + sym = self.table.lookup(name) + except KeyError: + return name if sym.is_global() or (self.table.get_type() == 'module' and sym.is_local()): return T('{}').format(name) elif sym.is_local(): @@ -177,7 +180,10 @@ def var(self, name): def store_var(self, name): name = self.mangle(name) - sym = self.table.lookup(name) + try: + sym = self.table.lookup(name) + except KeyError: + return name if sym.is_global(): return T('{__g}[{!r}]').format(name) elif sym.is_local(): @@ -727,7 +733,129 @@ def visit_While(self, tree): test, orelse)) def visit_With(self, tree): - raise NotImplementedError('Open problem: with') + + # Rewrite with-blocks as try-except statements as follows: + # with as + # + # ---------------------- + # __anonymous = + # bar = __anonymous.__enter__() + # try: + # + # except: + # if __anonymous.__exit__(*sys.exc_info()): + # raise + # else: + # __anonymous.__exit__(None, None, None) + if tree.optional_vars is not None: + ctx_bind_name = tree.optional_vars.id + else: + ctx_bind_name = '__anonymous__enter__result' + + ctx_manager_name = '__anonymous' + + manager_assign = ast.Assign( + targets=[ast.Name(id=ctx_manager_name, ctx=ast.Store())], + value=tree.context_expr, + ) + context_expr_assign = ast.Assign( + targets=[ast.Name(id=ctx_bind_name, ctx=ast.Store())], + value=ast.Call( + func=ast.Attribute( + value=ast.Name(id=ctx_manager_name, ctx=ast.Store()), + attr='__enter__', + ctx=ast.Load(), + ), + args=[], + keywords=[], + starargs=None, + kwargs=None, + ) + ) + # Rewrite the with-block body as: + # try: + # + # except: + # if not .__exit__(*sys.exc_info()): + # raise + sys_module = ast.Call( + func=ast.Name(id='__import__', ctx=ast.Load()), + args=[ + ast.Str(s='sys'), + ], + keywords=[], + starargs=None, + kwargs=None, + ) + none_node = ast.Name(id='None', ctx=ast.Load()) + + block = ast.TryExcept( + body=tree.body, + handlers=[ + ast.ExceptHandler( + type=None, + name=None, + body=[ + ast.If( + test=ast.UnaryOp( + op=ast.Not(), + operand=ast.Call( + func=ast.Attribute( + value=ast.Name( + id=ctx_manager_name, + ctx=ast.Load(), + ), + attr='__exit__', + ctx=ast.Load(), + ), + args=[], + keywords=[], + starargs=ast.Call( + func=ast.Attribute( + value=sys_module, + attr='exc_info', + ctx=ast.Load(), + ), + args=[], + keywords=[], + starargs=None, + kwargs=None, + ), + kwargs=None, + ), + ), + body=[ + ast.Raise( + type=None, + inst=None, + tback=None, + ), + ], + orelse=[], + ), + ], + ), + ], + orelse=[ + ast.Expr( + value=ast.Call( + func=ast.Attribute( + value=ast.Name( + id=ctx_manager_name, + ctx=ast.Load(), + ), + attr='__exit__', + ctx=ast.Load(), + ), + args=[none_node, none_node, none_node], + keywords=[], + starargs=None, + kwargs=None, + ), + ), + ], + ) + return self.many_to_one([manager_assign, context_expr_assign, block]) def visit_Yield(self, tree): raise NotImplementedError('Open problem: yield') diff --git a/tests/with.py b/tests/with.py new file mode 100644 index 0000000..7f8b223 --- /dev/null +++ b/tests/with.py @@ -0,0 +1,16 @@ +class ContextManager(object): + + def __enter__(self): + print "Entering" + return "__enter__ value" + + def __exit__(self, type, value, traceback): + print "Exiting" + +print "Before" +with ContextManager() as c: + print "In Body" + print c + +# This currently fails for reasons I don't quite understand. +# print "After" diff --git a/tests/with_raise.py b/tests/with_raise.py new file mode 100644 index 0000000..490a093 --- /dev/null +++ b/tests/with_raise.py @@ -0,0 +1,15 @@ +class ContextManager(object): + + def __enter__(self): + print "Entering" + + def __exit__(self, type, value, traceback): + print "Exiting" + + +print "Before" +with ContextManager() as c: + print "In Body" + raise ValueError("Raise") + +print "After" diff --git a/tests/with_suppressed_raise.py b/tests/with_suppressed_raise.py new file mode 100644 index 0000000..cd02414 --- /dev/null +++ b/tests/with_suppressed_raise.py @@ -0,0 +1,19 @@ +class ContextManager(object): + + def __enter__(self): + print "Entering" + return "__enter__ value" + + def __exit__(self, type, value, traceback): + print "Exiting" + # Suppress the exception. + return True + + +print "Before" +with ContextManager() as c: + print "Before Raise" + raise ValueError("Raise") + +# This currently fails for reasons I don't quite understand. +# print "After"