From 772927c53d9fa71fef52a0ec91e3b1a487578939 Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Wed, 8 May 2013 18:49:07 +0200 Subject: [PATCH] Drop the bare names that were sprinkled everywhere in the generated AST. This closes #162 We make sure not to drop *explicitly written* bare names. --- hy/compiler.py | 6 +++++- tests/compilers/test_compiler.py | 23 +++++++++++++++++++++++ 2 files changed, 28 insertions(+), 1 deletion(-) diff --git a/hy/compiler.py b/hy/compiler.py index dd6af76..4ec6eb0 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -193,9 +193,13 @@ class Result(object): This is useful when we want to use the stored expression in a statement context (for instance in a code branch). + We drop ast.Names if they are appended to statements, as they + can't have any side effect. "Bare" names still get converted to + statements. + If there is no expression context, return an empty result. """ - if self.expr: + if self.expr and not (isinstance(self.expr, ast.Name) and self.stmts): return Result() + ast.Expr(lineno=self.expr.lineno, col_offset=self.expr.col_offset, value=self.expr) diff --git a/tests/compilers/test_compiler.py b/tests/compilers/test_compiler.py index b5b3846..9291ffd 100644 --- a/tests/compilers/test_compiler.py +++ b/tests/compilers/test_compiler.py @@ -1,4 +1,5 @@ # Copyright (c) 2013 Julien Danjou +# Copyright (c) 2013 Nicolas Dandrimont # # Permission is hereby granted, free of charge, to any person obtaining a # copy of this software and associated documentation files (the "Software"), @@ -29,6 +30,7 @@ else: from hy.models.expression import HyExpression from hy.models.list import HyList +from hy.models.symbol import HySymbol from hy.compiler import HyASTCompiler @@ -63,3 +65,24 @@ class HyASTCompilerTest(unittest.TestCase): self.assertIsInstance(stmt.body[0], ast.Pass) self.assertIsInstance(ret.expr, ast.Name) + + def test_compiler_bare_names(self): + """Check that the compiler doesn't drop bare names from code branches""" + ret = self.c.compile(self._make_expression(HySymbol("do"), + HySymbol("a"), + HySymbol("b"), + HySymbol("c"))) + + # We expect two statements and a final expr. + self.assertEqual(len(ret.stmts), 2) + stmt = ret.stmts[0] + self.assertIsInstance(stmt, ast.Expr) + self.assertIsInstance(stmt.value, ast.Name) + self.assertEqual(stmt.value.id, "a") + stmt = ret.stmts[1] + self.assertIsInstance(stmt, ast.Expr) + self.assertIsInstance(stmt.value, ast.Name) + self.assertEqual(stmt.value.id, "b") + expr = ret.expr + self.assertIsInstance(expr, ast.Name) + self.assertEqual(expr.id, "c")